2014-10-21 14:15:07 -04:00
|
|
|
|
// 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/test/minidump_writable_test_util.h"
|
|
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
|
|
2014-10-22 18:35:18 -04:00
|
|
|
|
#include "base/strings/string16.h"
|
2014-10-21 14:15:07 -04:00
|
|
|
|
#include "gtest/gtest.h"
|
2014-11-07 09:44:09 -05:00
|
|
|
|
#include "util/file/file_writer.h"
|
2015-09-14 11:09:46 -07:00
|
|
|
|
#include "util/misc/implicit_cast.h"
|
2014-10-21 14:15:07 -04:00
|
|
|
|
|
|
|
|
|
namespace crashpad {
|
|
|
|
|
namespace test {
|
|
|
|
|
|
2014-11-06 16:47:57 -05:00
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
//! \brief Returns an untyped minidump object located within a minidump file’s
|
|
|
|
|
//! contents, where the offset of the object is known.
|
|
|
|
|
//!
|
|
|
|
|
//! \param[in] file_contents The contents of the minidump file.
|
|
|
|
|
//! \param[in] rva The offset within the minidump file of the desired object.
|
|
|
|
|
//!
|
|
|
|
|
//! \return If \a rva is within the range of \a file_contents, returns a pointer
|
|
|
|
|
//! into \a file_contents at offset \a rva. Otherwise, raises a gtest
|
|
|
|
|
//! assertion failure and returns `nullptr`.
|
|
|
|
|
//!
|
|
|
|
|
//! Do not call this function. Use the typed version, MinidumpWritableAtRVA<>(),
|
|
|
|
|
//! or another type-specific function.
|
2014-10-21 14:15:07 -04:00
|
|
|
|
const void* MinidumpWritableAtRVAInternal(const std::string& file_contents,
|
|
|
|
|
RVA rva) {
|
|
|
|
|
if (rva >= file_contents.size()) {
|
|
|
|
|
EXPECT_LT(rva, file_contents.size());
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return &file_contents[rva];
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-06 16:47:57 -05:00
|
|
|
|
} // namespace
|
|
|
|
|
|
2014-10-21 14:15:07 -04:00
|
|
|
|
const void* MinidumpWritableAtLocationDescriptorInternal(
|
|
|
|
|
const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location,
|
2014-10-22 18:35:18 -04:00
|
|
|
|
size_t expected_size,
|
|
|
|
|
bool allow_oversized_data) {
|
2014-10-21 14:15:07 -04:00
|
|
|
|
if (location.DataSize == 0) {
|
|
|
|
|
EXPECT_EQ(0u, location.Rva);
|
|
|
|
|
return nullptr;
|
2014-10-22 18:35:18 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (allow_oversized_data) {
|
|
|
|
|
if (location.DataSize < expected_size) {
|
|
|
|
|
EXPECT_GE(location.DataSize, expected_size);
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
} else if (location.DataSize != expected_size) {
|
|
|
|
|
EXPECT_EQ(expected_size, location.DataSize);
|
2014-10-21 14:15:07 -04:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RVA end = location.Rva + location.DataSize;
|
|
|
|
|
if (end > file_contents.size()) {
|
|
|
|
|
EXPECT_LE(end, file_contents.size());
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const void* rv = MinidumpWritableAtRVAInternal(file_contents, location.Rva);
|
|
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-22 18:35:18 -04:00
|
|
|
|
template <>
|
|
|
|
|
const IMAGE_DEBUG_MISC* MinidumpWritableAtLocationDescriptor<IMAGE_DEBUG_MISC>(
|
|
|
|
|
const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
const IMAGE_DEBUG_MISC* misc =
|
|
|
|
|
TMinidumpWritableAtLocationDescriptor<IMAGE_DEBUG_MISC>(file_contents,
|
|
|
|
|
location);
|
|
|
|
|
if (!misc) {
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (misc->DataType != IMAGE_DEBUG_MISC_EXENAME) {
|
2014-11-06 16:44:38 -05:00
|
|
|
|
EXPECT_EQ(implicit_cast<uint32_t>(IMAGE_DEBUG_MISC_EXENAME),
|
|
|
|
|
misc->DataType);
|
2014-10-22 18:35:18 -04:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (misc->Length != location.DataSize) {
|
|
|
|
|
EXPECT_EQ(location.DataSize, misc->Length);
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (misc->Unicode == 0) {
|
|
|
|
|
size_t string_length = misc->Length - offsetof(IMAGE_DEBUG_MISC, Data) - 1;
|
|
|
|
|
if (misc->Data[string_length] != '\0') {
|
|
|
|
|
EXPECT_EQ('\0', misc->Data[string_length]);
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
} else if (misc->Unicode == 1) {
|
2014-12-12 11:06:09 -08:00
|
|
|
|
if (misc->Length % sizeof(base::char16) != 0) {
|
|
|
|
|
EXPECT_EQ(0u, misc->Length % sizeof(base::char16));
|
2014-10-22 18:35:18 -04:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-12 11:06:09 -08:00
|
|
|
|
size_t string_length = (misc->Length - offsetof(IMAGE_DEBUG_MISC, Data)) /
|
|
|
|
|
sizeof(base::char16) -
|
|
|
|
|
1;
|
|
|
|
|
const base::char16* data16 =
|
|
|
|
|
reinterpret_cast<const base::char16*>(misc->Data);
|
2014-10-22 18:35:18 -04:00
|
|
|
|
if (data16[string_length] != '\0') {
|
|
|
|
|
EXPECT_EQ('\0', data16[string_length]);
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
ADD_FAILURE() << "misc->Unicode " << misc->Unicode;
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return misc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
const MINIDUMP_HEADER* MinidumpWritableAtLocationDescriptor<MINIDUMP_HEADER>(
|
|
|
|
|
const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
const MINIDUMP_HEADER* header =
|
|
|
|
|
TMinidumpWritableAtLocationDescriptor<MINIDUMP_HEADER>(file_contents,
|
|
|
|
|
location);
|
|
|
|
|
if (!header) {
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (header->Signature != MINIDUMP_SIGNATURE) {
|
2014-11-06 16:44:38 -05:00
|
|
|
|
EXPECT_EQ(implicit_cast<uint32_t>(MINIDUMP_SIGNATURE), header->Signature);
|
2014-10-22 18:35:18 -04:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
if (header->Version != MINIDUMP_VERSION) {
|
2014-11-06 16:44:38 -05:00
|
|
|
|
EXPECT_EQ(implicit_cast<uint32_t>(MINIDUMP_VERSION), header->Version);
|
2014-10-22 18:35:18 -04:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return header;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
struct MinidumpMemoryListTraits {
|
2014-11-05 14:09:01 -05:00
|
|
|
|
using ListType = MINIDUMP_MEMORY_LIST;
|
2015-02-05 09:13:13 -08:00
|
|
|
|
enum : size_t { kElementSize = sizeof(MINIDUMP_MEMORY_DESCRIPTOR) };
|
2014-10-22 18:35:18 -04:00
|
|
|
|
static size_t ElementCount(const ListType* list) {
|
|
|
|
|
return list->NumberOfMemoryRanges;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct MinidumpModuleListTraits {
|
2014-11-05 14:09:01 -05:00
|
|
|
|
using ListType = MINIDUMP_MODULE_LIST;
|
2015-02-05 09:13:13 -08:00
|
|
|
|
enum : size_t { kElementSize = sizeof(MINIDUMP_MODULE) };
|
2014-10-22 18:35:18 -04:00
|
|
|
|
static size_t ElementCount(const ListType* list) {
|
|
|
|
|
return list->NumberOfModules;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct MinidumpThreadListTraits {
|
2014-11-05 14:09:01 -05:00
|
|
|
|
using ListType = MINIDUMP_THREAD_LIST;
|
2015-02-05 09:13:13 -08:00
|
|
|
|
enum : size_t { kElementSize = sizeof(MINIDUMP_THREAD) };
|
2014-10-22 18:35:18 -04:00
|
|
|
|
static size_t ElementCount(const ListType* list) {
|
|
|
|
|
return list->NumberOfThreads;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2015-10-21 10:43:42 -07:00
|
|
|
|
struct MinidumpHandleDataStreamTraits {
|
|
|
|
|
using ListType = MINIDUMP_HANDLE_DATA_STREAM;
|
|
|
|
|
enum : size_t { kElementSize = sizeof(MINIDUMP_HANDLE_DESCRIPTOR) };
|
|
|
|
|
static size_t ElementCount(const ListType* list) {
|
|
|
|
|
return static_cast<size_t>(list->NumberOfDescriptors);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2015-10-13 13:15:44 -07:00
|
|
|
|
struct MinidumpMemoryInfoListTraits {
|
|
|
|
|
using ListType = MINIDUMP_MEMORY_INFO_LIST;
|
|
|
|
|
enum : size_t { kElementSize = sizeof(MINIDUMP_MEMORY_INFO) };
|
|
|
|
|
static size_t ElementCount(const ListType* list) {
|
|
|
|
|
return static_cast<size_t>(list->NumberOfEntries);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2015-03-04 10:53:34 -05:00
|
|
|
|
struct MinidumpModuleCrashpadInfoListTraits {
|
|
|
|
|
using ListType = MinidumpModuleCrashpadInfoList;
|
|
|
|
|
enum : size_t { kElementSize = sizeof(MinidumpModuleCrashpadInfoLink) };
|
2014-10-24 14:44:55 -04:00
|
|
|
|
static size_t ElementCount(const ListType* list) {
|
|
|
|
|
return list->count;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2014-10-22 18:35:18 -04:00
|
|
|
|
struct MinidumpSimpleStringDictionaryListTraits {
|
2014-11-05 14:09:01 -05:00
|
|
|
|
using ListType = MinidumpSimpleStringDictionary;
|
2015-02-05 09:13:13 -08:00
|
|
|
|
enum : size_t { kElementSize = sizeof(MinidumpSimpleStringDictionaryEntry) };
|
2014-10-22 18:35:18 -04:00
|
|
|
|
static size_t ElementCount(const ListType* list) {
|
|
|
|
|
return list->count;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
const typename T::ListType* MinidumpListAtLocationDescriptor(
|
|
|
|
|
const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
const typename T::ListType* list =
|
|
|
|
|
TMinidumpWritableAtLocationDescriptor<typename T::ListType>(file_contents,
|
|
|
|
|
location);
|
|
|
|
|
if (!list) {
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t expected_size =
|
|
|
|
|
sizeof(typename T::ListType) + T::ElementCount(list) * T::kElementSize;
|
|
|
|
|
if (location.DataSize != expected_size) {
|
|
|
|
|
EXPECT_EQ(expected_size, location.DataSize);
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return list;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
const MINIDUMP_MEMORY_LIST* MinidumpWritableAtLocationDescriptor<
|
|
|
|
|
MINIDUMP_MEMORY_LIST>(const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
return MinidumpListAtLocationDescriptor<MinidumpMemoryListTraits>(
|
|
|
|
|
file_contents, location);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
const MINIDUMP_MODULE_LIST* MinidumpWritableAtLocationDescriptor<
|
|
|
|
|
MINIDUMP_MODULE_LIST>(const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
return MinidumpListAtLocationDescriptor<MinidumpModuleListTraits>(
|
|
|
|
|
file_contents, location);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
const MINIDUMP_THREAD_LIST* MinidumpWritableAtLocationDescriptor<
|
|
|
|
|
MINIDUMP_THREAD_LIST>(const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
return MinidumpListAtLocationDescriptor<MinidumpThreadListTraits>(
|
|
|
|
|
file_contents, location);
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-21 10:43:42 -07:00
|
|
|
|
template <>
|
|
|
|
|
const MINIDUMP_HANDLE_DATA_STREAM* MinidumpWritableAtLocationDescriptor<
|
|
|
|
|
MINIDUMP_HANDLE_DATA_STREAM>(const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
return MinidumpListAtLocationDescriptor<MinidumpHandleDataStreamTraits>(
|
|
|
|
|
file_contents, location);
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-13 13:15:44 -07:00
|
|
|
|
template <>
|
|
|
|
|
const MINIDUMP_MEMORY_INFO_LIST* MinidumpWritableAtLocationDescriptor<
|
|
|
|
|
MINIDUMP_MEMORY_INFO_LIST>(const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
return MinidumpListAtLocationDescriptor<MinidumpMemoryInfoListTraits>(
|
|
|
|
|
file_contents, location);
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-24 14:44:55 -04:00
|
|
|
|
template <>
|
2015-03-04 10:53:34 -05:00
|
|
|
|
const MinidumpModuleCrashpadInfoList*
|
|
|
|
|
MinidumpWritableAtLocationDescriptor<MinidumpModuleCrashpadInfoList>(
|
2014-10-24 14:44:55 -04:00
|
|
|
|
const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
2015-03-04 10:53:34 -05:00
|
|
|
|
return MinidumpListAtLocationDescriptor<MinidumpModuleCrashpadInfoListTraits>(
|
2014-10-24 14:44:55 -04:00
|
|
|
|
file_contents, location);
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-22 18:35:18 -04:00
|
|
|
|
template <>
|
|
|
|
|
const MinidumpSimpleStringDictionary*
|
|
|
|
|
MinidumpWritableAtLocationDescriptor<MinidumpSimpleStringDictionary>(
|
|
|
|
|
const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
return MinidumpListAtLocationDescriptor<
|
|
|
|
|
MinidumpSimpleStringDictionaryListTraits>(file_contents, location);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
const T* MinidumpCVPDBAtLocationDescriptor(
|
|
|
|
|
const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
const T* cv_pdb =
|
|
|
|
|
TMinidumpWritableAtLocationDescriptor<T>(file_contents, location);
|
|
|
|
|
if (!cv_pdb) {
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (cv_pdb->signature != T::kSignature) {
|
|
|
|
|
EXPECT_EQ(T::kSignature, cv_pdb->signature);
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t string_length = location.DataSize - offsetof(T, pdb_name) - 1;
|
|
|
|
|
if (cv_pdb->pdb_name[string_length] != '\0') {
|
|
|
|
|
EXPECT_EQ('\0', cv_pdb->pdb_name[string_length]);
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return cv_pdb;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
template <>
|
2015-09-01 09:32:09 -07:00
|
|
|
|
const CodeViewRecordPDB20* MinidumpWritableAtLocationDescriptor<
|
|
|
|
|
CodeViewRecordPDB20>(const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
return MinidumpCVPDBAtLocationDescriptor<CodeViewRecordPDB20>(file_contents,
|
|
|
|
|
location);
|
2014-10-22 18:35:18 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <>
|
2015-09-01 09:32:09 -07:00
|
|
|
|
const CodeViewRecordPDB70* MinidumpWritableAtLocationDescriptor<
|
|
|
|
|
CodeViewRecordPDB70>(const std::string& file_contents,
|
|
|
|
|
const MINIDUMP_LOCATION_DESCRIPTOR& location) {
|
|
|
|
|
return MinidumpCVPDBAtLocationDescriptor<CodeViewRecordPDB70>(file_contents,
|
|
|
|
|
location);
|
2014-10-22 18:35:18 -04:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-07 09:44:09 -05:00
|
|
|
|
TestUInt32MinidumpWritable::TestUInt32MinidumpWritable(uint32_t value)
|
|
|
|
|
: MinidumpWritable(),
|
|
|
|
|
value_(value) {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TestUInt32MinidumpWritable::~TestUInt32MinidumpWritable() {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t TestUInt32MinidumpWritable::SizeOfObject() {
|
|
|
|
|
return sizeof(value_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool TestUInt32MinidumpWritable::WriteObject(FileWriterInterface* file_writer) {
|
|
|
|
|
return file_writer->Write(&value_, sizeof(value_));
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-21 14:15:07 -04:00
|
|
|
|
} // namespace test
|
|
|
|
|
} // namespace crashpad
|