// Copyright 2014 The Crashpad Authors. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "minidump/minidump_memory_writer.h" #include #include #include "base/basictypes.h" #include "gtest/gtest.h" #include "minidump/minidump_extensions.h" #include "minidump/minidump_file_writer.h" #include "minidump/minidump_file_writer_test_util.h" #include "minidump/minidump_memory_writer_test_util.h" #include "minidump/minidump_stream_writer.h" #include "util/file/string_file_writer.h" namespace crashpad { namespace test { namespace { const MinidumpStreamType kBogusStreamType = static_cast(1234); // expected_streams is the expected number of streams in the file. The memory // list must be the last stream. If there is another stream, it must come first, // have stream type kBogusStreamType, and have zero-length data. void GetMemoryListStream(const std::string& file_contents, const MINIDUMP_MEMORY_LIST** memory_list, const uint32_t expected_streams) { const size_t kDirectoryOffset = sizeof(MINIDUMP_HEADER); const size_t kMemoryListStreamOffset = kDirectoryOffset + expected_streams * sizeof(MINIDUMP_DIRECTORY); const size_t kMemoryDescriptorsOffset = kMemoryListStreamOffset + sizeof(MINIDUMP_MEMORY_LIST); ASSERT_GE(file_contents.size(), kMemoryDescriptorsOffset); const MINIDUMP_HEADER* header = reinterpret_cast(&file_contents[0]); ASSERT_NO_FATAL_FAILURE(VerifyMinidumpHeader(header, expected_streams, 0)); const MINIDUMP_DIRECTORY* directory = reinterpret_cast( &file_contents[kDirectoryOffset]); if (expected_streams > 1) { ASSERT_EQ(kBogusStreamType, directory->StreamType); ASSERT_EQ(0u, directory->Location.DataSize); ASSERT_EQ(kMemoryListStreamOffset, directory->Location.Rva); ++directory; } ASSERT_EQ(kMinidumpStreamTypeMemoryList, directory->StreamType); ASSERT_GE(directory->Location.DataSize, sizeof(MINIDUMP_MEMORY_LIST)); ASSERT_EQ(kMemoryListStreamOffset, directory->Location.Rva); *memory_list = reinterpret_cast( &file_contents[kMemoryListStreamOffset]); ASSERT_EQ(sizeof(MINIDUMP_MEMORY_LIST) + (*memory_list)->NumberOfMemoryRanges * sizeof(MINIDUMP_MEMORY_DESCRIPTOR), directory->Location.DataSize); } TEST(MinidumpMemoryWriter, EmptyMemoryList) { MinidumpFileWriter minidump_file_writer; MinidumpMemoryListWriter memory_list_writer; minidump_file_writer.AddStream(&memory_list_writer); StringFileWriter file_writer; ASSERT_TRUE(minidump_file_writer.WriteEverything(&file_writer)); ASSERT_EQ(sizeof(MINIDUMP_HEADER) + sizeof(MINIDUMP_DIRECTORY) + sizeof(MINIDUMP_MEMORY_LIST), file_writer.string().size()); const MINIDUMP_MEMORY_LIST* memory_list; ASSERT_NO_FATAL_FAILURE( GetMemoryListStream(file_writer.string(), &memory_list, 1)); EXPECT_EQ(0u, memory_list->NumberOfMemoryRanges); } TEST(MinidumpMemoryWriter, OneMemoryRegion) { MinidumpFileWriter minidump_file_writer; MinidumpMemoryListWriter memory_list_writer; const uint64_t kBaseAddress = 0xfedcba9876543210; const uint64_t kSize = 0x1000; const uint8_t kValue = 'm'; TestMinidumpMemoryWriter memory_writer(kBaseAddress, kSize, kValue); memory_list_writer.AddMemory(&memory_writer); minidump_file_writer.AddStream(&memory_list_writer); StringFileWriter file_writer; ASSERT_TRUE(minidump_file_writer.WriteEverything(&file_writer)); const MINIDUMP_MEMORY_LIST* memory_list; ASSERT_NO_FATAL_FAILURE( GetMemoryListStream(file_writer.string(), &memory_list, 1)); MINIDUMP_MEMORY_DESCRIPTOR expected; expected.StartOfMemoryRange = kBaseAddress; expected.Memory.DataSize = kSize; expected.Memory.Rva = sizeof(MINIDUMP_HEADER) + sizeof(MINIDUMP_DIRECTORY) + sizeof(MINIDUMP_MEMORY_LIST) + memory_list->NumberOfMemoryRanges * sizeof(MINIDUMP_MEMORY_DESCRIPTOR); ExpectMinidumpMemoryDescriptorAndContents(&expected, &memory_list->MemoryRanges[0], file_writer.string(), kValue, true); } TEST(MinidumpMemoryWriter, TwoMemoryRegions) { MinidumpFileWriter minidump_file_writer; MinidumpMemoryListWriter memory_list_writer; const uint64_t kBaseAddress0 = 0xc0ffee; const uint64_t kSize0 = 0x0100; const uint8_t kValue0 = '6'; const uint64_t kBaseAddress1 = 0xfac00fac; const uint64_t kSize1 = 0x0200; const uint8_t kValue1 = '!'; TestMinidumpMemoryWriter memory_writer_0(kBaseAddress0, kSize0, kValue0); memory_list_writer.AddMemory(&memory_writer_0); TestMinidumpMemoryWriter memory_writer_1(kBaseAddress1, kSize1, kValue1); memory_list_writer.AddMemory(&memory_writer_1); minidump_file_writer.AddStream(&memory_list_writer); StringFileWriter file_writer; ASSERT_TRUE(minidump_file_writer.WriteEverything(&file_writer)); const MINIDUMP_MEMORY_LIST* memory_list; ASSERT_NO_FATAL_FAILURE( GetMemoryListStream(file_writer.string(), &memory_list, 1)); EXPECT_EQ(2u, memory_list->NumberOfMemoryRanges); MINIDUMP_MEMORY_DESCRIPTOR expected; { SCOPED_TRACE("region 0"); expected.StartOfMemoryRange = kBaseAddress0; expected.Memory.DataSize = kSize0; expected.Memory.Rva = sizeof(MINIDUMP_HEADER) + sizeof(MINIDUMP_DIRECTORY) + sizeof(MINIDUMP_MEMORY_LIST) + memory_list->NumberOfMemoryRanges * sizeof(MINIDUMP_MEMORY_DESCRIPTOR); ExpectMinidumpMemoryDescriptorAndContents(&expected, &memory_list->MemoryRanges[0], file_writer.string(), kValue0, false); } { SCOPED_TRACE("region 1"); expected.StartOfMemoryRange = kBaseAddress1; expected.Memory.DataSize = kSize1; expected.Memory.Rva = memory_list->MemoryRanges[0].Memory.Rva + memory_list->MemoryRanges[0].Memory.DataSize; ExpectMinidumpMemoryDescriptorAndContents(&expected, &memory_list->MemoryRanges[1], file_writer.string(), kValue1, true); } } class TestMemoryStream final : public internal::MinidumpStreamWriter { public: TestMemoryStream(uint64_t base_address, size_t size, uint8_t value) : MinidumpStreamWriter(), memory_(base_address, size, value) {} ~TestMemoryStream() {} TestMinidumpMemoryWriter* memory() { return &memory_; } // MinidumpStreamWriter: MinidumpStreamType StreamType() const override { return kBogusStreamType; } protected: // MinidumpWritable: size_t SizeOfObject() override { EXPECT_GE(state(), kStateFrozen); return 0; } std::vector Children() override { EXPECT_GE(state(), kStateFrozen); std::vector children(1, memory()); return children; } bool WriteObject(FileWriterInterface* file_writer) override { EXPECT_EQ(kStateWritable, state()); return true; } private: TestMinidumpMemoryWriter memory_; DISALLOW_COPY_AND_ASSIGN(TestMemoryStream); }; TEST(MinidumpMemoryWriter, ExtraMemory) { // This tests MinidumpMemoryListWriter::AddExtraMemory(). That method adds // a MinidumpMemoryWriter to the MinidumpMemoryListWriter without making the // memory writer a child of the memory list writer. MinidumpFileWriter minidump_file_writer; const uint64_t kBaseAddress0 = 0x1000; const uint64_t kSize0 = 0x0400; const uint8_t kValue0 = '1'; TestMemoryStream test_memory_stream(kBaseAddress0, kSize0, kValue0); MinidumpMemoryListWriter memory_list_writer; memory_list_writer.AddExtraMemory(test_memory_stream.memory()); minidump_file_writer.AddStream(&test_memory_stream); const uint64_t kBaseAddress1 = 0x2000; const uint64_t kSize1 = 0x0400; const uint8_t kValue1 = 'm'; TestMinidumpMemoryWriter memory_writer(kBaseAddress1, kSize1, kValue1); memory_list_writer.AddMemory(&memory_writer); minidump_file_writer.AddStream(&memory_list_writer); StringFileWriter file_writer; ASSERT_TRUE(minidump_file_writer.WriteEverything(&file_writer)); const MINIDUMP_MEMORY_LIST* memory_list; ASSERT_NO_FATAL_FAILURE( GetMemoryListStream(file_writer.string(), &memory_list, 2)); EXPECT_EQ(2u, memory_list->NumberOfMemoryRanges); MINIDUMP_MEMORY_DESCRIPTOR expected; { SCOPED_TRACE("region 0"); expected.StartOfMemoryRange = kBaseAddress0; expected.Memory.DataSize = kSize0; expected.Memory.Rva = sizeof(MINIDUMP_HEADER) + 2 * sizeof(MINIDUMP_DIRECTORY) + sizeof(MINIDUMP_MEMORY_LIST) + memory_list->NumberOfMemoryRanges * sizeof(MINIDUMP_MEMORY_DESCRIPTOR); ExpectMinidumpMemoryDescriptorAndContents(&expected, &memory_list->MemoryRanges[0], file_writer.string(), kValue0, false); } { SCOPED_TRACE("region 1"); expected.StartOfMemoryRange = kBaseAddress1; expected.Memory.DataSize = kSize1; expected.Memory.Rva = memory_list->MemoryRanges[0].Memory.Rva + memory_list->MemoryRanges[0].Memory.DataSize; ExpectMinidumpMemoryDescriptorAndContents(&expected, &memory_list->MemoryRanges[1], file_writer.string(), kValue1, true); } } } // namespace } // namespace test } // namespace crashpad