diff options
Diffstat (limited to '')
-rw-r--r-- | toolkit/crashreporter/test/gtest/TestCrashThreadAnnotation.cpp | 362 | ||||
-rw-r--r-- | toolkit/crashreporter/test/gtest/moz.build | 15 |
2 files changed, 377 insertions, 0 deletions
diff --git a/toolkit/crashreporter/test/gtest/TestCrashThreadAnnotation.cpp b/toolkit/crashreporter/test/gtest/TestCrashThreadAnnotation.cpp new file mode 100644 index 0000000000..78362e2b5e --- /dev/null +++ b/toolkit/crashreporter/test/gtest/TestCrashThreadAnnotation.cpp @@ -0,0 +1,362 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#include "ThreadAnnotation.h" + +#include <string.h> + +#include "base/thread.h" +#include "gtest/gtest.h" +#include "mozilla/Monitor.h" +#include "mozilla/UniquePtr.h" +#include "mozilla/Unused.h" +#include "nsIThread.h" +#include "nsIRunnable.h" +#include "nsThreadUtils.h" + +using mozilla::Monitor; +using mozilla::MonitorAutoLock; +using mozilla::UniquePtr; + +namespace CrashReporter { +namespace { + +TEST(TestCrashThreadAnnotation, TestInitShutdown) +{ + InitThreadAnnotation(); + ShutdownThreadAnnotation(); +} + +TEST(TestCrashThreadAnnotation, TestNestedInitShutdown) +{ + // No bad things should happen in case we have extra init/shutdown calls. + InitThreadAnnotation(); + InitThreadAnnotation(); + ShutdownThreadAnnotation(); + ShutdownThreadAnnotation(); +} + +TEST(TestCrashThreadAnnotation, TestUnbalancedInit) +{ + // No bad things should happen in case we have unbalanced init/shutdown calls. + InitThreadAnnotation(); + InitThreadAnnotation(); + ShutdownThreadAnnotation(); +} + +TEST(TestCrashThreadAnnotation, TestUnbalancedShutdown) +{ + // No bad things should happen in case we have unbalanced init/shutdown calls. + InitThreadAnnotation(); + ShutdownThreadAnnotation(); + ShutdownThreadAnnotation(); +} + +TEST(TestCrashThreadAnnotation, TestGetFlatThreadAnnotation_BeforeInit) +{ + // GetFlatThreadAnnotation() should not return anything before init. + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { ASSERT_STREQ(aAnnotation, ""); }; + GetFlatThreadAnnotation(getThreadAnnotationCB); +} + +TEST(TestCrashThreadAnnotation, TestGetFlatThreadAnnotation_AfterShutdown) +{ + // GetFlatThreadAnnotation() should not return anything after shutdown. + InitThreadAnnotation(); + ShutdownThreadAnnotation(); + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { ASSERT_STREQ(aAnnotation, ""); }; + GetFlatThreadAnnotation(getThreadAnnotationCB); +} + +already_AddRefed<nsIThread> CreateTestThread(const char* aName, + Monitor& aMonitor, bool& aDone) { + nsCOMPtr<nsIRunnable> setNameRunnable = NS_NewRunnableFunction( + "CrashReporter::CreateTestThread", [aName, &aMonitor, &aDone]() -> void { + NS_SetCurrentThreadName(aName); + + MonitorAutoLock lock(aMonitor); + aDone = true; + aMonitor.NotifyAll(); + }); + nsCOMPtr<nsIThread> thread; + mozilla::Unused << NS_NewNamedThread("Test Thread", getter_AddRefs(thread), + setNameRunnable); + + return thread.forget(); +} + +TEST(TestCrashThreadAnnotation, TestGetFlatThreadAnnotation_OneThread) +{ + InitThreadAnnotation(); + + Monitor monitor("TestCrashThreadAnnotation"); + bool threadNameSet = false; + nsCOMPtr<nsIThread> thread = + CreateTestThread("Thread1", monitor, threadNameSet); + ASSERT_TRUE(!!thread); + + { + MonitorAutoLock lock(monitor); + while (!threadNameSet) { + monitor.Wait(); + } + } + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { + ASSERT_TRUE(!!strstr(aAnnotation, "Thread1")); + }; + GetFlatThreadAnnotation(getThreadAnnotationCB); + + ShutdownThreadAnnotation(); + + thread->Shutdown(); +} + +TEST(TestCrashThreadAnnotation, TestGetFlatThreadAnnotation_SetNameTwice) +{ + InitThreadAnnotation(); + + Monitor monitor("TestCrashThreadAnnotation"); + bool threadNameSet = false; + + nsCOMPtr<nsIRunnable> setNameRunnable = NS_NewRunnableFunction( + "CrashReporter::TestCrashThreadAnnotation_TestGetFlatThreadAnnotation_" + "SetNameTwice_Test::TestBody", + [&]() -> void { + NS_SetCurrentThreadName("Thread1"); + // Set the name again. We should get the latest name. + NS_SetCurrentThreadName("Thread1Again"); + + MonitorAutoLock lock(monitor); + threadNameSet = true; + monitor.NotifyAll(); + }); + nsCOMPtr<nsIThread> thread; + nsresult rv = + NS_NewNamedThread("Test Thread", getter_AddRefs(thread), setNameRunnable); + ASSERT_TRUE(NS_SUCCEEDED(rv)); + + { + MonitorAutoLock lock(monitor); + while (!threadNameSet) { + monitor.Wait(); + } + } + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { + ASSERT_TRUE(!!strstr(aAnnotation, "Thread1Again")); + }; + GetFlatThreadAnnotation(getThreadAnnotationCB); + + ShutdownThreadAnnotation(); + + thread->Shutdown(); +} + +TEST(TestCrashThreadAnnotation, TestGetFlatThreadAnnotation_TwoThreads) +{ + InitThreadAnnotation(); + + Monitor monitor("TestCrashThreadAnnotation"); + bool thread1NameSet = false; + bool thread2NameSet = false; + + nsCOMPtr<nsIThread> thread1 = + CreateTestThread("Thread1", monitor, thread1NameSet); + ASSERT_TRUE(!!thread1); + + nsCOMPtr<nsIThread> thread2 = + CreateTestThread("Thread2", monitor, thread2NameSet); + ASSERT_TRUE(!!thread2); + + { + MonitorAutoLock lock(monitor); + while (!(thread1NameSet && thread2NameSet)) { + monitor.Wait(); + } + } + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { + // Assert that Thread1 and Thread2 are both in the annotation data. + ASSERT_TRUE(!!strstr(aAnnotation, "Thread1")); + ASSERT_TRUE(!!strstr(aAnnotation, "Thread2")); + }; + GetFlatThreadAnnotation(getThreadAnnotationCB); + + ShutdownThreadAnnotation(); + + thread1->Shutdown(); + thread2->Shutdown(); +} + +TEST(TestCrashThreadAnnotation, TestGetFlatThreadAnnotation_ShutdownOneThread) +{ + InitThreadAnnotation(); + + Monitor monitor("TestCrashThreadAnnotation"); + bool thread1NameSet = false; + bool thread2NameSet = false; + + nsCOMPtr<nsIThread> thread1 = + CreateTestThread("Thread1", monitor, thread1NameSet); + ASSERT_TRUE(!!thread1); + + nsCOMPtr<nsIThread> thread2 = + CreateTestThread("Thread2", monitor, thread2NameSet); + ASSERT_TRUE(!!thread2); + + { + MonitorAutoLock lock(monitor); + while (!(thread1NameSet && thread2NameSet)) { + monitor.Wait(); + } + } + + nsresult rv = thread1->Shutdown(); + ASSERT_TRUE(NS_SUCCEEDED(rv)); + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { + // Assert that only Thread2 is present in the annotation data. + ASSERT_TRUE(!strstr(aAnnotation, "Thread1")); + ASSERT_TRUE(!!strstr(aAnnotation, "Thread2")); + }; + GetFlatThreadAnnotation(getThreadAnnotationCB); + + ShutdownThreadAnnotation(); + + thread2->Shutdown(); +} + +TEST(TestCrashThreadAnnotation, TestGetFlatThreadAnnotation_ShutdownBothThreads) +{ + InitThreadAnnotation(); + + Monitor monitor("TestCrashThreadAnnotation"); + bool thread1NameSet = false; + bool thread2NameSet = false; + + nsCOMPtr<nsIThread> thread1 = + CreateTestThread("Thread1", monitor, thread1NameSet); + ASSERT_TRUE(!!thread1); + + nsCOMPtr<nsIThread> thread2 = + CreateTestThread("Thread2", monitor, thread2NameSet); + ASSERT_TRUE(!!thread2); + + { + MonitorAutoLock lock(monitor); + while (!(thread1NameSet && thread2NameSet)) { + monitor.Wait(); + } + } + + nsresult rv = thread1->Shutdown(); + ASSERT_TRUE(NS_SUCCEEDED(rv)); + + rv = thread2->Shutdown(); + ASSERT_TRUE(NS_SUCCEEDED(rv)); + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { + // No leftover in annnotation data. + ASSERT_STREQ(aAnnotation, ""); + }; + GetFlatThreadAnnotation(getThreadAnnotationCB); + + ShutdownThreadAnnotation(); +} + +TEST(TestCrashThreadAnnotation, + TestGetFlatThreadAnnotation_TestNameOfBaseThread) +{ + InitThreadAnnotation(); + + Monitor monitor("TestCrashThreadAnnotation"); + + UniquePtr<base::Thread> thread1 = + mozilla::MakeUnique<base::Thread>("base thread 1"); + ASSERT_TRUE(!!thread1 && thread1->Start()); + + UniquePtr<base::Thread> thread2 = + mozilla::MakeUnique<base::Thread>("base thread 2"); + ASSERT_TRUE(!!thread2 && thread2->Start()); + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { + ASSERT_TRUE(!!strstr(aAnnotation, "base thread 1")); + ASSERT_TRUE(!!strstr(aAnnotation, "base thread 2")); + }; + GetFlatThreadAnnotation(getThreadAnnotationCB); + + thread1->Stop(); + thread2->Stop(); + + ShutdownThreadAnnotation(); +} + +TEST(TestCrashThreadAnnotation, + TestGetFlatThreadAnnotation_TestShutdownBaseThread) +{ + InitThreadAnnotation(); + + Monitor monitor("TestCrashThreadAnnotation"); + + UniquePtr<base::Thread> thread1 = + mozilla::MakeUnique<base::Thread>("base thread 1"); + ASSERT_TRUE(!!thread1 && thread1->Start()); + + UniquePtr<base::Thread> thread2 = + mozilla::MakeUnique<base::Thread>("base thread 2"); + ASSERT_TRUE(!!thread2 && thread2->Start()); + + thread1->Stop(); + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { + ASSERT_TRUE(!strstr(aAnnotation, "base thread 1")); + ASSERT_TRUE(!!strstr(aAnnotation, "base thread 2")); + }; + GetFlatThreadAnnotation(getThreadAnnotationCB); + + thread2->Stop(); + + ShutdownThreadAnnotation(); +} + +TEST(TestCrashThreadAnnotation, + TestGetFlatThreadAnnotation_TestShutdownBothBaseThreads) +{ + InitThreadAnnotation(); + + Monitor monitor("TestCrashThreadAnnotation"); + + UniquePtr<base::Thread> thread1 = + mozilla::MakeUnique<base::Thread>("base thread 1"); + ASSERT_TRUE(!!thread1 && thread1->Start()); + + UniquePtr<base::Thread> thread2 = + mozilla::MakeUnique<base::Thread>("base thread 2"); + ASSERT_TRUE(!!thread2 && thread2->Start()); + + thread1->Stop(); + thread2->Stop(); + + std::function<void(const char*)> getThreadAnnotationCB = + [&](const char* aAnnotation) -> void { + ASSERT_TRUE(!strlen(aAnnotation)); + }; + GetFlatThreadAnnotation(getThreadAnnotationCB); + + ShutdownThreadAnnotation(); +} + +} // Anonymous namespace. +} // namespace CrashReporter diff --git a/toolkit/crashreporter/test/gtest/moz.build b/toolkit/crashreporter/test/gtest/moz.build new file mode 100644 index 0000000000..576a53d3ac --- /dev/null +++ b/toolkit/crashreporter/test/gtest/moz.build @@ -0,0 +1,15 @@ +# -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*- +# vim: set filetype=python: +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. + +UNIFIED_SOURCES += [ + "TestCrashThreadAnnotation.cpp", +] + +include("/ipc/chromium/chromium-config.mozbuild") + +LOCAL_INCLUDES += ["../../"] + +FINAL_LIBRARY = "xul-gtest" |