// 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/win/pe_image_annotations_reader.h" #include #include #include "base/cxx17_backports.h" #include "base/logging.h" #include "base/strings/utf_string_conversions.h" #include "client/annotation.h" #include "client/simple_string_dictionary.h" #include "snapshot/snapshot_constants.h" #include "snapshot/win/pe_image_reader.h" #include "snapshot/win/process_reader_win.h" #include "util/win/process_structs.h" namespace crashpad { namespace process_types { template struct Annotation { typename Traits::Pointer link_node; typename Traits::Pointer name; typename Traits::Pointer value; uint32_t size; uint16_t type; }; template struct AnnotationList { typename Traits::Pointer tail_pointer; Annotation head; Annotation tail; }; } // namespace process_types PEImageAnnotationsReader::PEImageAnnotationsReader( ProcessReaderWin* process_reader, const PEImageReader* pe_image_reader, const std::wstring& name) : name_(name), process_reader_(process_reader), pe_image_reader_(pe_image_reader) { } std::map PEImageAnnotationsReader::SimpleMap() const { std::map simple_map_annotations; if (process_reader_->Is64Bit()) { ReadCrashpadSimpleAnnotations( &simple_map_annotations); } else { ReadCrashpadSimpleAnnotations( &simple_map_annotations); } return simple_map_annotations; } std::vector PEImageAnnotationsReader::AnnotationsList() const { std::vector annotations; if (process_reader_->Is64Bit()) { ReadCrashpadAnnotationsList( &annotations); } else { ReadCrashpadAnnotationsList( &annotations); } return annotations; } template void PEImageAnnotationsReader::ReadCrashpadSimpleAnnotations( std::map* simple_map_annotations) const { process_types::CrashpadInfo crashpad_info; if (!pe_image_reader_->GetCrashpadInfo(&crashpad_info) || !crashpad_info.simple_annotations) { return; } std::vector simple_annotations(SimpleStringDictionary::num_entries); if (!process_reader_->Memory()->Read( crashpad_info.simple_annotations, simple_annotations.size() * sizeof(simple_annotations[0]), &simple_annotations[0])) { LOG(WARNING) << "could not read simple annotations from " << base::WideToUTF8(name_); return; } for (const auto& entry : simple_annotations) { size_t key_length = strnlen(entry.key, sizeof(entry.key)); if (key_length) { std::string key(entry.key, key_length); std::string value(entry.value, strnlen(entry.value, sizeof(entry.value))); if (!simple_map_annotations->insert(std::make_pair(key, value)).second) { LOG(INFO) << "duplicate simple annotation " << key << " in " << base::WideToUTF8(name_); } } } } // TODO(rsesek): When there is a platform-agnostic remote memory reader // interface available, use it so that the implementation is not duplicated // in the MachOImageAnnotationsReader. template void PEImageAnnotationsReader::ReadCrashpadAnnotationsList( std::vector* vector_annotations) const { process_types::CrashpadInfo crashpad_info; if (!pe_image_reader_->GetCrashpadInfo(&crashpad_info) || !crashpad_info.annotations_list) { return; } process_types::AnnotationList annotation_list_object; if (!process_reader_->Memory()->Read(crashpad_info.annotations_list, sizeof(annotation_list_object), &annotation_list_object)) { LOG(WARNING) << "could not read annotations list object in " << base::WideToUTF8(name_); return; } process_types::Annotation current = annotation_list_object.head; for (size_t index = 0; current.link_node != annotation_list_object.tail_pointer && index < kMaxNumberOfAnnotations; ++index) { if (!process_reader_->Memory()->Read( current.link_node, sizeof(current), ¤t)) { LOG(WARNING) << "could not read annotation at index " << index << " in " << base::WideToUTF8(name_); return; } if (current.size == 0) { continue; } AnnotationSnapshot snapshot; snapshot.type = current.type; char name[Annotation::kNameMaxLength]; if (!process_reader_->Memory()->Read( current.name, base::size(name), name)) { LOG(WARNING) << "could not read annotation name at index " << index << " in " << base::WideToUTF8(name_); continue; } size_t name_length = strnlen(name, Annotation::kNameMaxLength); snapshot.name = std::string(name, name_length); size_t value_length = std::min(static_cast(current.size), Annotation::kValueMaxSize); snapshot.value.resize(value_length); if (!process_reader_->Memory()->Read( current.value, value_length, snapshot.value.data())) { LOG(WARNING) << "could not read annotation value at index " << index << " in " << base::WideToUTF8(name_); continue; } vector_annotations->push_back(std::move(snapshot)); } } } // namespace crashpad