mirror of
https://github.com/chromium/crashpad.git
synced 2024-12-29 00:32:35 +08:00
1aa478d161
This change was partially scripted and partially done manually with vim regex + manually placing the deleted constructors. The script change looked for destructors in the public: section of a class, if that existed the deleted constructors would go before the destructor. For manual placement I looked for any constructor in the public: section of the corresponding class. If there wasn't one, then it would ideally have gone as the first entry except below enums, classes and typedefs. This may not have been perfect, but is hopefully good enough. Fingers crossed. #include "base/macros.h" is removed from files that don't use ignore_result, which is the only other thing defined in base/macros.h. Bug: chromium:1010217 Change-Id: I099526255a40b1ac1264904b4ece2f3f503c9418 Reviewed-on: https://chromium-review.googlesource.com/c/crashpad/crashpad/+/3171034 Reviewed-by: Mark Mentovai <mark@chromium.org> Commit-Queue: Peter Boström <pbos@chromium.org>
693 lines
23 KiB
C++
693 lines
23 KiB
C++
// 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 <utility>
|
|
|
|
#include "base/cxx17_backports.h"
|
|
#include "base/format_macros.h"
|
|
#include "base/strings/stringprintf.h"
|
|
#include "gtest/gtest.h"
|
|
#include "minidump/minidump_extensions.h"
|
|
#include "minidump/minidump_file_writer.h"
|
|
#include "minidump/test/minidump_file_writer_test_util.h"
|
|
#include "minidump/test/minidump_memory_writer_test_util.h"
|
|
#include "minidump/test/minidump_writable_test_util.h"
|
|
#include "snapshot/test/test_memory_snapshot.h"
|
|
#include "util/file/string_file.h"
|
|
|
|
namespace crashpad {
|
|
namespace test {
|
|
namespace {
|
|
|
|
constexpr MinidumpStreamType kBogusStreamType =
|
|
static_cast<MinidumpStreamType>(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) {
|
|
constexpr 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_DIRECTORY* directory;
|
|
const MINIDUMP_HEADER* header =
|
|
MinidumpHeaderAtStart(file_contents, &directory);
|
|
ASSERT_NO_FATAL_FAILURE(VerifyMinidumpHeader(header, expected_streams, 0));
|
|
ASSERT_TRUE(directory);
|
|
|
|
size_t directory_index = 0;
|
|
if (expected_streams > 1) {
|
|
ASSERT_EQ(directory[directory_index].StreamType, kBogusStreamType);
|
|
ASSERT_EQ(directory[directory_index].Location.DataSize, 0u);
|
|
ASSERT_EQ(directory[directory_index].Location.Rva, kMemoryListStreamOffset);
|
|
++directory_index;
|
|
}
|
|
|
|
ASSERT_EQ(directory[directory_index].StreamType,
|
|
kMinidumpStreamTypeMemoryList);
|
|
EXPECT_EQ(directory[directory_index].Location.Rva, kMemoryListStreamOffset);
|
|
|
|
*memory_list = MinidumpWritableAtLocationDescriptor<MINIDUMP_MEMORY_LIST>(
|
|
file_contents, directory[directory_index].Location);
|
|
ASSERT_TRUE(memory_list);
|
|
}
|
|
|
|
TEST(MinidumpMemoryWriter, EmptyMemoryList) {
|
|
MinidumpFileWriter minidump_file_writer;
|
|
auto memory_list_writer = std::make_unique<MinidumpMemoryListWriter>();
|
|
|
|
ASSERT_TRUE(minidump_file_writer.AddStream(std::move(memory_list_writer)));
|
|
|
|
StringFile string_file;
|
|
ASSERT_TRUE(minidump_file_writer.WriteEverything(&string_file));
|
|
|
|
ASSERT_EQ(string_file.string().size(),
|
|
sizeof(MINIDUMP_HEADER) + sizeof(MINIDUMP_DIRECTORY) +
|
|
sizeof(MINIDUMP_MEMORY_LIST));
|
|
|
|
const MINIDUMP_MEMORY_LIST* memory_list = nullptr;
|
|
ASSERT_NO_FATAL_FAILURE(
|
|
GetMemoryListStream(string_file.string(), &memory_list, 1));
|
|
|
|
EXPECT_EQ(memory_list->NumberOfMemoryRanges, 0u);
|
|
}
|
|
|
|
TEST(MinidumpMemoryWriter, OneMemoryRegion) {
|
|
MinidumpFileWriter minidump_file_writer;
|
|
auto memory_list_writer = std::make_unique<MinidumpMemoryListWriter>();
|
|
|
|
constexpr uint64_t kBaseAddress = 0xfedcba9876543210;
|
|
constexpr size_t kSize = 0x1000;
|
|
constexpr uint8_t kValue = 'm';
|
|
|
|
auto memory_writer =
|
|
std::make_unique<TestMinidumpMemoryWriter>(kBaseAddress, kSize, kValue);
|
|
memory_list_writer->AddMemory(std::move(memory_writer));
|
|
|
|
ASSERT_TRUE(minidump_file_writer.AddStream(std::move(memory_list_writer)));
|
|
|
|
StringFile string_file;
|
|
ASSERT_TRUE(minidump_file_writer.WriteEverything(&string_file));
|
|
|
|
const MINIDUMP_MEMORY_LIST* memory_list = nullptr;
|
|
ASSERT_NO_FATAL_FAILURE(
|
|
GetMemoryListStream(string_file.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],
|
|
string_file.string(),
|
|
kValue,
|
|
true);
|
|
}
|
|
|
|
TEST(MinidumpMemoryWriter, TwoMemoryRegions) {
|
|
MinidumpFileWriter minidump_file_writer;
|
|
auto memory_list_writer = std::make_unique<MinidumpMemoryListWriter>();
|
|
|
|
constexpr uint64_t kBaseAddress0 = 0xc0ffee;
|
|
constexpr size_t kSize0 = 0x0100;
|
|
constexpr uint8_t kValue0 = '6';
|
|
constexpr uint64_t kBaseAddress1 = 0xfac00fac;
|
|
constexpr size_t kSize1 = 0x0200;
|
|
constexpr uint8_t kValue1 = '!';
|
|
|
|
auto memory_writer_0 = std::make_unique<TestMinidumpMemoryWriter>(
|
|
kBaseAddress0, kSize0, kValue0);
|
|
memory_list_writer->AddMemory(std::move(memory_writer_0));
|
|
auto memory_writer_1 = std::make_unique<TestMinidumpMemoryWriter>(
|
|
kBaseAddress1, kSize1, kValue1);
|
|
memory_list_writer->AddMemory(std::move(memory_writer_1));
|
|
|
|
ASSERT_TRUE(minidump_file_writer.AddStream(std::move(memory_list_writer)));
|
|
|
|
StringFile string_file;
|
|
ASSERT_TRUE(minidump_file_writer.WriteEverything(&string_file));
|
|
|
|
const MINIDUMP_MEMORY_LIST* memory_list = nullptr;
|
|
ASSERT_NO_FATAL_FAILURE(
|
|
GetMemoryListStream(string_file.string(), &memory_list, 1));
|
|
|
|
EXPECT_EQ(memory_list->NumberOfMemoryRanges, 2u);
|
|
|
|
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],
|
|
string_file.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],
|
|
string_file.string(),
|
|
kValue1,
|
|
true);
|
|
}
|
|
}
|
|
|
|
TEST(MinidumpMemoryWriter, RegionReadFails) {
|
|
MinidumpFileWriter minidump_file_writer;
|
|
auto memory_list_writer = std::make_unique<MinidumpMemoryListWriter>();
|
|
|
|
constexpr uint64_t kBaseAddress = 0xfedcba9876543210;
|
|
constexpr size_t kSize = 0x1000;
|
|
constexpr uint8_t kValue = 'm';
|
|
|
|
auto memory_writer =
|
|
std::make_unique<TestMinidumpMemoryWriter>(kBaseAddress, kSize, kValue);
|
|
|
|
// Make the read of that memory fail.
|
|
memory_writer->SetShouldFailRead(true);
|
|
|
|
memory_list_writer->AddMemory(std::move(memory_writer));
|
|
|
|
ASSERT_TRUE(minidump_file_writer.AddStream(std::move(memory_list_writer)));
|
|
|
|
StringFile string_file;
|
|
ASSERT_TRUE(minidump_file_writer.WriteEverything(&string_file));
|
|
|
|
const MINIDUMP_MEMORY_LIST* memory_list = nullptr;
|
|
ASSERT_NO_FATAL_FAILURE(
|
|
GetMemoryListStream(string_file.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],
|
|
string_file.string(),
|
|
0xfe, // Not kValue ('m'), but the value that the implementation inserts
|
|
// if memory is unreadable.
|
|
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(const TestMemoryStream&) = delete;
|
|
TestMemoryStream& operator=(const TestMemoryStream&) = delete;
|
|
|
|
~TestMemoryStream() override {}
|
|
|
|
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<MinidumpWritable*> Children() override {
|
|
EXPECT_GE(state(), kStateFrozen);
|
|
std::vector<MinidumpWritable*> children(1, memory());
|
|
return children;
|
|
}
|
|
|
|
bool WriteObject(FileWriterInterface* file_writer) override {
|
|
EXPECT_EQ(state(), kStateWritable);
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
TestMinidumpMemoryWriter memory_;
|
|
};
|
|
|
|
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;
|
|
|
|
constexpr uint64_t kBaseAddress0 = 0x1000;
|
|
constexpr size_t kSize0 = 0x0400;
|
|
constexpr uint8_t kValue0 = '1';
|
|
auto test_memory_stream =
|
|
std::make_unique<TestMemoryStream>(kBaseAddress0, kSize0, kValue0);
|
|
|
|
auto memory_list_writer = std::make_unique<MinidumpMemoryListWriter>();
|
|
memory_list_writer->AddNonOwnedMemory(test_memory_stream->memory());
|
|
|
|
ASSERT_TRUE(minidump_file_writer.AddStream(std::move(test_memory_stream)));
|
|
|
|
constexpr uint64_t kBaseAddress1 = 0x2000;
|
|
constexpr size_t kSize1 = 0x0400;
|
|
constexpr uint8_t kValue1 = 'm';
|
|
|
|
auto memory_writer = std::make_unique<TestMinidumpMemoryWriter>(
|
|
kBaseAddress1, kSize1, kValue1);
|
|
memory_list_writer->AddMemory(std::move(memory_writer));
|
|
|
|
ASSERT_TRUE(minidump_file_writer.AddStream(std::move(memory_list_writer)));
|
|
|
|
StringFile string_file;
|
|
ASSERT_TRUE(minidump_file_writer.WriteEverything(&string_file));
|
|
|
|
const MINIDUMP_MEMORY_LIST* memory_list = nullptr;
|
|
ASSERT_NO_FATAL_FAILURE(
|
|
GetMemoryListStream(string_file.string(), &memory_list, 2));
|
|
|
|
EXPECT_EQ(memory_list->NumberOfMemoryRanges, 2u);
|
|
|
|
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],
|
|
string_file.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],
|
|
string_file.string(),
|
|
kValue1,
|
|
true);
|
|
}
|
|
}
|
|
|
|
TEST(MinidumpMemoryWriter, AddFromSnapshot) {
|
|
MINIDUMP_MEMORY_DESCRIPTOR expect_memory_descriptors[3] = {};
|
|
uint8_t values[base::size(expect_memory_descriptors)] = {};
|
|
|
|
expect_memory_descriptors[0].StartOfMemoryRange = 0;
|
|
expect_memory_descriptors[0].Memory.DataSize = 0x1000;
|
|
values[0] = 0x01;
|
|
|
|
expect_memory_descriptors[1].StartOfMemoryRange = 0x2000;
|
|
expect_memory_descriptors[1].Memory.DataSize = 0x2000;
|
|
values[1] = 0xf4;
|
|
|
|
expect_memory_descriptors[2].StartOfMemoryRange = 0x7654321000000000;
|
|
expect_memory_descriptors[2].Memory.DataSize = 0x800;
|
|
values[2] = 0xa9;
|
|
|
|
std::vector<std::unique_ptr<TestMemorySnapshot>> memory_snapshots_owner;
|
|
std::vector<const MemorySnapshot*> memory_snapshots;
|
|
for (size_t index = 0; index < base::size(expect_memory_descriptors);
|
|
++index) {
|
|
memory_snapshots_owner.push_back(std::make_unique<TestMemorySnapshot>());
|
|
TestMemorySnapshot* memory_snapshot = memory_snapshots_owner.back().get();
|
|
memory_snapshot->SetAddress(
|
|
expect_memory_descriptors[index].StartOfMemoryRange);
|
|
memory_snapshot->SetSize(expect_memory_descriptors[index].Memory.DataSize);
|
|
memory_snapshot->SetValue(values[index]);
|
|
memory_snapshots.push_back(memory_snapshot);
|
|
}
|
|
|
|
auto memory_list_writer = std::make_unique<MinidumpMemoryListWriter>();
|
|
memory_list_writer->AddFromSnapshot(memory_snapshots);
|
|
|
|
MinidumpFileWriter minidump_file_writer;
|
|
ASSERT_TRUE(minidump_file_writer.AddStream(std::move(memory_list_writer)));
|
|
|
|
StringFile string_file;
|
|
ASSERT_TRUE(minidump_file_writer.WriteEverything(&string_file));
|
|
|
|
const MINIDUMP_MEMORY_LIST* memory_list = nullptr;
|
|
ASSERT_NO_FATAL_FAILURE(
|
|
GetMemoryListStream(string_file.string(), &memory_list, 1));
|
|
|
|
ASSERT_EQ(memory_list->NumberOfMemoryRanges, 3u);
|
|
|
|
for (size_t index = 0; index < memory_list->NumberOfMemoryRanges; ++index) {
|
|
SCOPED_TRACE(base::StringPrintf("index %" PRIuS, index));
|
|
ExpectMinidumpMemoryDescriptorAndContents(
|
|
&expect_memory_descriptors[index],
|
|
&memory_list->MemoryRanges[index],
|
|
string_file.string(),
|
|
values[index],
|
|
index == memory_list->NumberOfMemoryRanges - 1);
|
|
}
|
|
}
|
|
|
|
TEST(MinidumpMemoryWriter, CoalesceExplicitMultiple) {
|
|
MINIDUMP_MEMORY_DESCRIPTOR expect_memory_descriptors[4] = {};
|
|
uint8_t values[base::size(expect_memory_descriptors)] = {};
|
|
|
|
expect_memory_descriptors[0].StartOfMemoryRange = 0;
|
|
expect_memory_descriptors[0].Memory.DataSize = 1000;
|
|
values[0] = 0x01;
|
|
|
|
expect_memory_descriptors[1].StartOfMemoryRange = 10000;
|
|
expect_memory_descriptors[1].Memory.DataSize = 2000;
|
|
values[1] = 0xf4;
|
|
|
|
expect_memory_descriptors[2].StartOfMemoryRange = 0x1111111111111111;
|
|
expect_memory_descriptors[2].Memory.DataSize = 1024;
|
|
values[2] = 0x99;
|
|
|
|
expect_memory_descriptors[3].StartOfMemoryRange = 0xfedcba9876543210;
|
|
expect_memory_descriptors[3].Memory.DataSize = 1024;
|
|
values[3] = 0x88;
|
|
|
|
struct {
|
|
uint64_t base;
|
|
size_t size;
|
|
uint8_t value;
|
|
} snapshots_to_add[] = {
|
|
// Various overlapping.
|
|
{0, 500, 0x01},
|
|
{0, 500, 0x01},
|
|
{250, 500, 0x01},
|
|
{600, 400, 0x01},
|
|
|
|
// Empty removed.
|
|
{0, 0, 0xbb},
|
|
{300, 0, 0xcc},
|
|
{1000, 0, 0xdd},
|
|
{12000, 0, 0xee},
|
|
|
|
// Abutting.
|
|
{10000, 500, 0xf4},
|
|
{10500, 500, 0xf4},
|
|
{11000, 1000, 0xf4},
|
|
|
|
// Large base addresses.
|
|
{ 0xfedcba9876543210, 1024, 0x88 },
|
|
{ 0x1111111111111111, 1024, 0x99 },
|
|
};
|
|
|
|
std::vector<std::unique_ptr<TestMemorySnapshot>> memory_snapshots_owner;
|
|
std::vector<const MemorySnapshot*> memory_snapshots;
|
|
for (const auto& to_add : snapshots_to_add) {
|
|
memory_snapshots_owner.push_back(std::make_unique<TestMemorySnapshot>());
|
|
TestMemorySnapshot* memory_snapshot = memory_snapshots_owner.back().get();
|
|
memory_snapshot->SetAddress(to_add.base);
|
|
memory_snapshot->SetSize(to_add.size);
|
|
memory_snapshot->SetValue(to_add.value);
|
|
memory_snapshots.push_back(memory_snapshot);
|
|
}
|
|
|
|
auto memory_list_writer = std::make_unique<MinidumpMemoryListWriter>();
|
|
memory_list_writer->AddFromSnapshot(memory_snapshots);
|
|
|
|
MinidumpFileWriter minidump_file_writer;
|
|
minidump_file_writer.AddStream(std::move(memory_list_writer));
|
|
|
|
StringFile string_file;
|
|
ASSERT_TRUE(minidump_file_writer.WriteEverything(&string_file));
|
|
|
|
const MINIDUMP_MEMORY_LIST* memory_list = nullptr;
|
|
ASSERT_NO_FATAL_FAILURE(
|
|
GetMemoryListStream(string_file.string(), &memory_list, 1));
|
|
|
|
ASSERT_EQ(4u, memory_list->NumberOfMemoryRanges);
|
|
|
|
for (size_t index = 0; index < memory_list->NumberOfMemoryRanges; ++index) {
|
|
SCOPED_TRACE(base::StringPrintf("index %" PRIuS, index));
|
|
ExpectMinidumpMemoryDescriptorAndContents(
|
|
&expect_memory_descriptors[index],
|
|
&memory_list->MemoryRanges[index],
|
|
string_file.string(),
|
|
values[index],
|
|
index == memory_list->NumberOfMemoryRanges - 1);
|
|
}
|
|
}
|
|
|
|
struct TestRange {
|
|
TestRange(uint64_t base, size_t size) : base(base), size(size) {}
|
|
|
|
uint64_t base;
|
|
size_t size;
|
|
};
|
|
|
|
// Parses a string spec to build a list of ranges suitable for CoalesceTest().
|
|
std::vector<TestRange> ParseCoalesceSpec(const char* spec) {
|
|
std::vector<TestRange> result;
|
|
enum { kNone, kSpace, kDot } state = kNone;
|
|
const char* range_started_at = nullptr;
|
|
for (const char* p = spec;; ++p) {
|
|
EXPECT_TRUE(*p == ' ' || *p == '.' || *p == 0);
|
|
if (*p == ' ' || *p == 0) {
|
|
if (state == kDot) {
|
|
result.push_back(
|
|
TestRange(range_started_at - spec, p - range_started_at));
|
|
}
|
|
state = kSpace;
|
|
range_started_at = nullptr;
|
|
} else if (*p == '.') {
|
|
if (state != kDot) {
|
|
range_started_at = p;
|
|
state = kDot;
|
|
}
|
|
}
|
|
|
|
if (*p == 0)
|
|
break;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
TEST(MinidumpMemoryWriter, CoalesceSpecHelperParse) {
|
|
const auto empty = ParseCoalesceSpec("");
|
|
ASSERT_EQ(empty.size(), 0u);
|
|
|
|
const auto a = ParseCoalesceSpec("...");
|
|
ASSERT_EQ(a.size(), 1u);
|
|
EXPECT_EQ(a[0].base, 0u);
|
|
EXPECT_EQ(a[0].size, 3u);
|
|
|
|
const auto b = ParseCoalesceSpec(" ...");
|
|
ASSERT_EQ(b.size(), 1u);
|
|
EXPECT_EQ(b[0].base, 2u);
|
|
EXPECT_EQ(b[0].size, 3u);
|
|
|
|
const auto c = ParseCoalesceSpec(" ... ");
|
|
ASSERT_EQ(c.size(), 1u);
|
|
EXPECT_EQ(c[0].base, 2u);
|
|
EXPECT_EQ(c[0].size, 3u);
|
|
|
|
const auto d = ParseCoalesceSpec(" ... ....");
|
|
ASSERT_EQ(d.size(), 2u);
|
|
EXPECT_EQ(d[0].base, 2u);
|
|
EXPECT_EQ(d[0].size, 3u);
|
|
EXPECT_EQ(d[1].base, 7u);
|
|
EXPECT_EQ(d[1].size, 4u);
|
|
|
|
const auto e = ParseCoalesceSpec(" ... ...... ... ");
|
|
ASSERT_EQ(e.size(), 3u);
|
|
EXPECT_EQ(e[0].base, 2u);
|
|
EXPECT_EQ(e[0].size, 3u);
|
|
EXPECT_EQ(e[1].base, 7u);
|
|
EXPECT_EQ(e[1].size, 6u);
|
|
EXPECT_EQ(e[2].base, 14u);
|
|
EXPECT_EQ(e[2].size, 3u);
|
|
}
|
|
|
|
constexpr uint8_t kMemoryValue = 0xcd;
|
|
|
|
// Builds a coalesce test out of specs of ' ' and '.'. Tests that when the two
|
|
// ranges are added and coalesced, the result is equal to expected.
|
|
void CoalesceTest(const char* r1_spec,
|
|
const char* r2_spec,
|
|
const char* expected_spec) {
|
|
auto r1 = ParseCoalesceSpec(r1_spec);
|
|
auto r2 = ParseCoalesceSpec(r2_spec);
|
|
auto expected = ParseCoalesceSpec(expected_spec);
|
|
|
|
std::vector<MINIDUMP_MEMORY_DESCRIPTOR> expect_memory_descriptors;
|
|
for (const auto& range : expected) {
|
|
MINIDUMP_MEMORY_DESCRIPTOR mmd = {};
|
|
mmd.StartOfMemoryRange = range.base;
|
|
mmd.Memory.DataSize = static_cast<uint32_t>(range.size);
|
|
expect_memory_descriptors.push_back(mmd);
|
|
}
|
|
|
|
std::vector<std::unique_ptr<TestMemorySnapshot>> memory_snapshots_owner;
|
|
std::vector<const MemorySnapshot*> memory_snapshots;
|
|
|
|
const auto add_test_memory_snapshots = [&memory_snapshots_owner,
|
|
&memory_snapshots](
|
|
std::vector<TestRange> ranges) {
|
|
for (const auto& r : ranges) {
|
|
memory_snapshots_owner.push_back(std::make_unique<TestMemorySnapshot>());
|
|
TestMemorySnapshot* memory_snapshot = memory_snapshots_owner.back().get();
|
|
memory_snapshot->SetAddress(r.base);
|
|
memory_snapshot->SetSize(r.size);
|
|
memory_snapshot->SetValue(kMemoryValue);
|
|
memory_snapshots.push_back(memory_snapshot);
|
|
}
|
|
};
|
|
add_test_memory_snapshots(r1);
|
|
add_test_memory_snapshots(r2);
|
|
|
|
auto memory_list_writer = std::make_unique<MinidumpMemoryListWriter>();
|
|
memory_list_writer->AddFromSnapshot(memory_snapshots);
|
|
|
|
MinidumpFileWriter minidump_file_writer;
|
|
minidump_file_writer.AddStream(std::move(memory_list_writer));
|
|
|
|
StringFile string_file;
|
|
ASSERT_TRUE(minidump_file_writer.WriteEverything(&string_file));
|
|
|
|
const MINIDUMP_MEMORY_LIST* memory_list = nullptr;
|
|
ASSERT_NO_FATAL_FAILURE(
|
|
GetMemoryListStream(string_file.string(), &memory_list, 1));
|
|
|
|
ASSERT_EQ(expected.size(), memory_list->NumberOfMemoryRanges);
|
|
|
|
for (size_t index = 0; index < memory_list->NumberOfMemoryRanges; ++index) {
|
|
SCOPED_TRACE(base::StringPrintf("index %" PRIuS, index));
|
|
ExpectMinidumpMemoryDescriptorAndContents(
|
|
&expect_memory_descriptors[index],
|
|
&memory_list->MemoryRanges[index],
|
|
string_file.string(),
|
|
kMemoryValue,
|
|
index == memory_list->NumberOfMemoryRanges - 1);
|
|
}
|
|
}
|
|
|
|
TEST(MinidumpMemoryWriter, CoalescePairsVariousCases) {
|
|
// clang-format off
|
|
|
|
CoalesceTest(" .........",
|
|
" .......",
|
|
/* result */ " ..............");
|
|
|
|
CoalesceTest(" .......",
|
|
" .........",
|
|
/* result */ " ..............");
|
|
|
|
CoalesceTest(" ...",
|
|
" .........",
|
|
/* result */ " .........");
|
|
|
|
CoalesceTest(" .........",
|
|
" ......",
|
|
/* result */ " .........");
|
|
|
|
CoalesceTest(" ...",
|
|
" ........",
|
|
/* result */ " ........");
|
|
|
|
CoalesceTest(" ........",
|
|
" ...",
|
|
/* result */ " ........");
|
|
|
|
CoalesceTest(" ...",
|
|
" ........",
|
|
/* result */ " ........");
|
|
|
|
CoalesceTest(" ........",
|
|
" ...",
|
|
/* result */ " ........");
|
|
|
|
CoalesceTest(" ... ",
|
|
" ...",
|
|
/* result */ " ... ...");
|
|
|
|
CoalesceTest(" ...",
|
|
" ... ",
|
|
/* result */ " ... ...");
|
|
|
|
CoalesceTest("...",
|
|
".....",
|
|
/* result */ ".....");
|
|
|
|
CoalesceTest("...",
|
|
" ..",
|
|
/* result */ ".....");
|
|
|
|
CoalesceTest(" .....",
|
|
" ..",
|
|
/* result */ " .......");
|
|
|
|
CoalesceTest(" ......... ......",
|
|
" .......",
|
|
/* result */ " ..................");
|
|
|
|
CoalesceTest(" .......",
|
|
" ......... ......",
|
|
/* result */ " ..................");
|
|
|
|
CoalesceTest(" .....",
|
|
" ......... ......",
|
|
/* result */ " ......... ......");
|
|
|
|
CoalesceTest(" ......... ....... .... .",
|
|
" ......... ...... ....",
|
|
/* result */ " .......................... .......");
|
|
|
|
// clang-format on
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace test
|
|
} // namespace crashpad
|