// Copyright 2015 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 "snapshot/minidump/module_snapshot_minidump.h" #include #include "minidump/minidump_extensions.h" #include "snapshot/minidump/minidump_annotation_reader.h" #include "snapshot/minidump/minidump_string_reader.h" #include "snapshot/minidump/minidump_simple_string_dictionary_reader.h" #include "snapshot/minidump/minidump_string_list_reader.h" #include "util/misc/pdb_structures.h" namespace crashpad { namespace internal { ModuleSnapshotMinidump::ModuleSnapshotMinidump() : ModuleSnapshot(), minidump_module_(), annotations_vector_(), annotations_simple_map_(), annotation_objects_(), uuid_(), name_(), age_(0), initialized_() { } ModuleSnapshotMinidump::~ModuleSnapshotMinidump() { } bool ModuleSnapshotMinidump::Initialize( FileReaderInterface* file_reader, RVA minidump_module_rva, const MINIDUMP_LOCATION_DESCRIPTOR* minidump_module_crashpad_info_location) { INITIALIZATION_STATE_SET_INITIALIZING(initialized_); if (!file_reader->SeekSet(minidump_module_rva)) { return false; } if (!file_reader->ReadExactly(&minidump_module_, sizeof(minidump_module_))) { return false; } if (!InitializeModuleCrashpadInfo(file_reader, minidump_module_crashpad_info_location)) { return false; } ReadMinidumpUTF16String(file_reader, minidump_module_.ModuleNameRva, &name_); if (minidump_module_.CvRecord.Rva != 0) { CodeViewRecordPDB70 cv; if (!file_reader->SeekSet(minidump_module_.CvRecord.Rva)) { return false; } if (!file_reader->ReadExactly(&cv, sizeof(cv))) { return false; } if (cv.signature == 'SDSR') { age_ = cv.age; uuid_ = cv.uuid; } else if (cv.signature != '01BN') { LOG(ERROR) << "Bad CodeView signature in module"; return false; } else { LOG(ERROR) << "NB10 not supported"; return false; } } INITIALIZATION_STATE_SET_VALID(initialized_); return true; } std::string ModuleSnapshotMinidump::Name() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return name_; } uint64_t ModuleSnapshotMinidump::Address() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return minidump_module_.BaseOfImage; } uint64_t ModuleSnapshotMinidump::Size() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return minidump_module_.SizeOfImage; } time_t ModuleSnapshotMinidump::Timestamp() const { return minidump_module_.TimeDateStamp; } void ModuleSnapshotMinidump::FileVersion(uint16_t* version_0, uint16_t* version_1, uint16_t* version_2, uint16_t* version_3) const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); uint32_t version_01 = minidump_module_.VersionInfo.dwFileVersionMS; uint32_t version_23 = minidump_module_.VersionInfo.dwFileVersionLS; *version_0 = static_cast(version_01 >> 16); *version_1 = static_cast(version_01 & 0xFFFF); *version_2 = static_cast(version_23 >> 16); *version_3 = static_cast(version_23 & 0xFFFF); } void ModuleSnapshotMinidump::SourceVersion(uint16_t* version_0, uint16_t* version_1, uint16_t* version_2, uint16_t* version_3) const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); uint32_t version_01 = minidump_module_.VersionInfo.dwProductVersionMS; uint32_t version_23 = minidump_module_.VersionInfo.dwProductVersionLS; *version_0 = static_cast(version_01 >> 16); *version_1 = static_cast(version_01 & 0xFFFF); *version_2 = static_cast(version_23 >> 16); *version_3 = static_cast(version_23 & 0xFFFF); } ModuleSnapshot::ModuleType ModuleSnapshotMinidump::GetModuleType() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); switch (minidump_module_.VersionInfo.dwFileType) { case VFT_APP: return kModuleTypeExecutable; case VFT_DLL: return kModuleTypeSharedLibrary; } return kModuleTypeUnknown; } void ModuleSnapshotMinidump::UUIDAndAge(crashpad::UUID* uuid, uint32_t* age) const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); *uuid = uuid_; *age = age_; } std::string ModuleSnapshotMinidump::DebugFileName() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); NOTREACHED(); // https://crashpad.chromium.org/bug/10 return std::string(); } std::vector ModuleSnapshotMinidump::AnnotationsVector() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return annotations_vector_; } std::map ModuleSnapshotMinidump::AnnotationsSimpleMap() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return annotations_simple_map_; } std::vector ModuleSnapshotMinidump::AnnotationObjects() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return annotation_objects_; } std::set> ModuleSnapshotMinidump::ExtraMemoryRanges() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); NOTREACHED(); // https://crashpad.chromium.org/bug/10 return std::set>(); } std::vector ModuleSnapshotMinidump::CustomMinidumpStreams() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); NOTREACHED(); // https://crashpad.chromium.org/bug/10 return std::vector(); } bool ModuleSnapshotMinidump::InitializeModuleCrashpadInfo( FileReaderInterface* file_reader, const MINIDUMP_LOCATION_DESCRIPTOR* minidump_module_crashpad_info_location) { if (!minidump_module_crashpad_info_location || minidump_module_crashpad_info_location->Rva == 0) { return true; } MinidumpModuleCrashpadInfo minidump_module_crashpad_info; if (minidump_module_crashpad_info_location->DataSize < sizeof(minidump_module_crashpad_info)) { LOG(ERROR) << "minidump_module_crashpad_info size mismatch"; return false; } if (!file_reader->SeekSet(minidump_module_crashpad_info_location->Rva)) { return false; } if (!file_reader->ReadExactly(&minidump_module_crashpad_info, sizeof(minidump_module_crashpad_info))) { return false; } if (minidump_module_crashpad_info.version != MinidumpModuleCrashpadInfo::kVersion) { LOG(ERROR) << "minidump_module_crashpad_info version mismatch"; return false; } if (!ReadMinidumpStringList(file_reader, minidump_module_crashpad_info.list_annotations, &annotations_vector_)) { return false; } if (!ReadMinidumpSimpleStringDictionary( file_reader, minidump_module_crashpad_info.simple_annotations, &annotations_simple_map_)) { return false; } if (!ReadMinidumpAnnotationList( file_reader, minidump_module_crashpad_info.annotation_objects, &annotation_objects_)) { return false; } return true; } } // namespace internal } // namespace crashpad