Index: src/client/linux/handler/exception_handler_unittest.cc |
=================================================================== |
--- a/src/client/linux/handler/exception_handler_unittest.cc |
+++ b/src/client/linux/handler/exception_handler_unittest.cc |
@@ -890,16 +890,35 @@ TEST(ExceptionHandlerTest, ExternalDumpe |
ASSERT_NO_FATAL_FAILURE(WaitForProcessToTerminate(child, SIGSEGV)); |
struct stat st; |
ASSERT_EQ(0, stat(templ.c_str(), &st)); |
ASSERT_GT(st.st_size, 0U); |
unlink(templ.c_str()); |
} |
+TEST(ExceptionHandlerTest, WriteMinidumpExceptionStream) { |
+ AutoTempDir temp_dir; |
+ ExceptionHandler handler(MinidumpDescriptor(temp_dir.path()), NULL, NULL, |
+ NULL, false, -1); |
+ ASSERT_TRUE(handler.WriteMinidump()); |
+ |
+ string minidump_path = handler.minidump_descriptor().path(); |
+ |
+ // Read the minidump and check the exception stream. |
+ Minidump minidump(minidump_path); |
+ ASSERT_TRUE(minidump.Read()); |
+ MinidumpException* exception = minidump.GetException(); |
+ ASSERT_TRUE(exception); |
+ const MDRawExceptionStream* raw = exception->exception(); |
+ ASSERT_TRUE(raw); |
+ EXPECT_EQ(MD_EXCEPTION_CODE_LIN_DUMP_REQUESTED, |
+ raw->exception_record.exception_code); |
+} |
+ |
TEST(ExceptionHandlerTest, GenerateMultipleDumpsWithFD) { |
AutoTempDir temp_dir; |
std::string path; |
const int fd = CreateTMPFile(temp_dir.path(), &path); |
ExceptionHandler handler(MinidumpDescriptor(fd), NULL, NULL, NULL, false, -1); |
ASSERT_TRUE(handler.WriteMinidump()); |
// Check by the size of the data written to the FD that a minidump was |
// generated. |
@@ -1016,8 +1035,55 @@ TEST(ExceptionHandlerTest, AdditionalMem |
MinidumpMemoryList* dump_memory_list = minidump.GetMemoryList(); |
ASSERT_TRUE(dump_memory_list); |
const MinidumpMemoryRegion* region = |
dump_memory_list->GetMemoryRegionForAddress(kMemoryAddress); |
EXPECT_FALSE(region); |
delete[] memory; |
} |
+ |
+static bool SimpleCallback(const MinidumpDescriptor& descriptor, |
+ void* context, |
+ bool succeeded) { |
+ string* filename = reinterpret_cast<string*>(context); |
+ *filename = descriptor.path(); |
+ return true; |
+} |
+ |
+TEST(ExceptionHandlerTest, WriteMinidumpForChild) { |
+ int fds[2]; |
+ ASSERT_NE(-1, pipe(fds)); |
+ |
+ const pid_t child = fork(); |
+ if (child == 0) { |
+ close(fds[1]); |
+ char b; |
+ HANDLE_EINTR(read(fds[0], &b, sizeof(b))); |
+ close(fds[0]); |
+ syscall(__NR_exit); |
+ } |
+ close(fds[0]); |
+ |
+ AutoTempDir temp_dir; |
+ string minidump_filename; |
+ ASSERT_TRUE( |
+ ExceptionHandler::WriteMinidumpForChild(child, child, |
+ temp_dir.path(), SimpleCallback, |
+ (void*)&minidump_filename)); |
+ |
+ Minidump minidump(minidump_filename); |
+ ASSERT_TRUE(minidump.Read()); |
+ // Check that the crashing thread is the main thread of |child| |
+ MinidumpException* exception = minidump.GetException(); |
+ ASSERT_TRUE(exception); |
+ u_int32_t thread_id; |
+ ASSERT_TRUE(exception->GetThreadID(&thread_id)); |
+ EXPECT_EQ(child, thread_id); |
+ |
+ const MDRawExceptionStream* raw = exception->exception(); |
+ ASSERT_TRUE(raw); |
+ EXPECT_EQ(MD_EXCEPTION_CODE_LIN_DUMP_REQUESTED, |
+ raw->exception_record.exception_code); |
+ |
+ close(fds[1]); |
+ unlink(minidump_filename.c_str()); |
+} |