Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1017)

Unified Diff: src/processor/minidump.cc

Issue 1251593007: Add support for Linux memory mapping stream and remove ELF header usage (Closed) Base URL: http://google-breakpad.googlecode.com/svn/trunk/
Patch Set: Created 5 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: src/processor/minidump.cc
===================================================================
--- src/processor/minidump.cc (revision 1474)
+++ src/processor/minidump.cc (working copy)
@@ -55,6 +55,7 @@
#include <iostream>
#include <limits>
#include <map>
+#include <sstream>
#include <vector>
#include "processor/range_map-inl.h"
@@ -71,6 +72,7 @@
using std::istream;
using std::ifstream;
using std::numeric_limits;
+using std::stringstream;
using std::vector;
// Returns true iff |context_size| matches exactly one of the sizes of the
@@ -3972,8 +3974,274 @@
}
}
+//
+// MinidumpLinuxMaps
+//
+MinidumpLinuxMaps::MinidumpLinuxMaps(Minidump *minidump)
+ : MinidumpObject(minidump),
+ pathname_(),
+ raw_linux_maps_(NULL) {
+}
+
+MinidumpLinuxMaps::~MinidumpLinuxMaps() {
+ delete raw_linux_maps_;
+}
+
+bool MinidumpLinuxMaps::Read(vector<char> *data) {
ivanpe 2015/07/23 01:17:12 There is already an implementation of this in Chro
liuandrew 2015/07/23 16:14:50 How do I reuse the Chromium code? Do I just copy t
liuandrew 2015/07/24 23:25:36 Done.
+ valid_ = false;
+
+ // Convert data vector into stringstream.
+ string str = "";
+ for (unsigned int i = 0; i < data->size(); i++) {
+ str.push_back((*data)[i]);
+ }
+ stringstream stream(str);
+ delete data;
+ data = NULL;
+
+ // Tokenize the string stream.
+ string address, perms, offset, dev, inode;
+ stream >> address >> perms >> offset >> dev >> inode >> pathname_;
+ // Check if the tokens are filled. The pathname is optional.
+ if (address.size() <= 0
+ || perms.size() <= 0
+ || offset.size() <= 0
+ || dev.size() <= 0
+ || inode.size() <= 0) {
+ BPLOG(ERROR) << "Malformed input string: not enough tokens";
+ return false;
+ }
+
+ // Load base address and region size.
+ size_t address_delimiter_index = address.find("-");
+ if (address_delimiter_index == string::npos) {
+ BPLOG(ERROR) << "Address delimiter not found";
+ return false;
+ }
+ uint64_t end_address = 0;
+ stringstream(address.substr(0, address_delimiter_index))
+ >> std::hex
+ >> base_address_;
+ stringstream(address.substr(address_delimiter_index + 1, address.size()))
+ >> std::hex
+ >> end_address;
+ if (base_address_ > end_address) {
+ BPLOG(ERROR) << "Base address greater than end address: "
+ << base_address_
+ << " > "
+ << end_address;
+ return false;
+ }
+ region_size_ = end_address - base_address_;
+
+ // Load permissions.
+ if (perms.size() != 4) {
+ BPLOG(ERROR) << "Permission string incorrectly loaded";
+ return false;
+ }
+ if (perms[0] == 'r') {
+ is_readable_ = true;
+ } else if (perms[0] == '-') {
+ is_readable_ = false;
+ } else {
+ BPLOG(ERROR) << "Invalid permission character: " << perms[0];
+ return false;
+ }
+ if (perms[1] == 'w') {
+ is_writeable_ = true;
+ } else if (perms[1] == '-') {
+ is_writeable_ = false;
+ } else {
+ BPLOG(ERROR) << "Invalid permission character: " << perms[1];
+ return false;
+ }
+ if (perms[2] == 'x') {
+ is_executable_ = true;
+ } else if (perms[2] == '-') {
+ is_executable_ = false;
+ } else {
+ BPLOG(ERROR) << "Invalid permission character: " << perms[2];
+ return false;
+ }
+ if (perms[3] == 's') {
+ is_shared_ = true;
+ } else if (perms[3] == 'p') {
+ is_shared_ = false;
+ } else {
+ BPLOG(ERROR) << "Invalid permission character: " << perms[3];
+ return false;
+ }
+
+ // Load offset.
+ stringstream(offset) >> std::hex >> offset_;
+
+ // Load major and minor devices.
+ size_t device_delimiter_index = dev.find(":");
+ if (device_delimiter_index == string::npos) {
+ BPLOG(ERROR) << "Device delimiter not found";
+ return false;
+ }
+ uint64_t major_container, minor_container;
+ stringstream(dev.substr(0, device_delimiter_index))
+ >> std::hex
+ >> major_container;
+ stringstream(dev.substr(device_delimiter_index + 1, dev.size()))
+ >> std::hex
+ >> minor_container;
+ major_device_ = static_cast<uint8_t>(major_container);
+ minor_device_ = static_cast<uint8_t>(minor_container);
+
+ // Load inode.
+ stringstream(inode) >> inode_;
+
+ valid_ = true;
+ return true;
+}
+
+void MinidumpLinuxMaps::Print() {
+ if (!valid_) {
+ BPLOG(ERROR) << "MinidumpLinuxMaps cannot print invalid data";
+ return;
+ }
+ // Print the contents of raw_linux_maps_.
+ for (unsigned int i = 0; i < raw_linux_maps_->size(); i++) {
+ printf("%c", (*raw_linux_maps_)[i]);
+ }
+}
+
//
+// MinidumpLinuxMapsList
+//
+
+MinidumpLinuxMapsList::MinidumpLinuxMapsList(Minidump *minidump)
+ : MinidumpStream(minidump),
+ maps_(NULL),
+ maps_count_(0),
+ raw_linux_maps_list_() {
+}
+
+MinidumpLinuxMapsList::~MinidumpLinuxMapsList() {
+ for (unsigned int i = 0; i < maps_->size(); i++) {
+ delete (*maps_)[i];
+ }
+ delete maps_;
+}
+
+const MinidumpLinuxMaps *MinidumpLinuxMapsList::GetLinuxMapsForAddress(
+ uint64_t address) const {
+ if (!valid_) {
+ BPLOG(ERROR) << "Invalid MinidumpLinuxMapsList for GetLinuxMapsForAddress";
+ return NULL;
+ }
+
+ // Search every memory mapping.
+ for (unsigned int index = 0; index < maps_count_; index++) {
+ // Check if address is within bounds of the current memory region.
+ if ((*maps_)[index]->GetBase() <= address &&
+ (*maps_)[index]->GetBase() + (*maps_)[index]->GetSize() > address) {
+ return (*maps_)[index];
+ }
+ }
+
+ // No mapping encloses the memory address.
+ BPLOG(ERROR) << "MinidumpLinuxMapsList has no mapping at "
+ << HexString(address);
+ return NULL;
+}
+
+const MinidumpLinuxMaps *MinidumpLinuxMapsList::GetLinuxMapsAtIndex(
+ unsigned int index) const {
+ if (!valid_) {
+ BPLOG(ERROR) << "Invalid MinidumpLinuxMapsList for GetLinuxMapsAtIndex";
+ return NULL;
+ }
+
+ // Index out of bounds.
+ if (index >= maps_count_) {
+ BPLOG(ERROR) << "MinidumpLinuxMapsList index of out range: "
+ << index
+ << "/"
+ << maps_count_;
+ return NULL;
+ }
+ return (*maps_)[index];
+}
+
+bool MinidumpLinuxMapsList::Read(uint32_t expected_size) {
+ // Invalidate cached data.
+ delete maps_;
+ maps_ = NULL;
+ maps_count_ = 0;
+ raw_linux_maps_list_.clear();
+
+ valid_ = false;
+
+ // Load and check expected stream length.
+ uint32_t length = 0;
+ if (!minidump_->SeekToStreamType(MD_LINUX_MAPS, &length)) {
+ BPLOG(ERROR) << "MinidumpLinuxMapsList stream type not found";
+ return false;
+ }
+ if (expected_size != length) {
+ BPLOG(ERROR) << "MinidumpLinuxMapsList size mismatch: "
+ << expected_size
+ << " != "
+ << length;
+ return false;
+ }
+
+ // Read data from stream into raw_linux_maps_list_.
+ raw_linux_maps_list_.resize(length);
+ if (!minidump_->ReadBytes(&raw_linux_maps_list_[0], length)) {
+ BPLOG(ERROR) << "MinidumpLinuxMapsList failed to read bytes";
+ return false;
+ }
+ scoped_ptr<vector<char> > map_data(new vector<char>());
+ scoped_ptr<MinidumpLinuxMaps> map_entry(new MinidumpLinuxMaps(minidump_));
+ scoped_ptr<MinidumpLinuxMappings> maps(new MinidumpLinuxMappings());
+
+ // Load all of the raw list into a list of MinidumpLinuxMaps.
+ for (unsigned int i = 0; i < raw_linux_maps_list_.size(); i++) {
+ // Push character into data buffer.
+ map_data->push_back(raw_linux_maps_list_[i]);
+
+ // A newline marks the end of a mapping entry.
+ if (raw_linux_maps_list_[i] == '\n') {
+ // Read entry data into MinidumpLinuxMaps object.
+ if (!map_entry->Read(map_data.release())) {
+ BPLOG(ERROR) << "MinidumpLinuxMapsList failed to read entry";
+ return false;
+ }
+ // Add completed map entry into vector.
+ maps->push_back(map_entry.release());
+
+ // Reset scoped pointers.
+ map_data.reset(new vector<char>());
+ map_entry.reset(new MinidumpLinuxMaps(minidump_));
+ }
+ }
+
+ // Set instance variables.
+ maps_ = maps.release();
+ maps_count_ = maps_->size();
+ valid_ = true;
+ return true;
+}
+
+void MinidumpLinuxMapsList::Print() {
+ if (!valid_) {
+ BPLOG(ERROR) << "MinidumpLinuxMapsList cannot print valid data";
+ return;
+ }
+
+ // Print contents of raw_linux_maps_list_.
+ for (unsigned int i = 0; i < raw_linux_maps_list_.size(); i++) {
+ printf("%c", raw_linux_maps_list_[i]);
+ }
+}
+
+//
// Minidump
//
@@ -4292,6 +4560,11 @@
return GetStream(&memory_info_list);
}
+MinidumpLinuxMapsList *Minidump::GetLinuxMapsList() {
+ MinidumpLinuxMapsList *linux_maps_list;
+ return GetStream(&linux_maps_list);
+}
+
static const char* get_stream_name(uint32_t stream_type) {
switch (stream_type) {
case MD_UNUSED_STREAM:

Powered by Google App Engine
This is Rietveld 408576698