Index: snapshot/minidump/process_snapshot_minidump.cc |
diff --git a/snapshot/minidump/process_snapshot_minidump.cc b/snapshot/minidump/process_snapshot_minidump.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..a882c7cdb4818201b16f0fbdcffdd104c2560e8e |
--- /dev/null |
+++ b/snapshot/minidump/process_snapshot_minidump.cc |
@@ -0,0 +1,189 @@ |
+// 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/process_snapshot_minidump.h" |
+ |
+#include "util/file/file_io.h" |
+#include "snapshot/minidump/minidump_simple_string_dictionary_reader.h" |
+ |
+namespace crashpad { |
+ |
+ProcessSnapshotMinidump::ProcessSnapshotMinidump() |
+ : ProcessSnapshot(), |
+ header_(), |
+ stream_directory_(), |
+ stream_map_(), |
+ crashpad_info_(), |
+ annotations_simple_map_(), |
+ file_reader_(nullptr), |
+ initialized_() { |
+} |
+ |
+ProcessSnapshotMinidump::~ProcessSnapshotMinidump() { |
+} |
+ |
+bool ProcessSnapshotMinidump::Initialize(FileReaderInterface* file_reader) { |
+ INITIALIZATION_STATE_SET_INITIALIZING(initialized_); |
+ |
+ file_reader_ = file_reader; |
+ |
+ if (!file_reader_->SeekSet(0)) { |
+ return false; |
+ } |
+ |
+ if (!file_reader_->ReadExactly(&header_, sizeof(header_))) { |
+ return false; |
+ } |
+ |
+ if (header_.Signature != MINIDUMP_SIGNATURE) { |
+ LOG(ERROR) << "minidump signature mismatch"; |
+ return false; |
+ } |
+ |
+ if (header_.Version != MINIDUMP_VERSION) { |
+ LOG(ERROR) << "minidump version mismatch"; |
+ return false; |
+ } |
+ |
+ if (!file_reader->SeekSet(header_.StreamDirectoryRva)) { |
+ return false; |
+ } |
+ |
+ stream_directory_.resize(header_.NumberOfStreams); |
+ if (!file_reader_->ReadExactly( |
+ &stream_directory_[0], |
+ header_.NumberOfStreams * sizeof(stream_directory_[0]))) { |
+ return false; |
+ } |
+ |
+ for (const MINIDUMP_DIRECTORY& directory : stream_directory_) { |
+ const MinidumpStreamType stream_type = |
+ static_cast<MinidumpStreamType>(directory.StreamType); |
+ if (stream_map_.find(stream_type) != stream_map_.end()) { |
+ LOG(ERROR) << "duplicate streams for type " << directory.StreamType; |
+ return false; |
+ } |
+ |
+ stream_map_[stream_type] = &directory.Location; |
+ } |
+ |
+ INITIALIZATION_STATE_SET_VALID(initialized_); |
+ |
+ InitializeCrashpadInfo(); |
+ |
+ return true; |
+} |
+ |
+pid_t ProcessSnapshotMinidump::ProcessID() const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ return 0; |
+} |
+ |
+pid_t ProcessSnapshotMinidump::ParentProcessID() const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ return 0; |
+} |
+ |
+void ProcessSnapshotMinidump::SnapshotTime(timeval* snapshot_time) const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ snapshot_time->tv_sec = 0; |
+ snapshot_time->tv_usec = 0; |
+} |
+ |
+void ProcessSnapshotMinidump::ProcessStartTime(timeval* start_time) const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ start_time->tv_sec = 0; |
+ start_time->tv_usec = 0; |
+} |
+ |
+void ProcessSnapshotMinidump::ProcessCPUTimes(timeval* user_time, |
+ timeval* system_time) const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ user_time->tv_sec = 0; |
+ user_time->tv_usec = 0; |
+ system_time->tv_sec = 0; |
+ system_time->tv_usec = 0; |
+} |
+ |
+const std::map<std::string, std::string>& |
+ProcessSnapshotMinidump::AnnotationsSimpleMap() const { |
+ // TODO(mark): This method should not be const, although the interface |
+ // currently imposes this requirement. Making it non-const would allow |
+ // annotations_simple_map_ to be lazily constructed: InitializeCrashpadInfo() |
+ // could be called here, and from other locations that require it, rather than |
+ // calling it from Initialize(). |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ return annotations_simple_map_; |
+} |
+ |
+const SystemSnapshot* ProcessSnapshotMinidump::System() const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ return nullptr; |
+} |
+ |
+std::vector<const ThreadSnapshot*> ProcessSnapshotMinidump::Threads() const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ return std::vector<const ThreadSnapshot*>(); |
+} |
+ |
+std::vector<const ModuleSnapshot*> ProcessSnapshotMinidump::Modules() const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ return std::vector<const ModuleSnapshot*>(); |
+} |
+ |
+const ExceptionSnapshot* ProcessSnapshotMinidump::Exception() const { |
+ INITIALIZATION_STATE_DCHECK_VALID(initialized_); |
+ NOTREACHED(); |
+ return nullptr; |
+} |
+ |
+void ProcessSnapshotMinidump::InitializeCrashpadInfo() { |
+ const auto& it = stream_map_.find(kMinidumpStreamTypeCrashpadInfo); |
+ if (it == stream_map_.end()) { |
+ return; |
+ } |
+ |
+ if (it->second->DataSize < sizeof(crashpad_info_)) { |
+ LOG(ERROR) << "crashpad_info size mismatch"; |
+ return; |
+ } |
+ |
+ if (!file_reader_->SeekSet(it->second->Rva)) { |
+ return; |
+ } |
+ |
+ if (!file_reader_->ReadExactly(&crashpad_info_, sizeof(crashpad_info_))) { |
+ return; |
+ } |
+ |
+ if (crashpad_info_.version != MinidumpCrashpadInfo::kVersion) { |
+ LOG(ERROR) << "crashpad_info version mismatch"; |
+ return; |
+ } |
+ |
+ internal::ReadMinidumpSimpleStringDictionary( |
+ file_reader_, |
+ crashpad_info_.simple_annotations, |
+ &annotations_simple_map_); |
+} |
+ |
+} // namespace crashpad |