183e085b7a331c96237cf8e814f97b3ef4c36a70fjimblandy// Copyright (c) 2010 Google Inc. 27daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// All rights reserved. 33261e8b6eac44a41341f112821482bee6c940c98mmentovai// 47daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// Redistribution and use in source and binary forms, with or without 57daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// modification, are permitted provided that the following conditions are 67daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// met: 73261e8b6eac44a41341f112821482bee6c940c98mmentovai// 87daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// * Redistributions of source code must retain the above copyright 97daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// notice, this list of conditions and the following disclaimer. 107daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// * Redistributions in binary form must reproduce the above 117daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// copyright notice, this list of conditions and the following disclaimer 127daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// in the documentation and/or other materials provided with the 137daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// distribution. 147daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// * Neither the name of Google Inc. nor the names of its 157daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// contributors may be used to endorse or promote products derived from 167daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// this software without specific prior written permission. 173261e8b6eac44a41341f112821482bee6c940c98mmentovai// 187daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 197daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 207daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 217daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 227daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 237daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 247daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 257daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 267daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 277daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 287daf246e4baf0837e25429668cc23e92b6afe3b3mmentovai// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 293261e8b6eac44a41341f112821482bee6c940c98mmentovai 303261e8b6eac44a41341f112821482bee6c940c98mmentovai// minidump.cc: A minidump reader. 313261e8b6eac44a41341f112821482bee6c940c98mmentovai// 323261e8b6eac44a41341f112821482bee6c940c98mmentovai// See minidump.h for documentation. 333261e8b6eac44a41341f112821482bee6c940c98mmentovai// 343261e8b6eac44a41341f112821482bee6c940c98mmentovai// Author: Mark Mentovai 353261e8b6eac44a41341f112821482bee6c940c98mmentovai 36e1930985430ce289f4fe8525f51050e5d78cc44eted.mielczarek#include "google_breakpad/processor/minidump.h" 373261e8b6eac44a41341f112821482bee6c940c98mmentovai 38e1930985430ce289f4fe8525f51050e5d78cc44eted.mielczarek#include <assert.h> 396dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai#include <fcntl.h> 40f2e56b1fde88a521971cb6e74d53ce6ac04aa186ted.mielczarek@gmail.com#include <stddef.h> 413261e8b6eac44a41341f112821482bee6c940c98mmentovai#include <stdio.h> 429fcf4db315427032a92078d1212fece2655bf049mmentovai#include <string.h> 433261e8b6eac44a41341f112821482bee6c940c98mmentovai#include <time.h> 44c77fc8a32c8af421c7e0d5fe81eccfb62329dff2ted.mielczarek 453261e8b6eac44a41341f112821482bee6c940c98mmentovai#ifdef _WIN32 463261e8b6eac44a41341f112821482bee6c940c98mmentovai#include <io.h> 47c77fc8a32c8af421c7e0d5fe81eccfb62329dff2ted.mielczarek#define PRIx64 "llx" 48c77fc8a32c8af421c7e0d5fe81eccfb62329dff2ted.mielczarek#define PRIx32 "lx" 49c77fc8a32c8af421c7e0d5fe81eccfb62329dff2ted.mielczarek#define snprintf _snprintf 5080e98391dc7ff361355e72c24c0fb222518bcdfcmmentovai#else // _WIN32 51c77fc8a32c8af421c7e0d5fe81eccfb62329dff2ted.mielczarek#include <unistd.h> 5280e98391dc7ff361355e72c24c0fb222518bcdfcmmentovai#endif // _WIN32 533261e8b6eac44a41341f112821482bee6c940c98mmentovai 540cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek#include <fstream> 550cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek#include <iostream> 56fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai#include <limits> 573261e8b6eac44a41341f112821482bee6c940c98mmentovai#include <map> 583261e8b6eac44a41341f112821482bee6c940c98mmentovai#include <vector> 593261e8b6eac44a41341f112821482bee6c940c98mmentovai 608c2a4def4ecfbf6293b27eff4359a274e9774b4emmentovai#include "processor/range_map-inl.h" 6180e98391dc7ff361355e72c24c0fb222518bcdfcmmentovai 622cc15ba4327831f917ff55b87e6d5fc3c7750085ted.mielczarek@gmail.com#include "common/scoped_ptr.h" 63c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org#include "google_breakpad/processor/dump_context.h" 64db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai#include "processor/basic_code_module.h" 65db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai#include "processor/basic_code_modules.h" 66af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai#include "processor/logging.h" 672cc15ba4327831f917ff55b87e6d5fc3c7750085ted.mielczarek@gmail.com 68e5dc60822e5938fea2ae892ccddb906641ba174emmentovainamespace google_breakpad { 693261e8b6eac44a41341f112821482bee6c940c98mmentovai 703261e8b6eac44a41341f112821482bee6c940c98mmentovai 710cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarekusing std::istream; 720cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarekusing std::ifstream; 73fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovaiusing std::numeric_limits; 743261e8b6eac44a41341f112821482bee6c940c98mmentovaiusing std::vector; 753261e8b6eac44a41341f112821482bee6c940c98mmentovai 7639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org// Returns true iff |context_size| matches exactly one of the sizes of the 7739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org// various MDRawContext* types. 7839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org// TODO(blundell): This function can be removed once 7939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org// http://code.google.com/p/google-breakpad/issues/detail?id=550 is fixed. 8039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.orgstatic bool IsContextSizeUnique(uint32_t context_size) { 8139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org int num_matching_contexts = 0; 8239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (context_size == sizeof(MDRawContextX86)) 8339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org num_matching_contexts++; 8439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (context_size == sizeof(MDRawContextPPC)) 8539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org num_matching_contexts++; 8639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (context_size == sizeof(MDRawContextPPC64)) 8739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org num_matching_contexts++; 8839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (context_size == sizeof(MDRawContextAMD64)) 8939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org num_matching_contexts++; 9039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (context_size == sizeof(MDRawContextSPARC)) 9139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org num_matching_contexts++; 9239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (context_size == sizeof(MDRawContextARM)) 9339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org num_matching_contexts++; 9439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (context_size == sizeof(MDRawContextARM64)) 9539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org num_matching_contexts++; 9639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (context_size == sizeof(MDRawContextMIPS)) 9739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org num_matching_contexts++; 9839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return num_matching_contexts == 1; 9939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org} 1003261e8b6eac44a41341f112821482bee6c940c98mmentovai 1013261e8b6eac44a41341f112821482bee6c940c98mmentovai// 1023261e8b6eac44a41341f112821482bee6c940c98mmentovai// Swapping routines 1033261e8b6eac44a41341f112821482bee6c940c98mmentovai// 1043261e8b6eac44a41341f112821482bee6c940c98mmentovai// Inlining these doesn't increase code size significantly, and it saves 1053261e8b6eac44a41341f112821482bee6c940c98mmentovai// a whole lot of unnecessary jumping back and forth. 1063261e8b6eac44a41341f112821482bee6c940c98mmentovai// 1073261e8b6eac44a41341f112821482bee6c940c98mmentovai 1083261e8b6eac44a41341f112821482bee6c940c98mmentovai 1093261e8b6eac44a41341f112821482bee6c940c98mmentovai// Swapping an 8-bit quantity is a no-op. This function is only provided 1103261e8b6eac44a41341f112821482bee6c940c98mmentovai// to account for certain templatized operations that require swapping for 1116162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com// wider types but handle uint8_t too 1123261e8b6eac44a41341f112821482bee6c940c98mmentovai// (MinidumpMemoryRegion::GetMemoryAtAddressInternal). 1136162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comstatic inline void Swap(uint8_t* value) { 1143261e8b6eac44a41341f112821482bee6c940c98mmentovai} 1153261e8b6eac44a41341f112821482bee6c940c98mmentovai 1163261e8b6eac44a41341f112821482bee6c940c98mmentovai 1173261e8b6eac44a41341f112821482bee6c940c98mmentovai// Optimization: don't need to AND the furthest right shift, because we're 1183261e8b6eac44a41341f112821482bee6c940c98mmentovai// shifting an unsigned quantity. The standard requires zero-filling in this 1193261e8b6eac44a41341f112821482bee6c940c98mmentovai// case. If the quantities were signed, a bitmask whould be needed for this 1203261e8b6eac44a41341f112821482bee6c940c98mmentovai// right shift to avoid an arithmetic shift (which retains the sign bit). 1213261e8b6eac44a41341f112821482bee6c940c98mmentovai// The furthest left shift never needs to be ANDed bitmask. 1223261e8b6eac44a41341f112821482bee6c940c98mmentovai 1233261e8b6eac44a41341f112821482bee6c940c98mmentovai 1246162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comstatic inline void Swap(uint16_t* value) { 1253261e8b6eac44a41341f112821482bee6c940c98mmentovai *value = (*value >> 8) | 1263261e8b6eac44a41341f112821482bee6c940c98mmentovai (*value << 8); 1273261e8b6eac44a41341f112821482bee6c940c98mmentovai} 1283261e8b6eac44a41341f112821482bee6c940c98mmentovai 1293261e8b6eac44a41341f112821482bee6c940c98mmentovai 1306162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comstatic inline void Swap(uint32_t* value) { 1313261e8b6eac44a41341f112821482bee6c940c98mmentovai *value = (*value >> 24) | 1323261e8b6eac44a41341f112821482bee6c940c98mmentovai ((*value >> 8) & 0x0000ff00) | 1333261e8b6eac44a41341f112821482bee6c940c98mmentovai ((*value << 8) & 0x00ff0000) | 1343261e8b6eac44a41341f112821482bee6c940c98mmentovai (*value << 24); 1353261e8b6eac44a41341f112821482bee6c940c98mmentovai} 1363261e8b6eac44a41341f112821482bee6c940c98mmentovai 1373261e8b6eac44a41341f112821482bee6c940c98mmentovai 1386162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comstatic inline void Swap(uint64_t* value) { 1396162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t* value32 = reinterpret_cast<uint32_t*>(value); 1402e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai Swap(&value32[0]); 1412e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai Swap(&value32[1]); 1426162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t temp = value32[0]; 1432e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai value32[0] = value32[1]; 1442e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai value32[1] = temp; 1453261e8b6eac44a41341f112821482bee6c940c98mmentovai} 1463261e8b6eac44a41341f112821482bee6c940c98mmentovai 1473261e8b6eac44a41341f112821482bee6c940c98mmentovai 1481d78cad82e3c7aa2315ed7438211a1901a91ed34bryner// Given a pointer to a 128-bit int in the minidump data, set the "low" 1491d78cad82e3c7aa2315ed7438211a1901a91ed34bryner// and "high" fields appropriately. 1506162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comstatic void Normalize128(uint128_struct* value, bool is_big_endian) { 1511d78cad82e3c7aa2315ed7438211a1901a91ed34bryner // The struct format is [high, low], so if the format is big-endian, 1521d78cad82e3c7aa2315ed7438211a1901a91ed34bryner // the most significant bytes will already be in the high field. 1531d78cad82e3c7aa2315ed7438211a1901a91ed34bryner if (!is_big_endian) { 1546162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t temp = value->low; 1551d78cad82e3c7aa2315ed7438211a1901a91ed34bryner value->low = value->high; 1561d78cad82e3c7aa2315ed7438211a1901a91ed34bryner value->high = temp; 1571d78cad82e3c7aa2315ed7438211a1901a91ed34bryner } 1581d78cad82e3c7aa2315ed7438211a1901a91ed34bryner} 1593402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 1601d78cad82e3c7aa2315ed7438211a1901a91ed34bryner// This just swaps each int64 half of the 128-bit value. 1611d78cad82e3c7aa2315ed7438211a1901a91ed34bryner// The value should also be normalized by calling Normalize128(). 1626162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comstatic void Swap(uint128_struct* value) { 1631d78cad82e3c7aa2315ed7438211a1901a91ed34bryner Swap(&value->low); 1641d78cad82e3c7aa2315ed7438211a1901a91ed34bryner Swap(&value->high); 1653402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai} 1663402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 1673562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com// Swapping signed integers 1683562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.comstatic inline void Swap(int16_t* value) { 1693562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(reinterpret_cast<uint16_t*>(value)); 1703562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com} 1713562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com 1723562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.comstatic inline void Swap(int32_t* value) { 1733562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(reinterpret_cast<uint32_t*>(value)); 1743562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com} 1753562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com 1763562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.comstatic inline void Swap(int64_t* value) { 1773562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(reinterpret_cast<uint64_t*>(value)); 1783562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com} 1793562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com 1803402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 1813261e8b6eac44a41341f112821482bee6c940c98mmentovaistatic inline void Swap(MDLocationDescriptor* location_descriptor) { 1823261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&location_descriptor->data_size); 1833261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&location_descriptor->rva); 1843261e8b6eac44a41341f112821482bee6c940c98mmentovai} 1853261e8b6eac44a41341f112821482bee6c940c98mmentovai 1863261e8b6eac44a41341f112821482bee6c940c98mmentovai 1873261e8b6eac44a41341f112821482bee6c940c98mmentovaistatic inline void Swap(MDMemoryDescriptor* memory_descriptor) { 1883261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&memory_descriptor->start_of_memory_range); 1893261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&memory_descriptor->memory); 1903261e8b6eac44a41341f112821482bee6c940c98mmentovai} 1913261e8b6eac44a41341f112821482bee6c940c98mmentovai 1923261e8b6eac44a41341f112821482bee6c940c98mmentovai 1933261e8b6eac44a41341f112821482bee6c940c98mmentovaistatic inline void Swap(MDGUID* guid) { 1943261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&guid->data1); 1953261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&guid->data2); 1963261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&guid->data3); 1973261e8b6eac44a41341f112821482bee6c940c98mmentovai // Don't swap guid->data4[] because it contains 8-bit quantities. 1983261e8b6eac44a41341f112821482bee6c940c98mmentovai} 1993261e8b6eac44a41341f112821482bee6c940c98mmentovai 2003562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.comstatic inline void Swap(MDSystemTime* system_time) { 2013562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&system_time->year); 2023562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&system_time->month); 2033562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&system_time->day_of_week); 2043562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&system_time->day); 2053562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&system_time->hour); 2063562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&system_time->minute); 2073562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&system_time->second); 2083562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&system_time->milliseconds); 2093562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com} 2103562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com 2113562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.comstatic inline void Swap(uint16_t* data, size_t size_in_bytes) { 2123562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com size_t data_length = size_in_bytes / sizeof(data[0]); 2133562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com for (size_t i = 0; i < data_length; i++) { 2143562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&data[i]); 2153562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 2163562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com} 2173261e8b6eac44a41341f112821482bee6c940c98mmentovai 2183261e8b6eac44a41341f112821482bee6c940c98mmentovai// 2193261e8b6eac44a41341f112821482bee6c940c98mmentovai// Character conversion routines 2203261e8b6eac44a41341f112821482bee6c940c98mmentovai// 2213261e8b6eac44a41341f112821482bee6c940c98mmentovai 2223261e8b6eac44a41341f112821482bee6c940c98mmentovai 2233261e8b6eac44a41341f112821482bee6c940c98mmentovai// Standard wide-character conversion routines depend on the system's own 2243261e8b6eac44a41341f112821482bee6c940c98mmentovai// idea of what width a wide character should be: some use 16 bits, and 2253261e8b6eac44a41341f112821482bee6c940c98mmentovai// some use 32 bits. For the purposes of a minidump, wide strings are 2263261e8b6eac44a41341f112821482bee6c940c98mmentovai// always represented with 16-bit UTF-16 chracters. iconv isn't available 2273261e8b6eac44a41341f112821482bee6c940c98mmentovai// everywhere, and its interface varies where it is available. iconv also 2283261e8b6eac44a41341f112821482bee6c940c98mmentovai// deals purely with char* pointers, so in addition to considering the swap 2293261e8b6eac44a41341f112821482bee6c940c98mmentovai// parameter, a converter that uses iconv would also need to take the host 2303261e8b6eac44a41341f112821482bee6c940c98mmentovai// CPU's endianness into consideration. It doesn't seems worth the trouble 2313261e8b6eac44a41341f112821482bee6c940c98mmentovai// of making it a dependency when we don't care about anything but UTF-16. 2326162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comstatic string* UTF16ToUTF8(const vector<uint16_t>& in, 2333562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com bool swap) { 2342466d8e993a800a17e00deda2f3a27e0505140e1mmentovai scoped_ptr<string> out(new string()); 2353261e8b6eac44a41341f112821482bee6c940c98mmentovai 2363261e8b6eac44a41341f112821482bee6c940c98mmentovai // Set the string's initial capacity to the number of UTF-16 characters, 2373261e8b6eac44a41341f112821482bee6c940c98mmentovai // because the UTF-8 representation will always be at least this long. 2383261e8b6eac44a41341f112821482bee6c940c98mmentovai // If the UTF-8 representation is longer, the string will grow dynamically. 2393261e8b6eac44a41341f112821482bee6c940c98mmentovai out->reserve(in.size()); 2403261e8b6eac44a41341f112821482bee6c940c98mmentovai 2416162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com for (vector<uint16_t>::const_iterator iterator = in.begin(); 2423261e8b6eac44a41341f112821482bee6c940c98mmentovai iterator != in.end(); 2433261e8b6eac44a41341f112821482bee6c940c98mmentovai ++iterator) { 2443261e8b6eac44a41341f112821482bee6c940c98mmentovai // Get a 16-bit value from the input 2456162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint16_t in_word = *iterator; 2463261e8b6eac44a41341f112821482bee6c940c98mmentovai if (swap) 2473261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&in_word); 2483261e8b6eac44a41341f112821482bee6c940c98mmentovai 2493261e8b6eac44a41341f112821482bee6c940c98mmentovai // Convert the input value (in_word) into a Unicode code point (unichar). 2506162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t unichar; 2513261e8b6eac44a41341f112821482bee6c940c98mmentovai if (in_word >= 0xdc00 && in_word <= 0xdcff) { 252af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "UTF16ToUTF8 found low surrogate " << 253af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(in_word) << " without high"; 2543261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2553261e8b6eac44a41341f112821482bee6c940c98mmentovai } else if (in_word >= 0xd800 && in_word <= 0xdbff) { 2563261e8b6eac44a41341f112821482bee6c940c98mmentovai // High surrogate. 2573261e8b6eac44a41341f112821482bee6c940c98mmentovai unichar = (in_word - 0xd7c0) << 10; 2583261e8b6eac44a41341f112821482bee6c940c98mmentovai if (++iterator == in.end()) { 259af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "UTF16ToUTF8 found high surrogate " << 260af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(in_word) << " at end of string"; 2613261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2623261e8b6eac44a41341f112821482bee6c940c98mmentovai } 2636162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t high_word = in_word; 2643261e8b6eac44a41341f112821482bee6c940c98mmentovai in_word = *iterator; 2653261e8b6eac44a41341f112821482bee6c940c98mmentovai if (in_word < 0xdc00 || in_word > 0xdcff) { 266af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "UTF16ToUTF8 found high surrogate " << 267af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(high_word) << " without low " << 268af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(in_word); 2693261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2703261e8b6eac44a41341f112821482bee6c940c98mmentovai } 2713261e8b6eac44a41341f112821482bee6c940c98mmentovai unichar |= in_word & 0x03ff; 2723261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 2733261e8b6eac44a41341f112821482bee6c940c98mmentovai // The ordinary case, a single non-surrogate Unicode character encoded 2743261e8b6eac44a41341f112821482bee6c940c98mmentovai // as a single 16-bit value. 2753261e8b6eac44a41341f112821482bee6c940c98mmentovai unichar = in_word; 2763261e8b6eac44a41341f112821482bee6c940c98mmentovai } 2773261e8b6eac44a41341f112821482bee6c940c98mmentovai 2783261e8b6eac44a41341f112821482bee6c940c98mmentovai // Convert the Unicode code point (unichar) into its UTF-8 representation, 2793261e8b6eac44a41341f112821482bee6c940c98mmentovai // appending it to the out string. 2803261e8b6eac44a41341f112821482bee6c940c98mmentovai if (unichar < 0x80) { 281f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += static_cast<char>(unichar); 2823261e8b6eac44a41341f112821482bee6c940c98mmentovai } else if (unichar < 0x800) { 283f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0xc0 | static_cast<char>(unichar >> 6); 284f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0x80 | static_cast<char>(unichar & 0x3f); 2853261e8b6eac44a41341f112821482bee6c940c98mmentovai } else if (unichar < 0x10000) { 286f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0xe0 | static_cast<char>(unichar >> 12); 287f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0x80 | static_cast<char>((unichar >> 6) & 0x3f); 288f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0x80 | static_cast<char>(unichar & 0x3f); 2893261e8b6eac44a41341f112821482bee6c940c98mmentovai } else if (unichar < 0x200000) { 290f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0xf0 | static_cast<char>(unichar >> 18); 291f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0x80 | static_cast<char>((unichar >> 12) & 0x3f); 292f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0x80 | static_cast<char>((unichar >> 6) & 0x3f); 293f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com (*out) += 0x80 | static_cast<char>(unichar & 0x3f); 2943261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 295af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "UTF16ToUTF8 cannot represent high value " << 296af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(unichar) << " in UTF-8"; 2973261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2983261e8b6eac44a41341f112821482bee6c940c98mmentovai } 2993261e8b6eac44a41341f112821482bee6c940c98mmentovai } 3003261e8b6eac44a41341f112821482bee6c940c98mmentovai 3013261e8b6eac44a41341f112821482bee6c940c98mmentovai return out.release(); 3023261e8b6eac44a41341f112821482bee6c940c98mmentovai} 3033261e8b6eac44a41341f112821482bee6c940c98mmentovai 3040314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek// Return the smaller of the number of code units in the UTF-16 string, 3050314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek// not including the terminating null word, or maxlen. 3066162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comstatic size_t UTF16codeunits(const uint16_t *string, size_t maxlen) { 3070314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek size_t count = 0; 3080314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek while (count < maxlen && string[count] != 0) 3090314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek count++; 3100314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek return count; 3110314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek} 3120314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 3133562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.comstatic inline void Swap(MDTimeZoneInformation* time_zone) { 3143562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&time_zone->bias); 3153562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Skip time_zone->standard_name. No need to swap UTF-16 fields. 3163562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // The swap will be done as part of the conversion to UTF-8. 3173562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&time_zone->standard_date); 3183562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&time_zone->standard_bias); 3193562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Skip time_zone->daylight_name. No need to swap UTF-16 fields. 3203562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // The swap will be done as part of the conversion to UTF-8. 3213562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&time_zone->daylight_date); 3223562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&time_zone->daylight_bias); 3233562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com} 3243562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com 3253562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.comstatic void ConvertUTF16BufferToUTF8String(const uint16_t* utf16_data, 3263562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com size_t max_length_in_bytes, 3273562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com string* utf8_result, 3283562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com bool swap) { 3293562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Since there is no explicit byte length for each string, use 3303562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // UTF16codeunits to calculate word length, then derive byte 3313562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // length from that. 3323562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com size_t max_word_length = max_length_in_bytes / sizeof(utf16_data[0]); 3333562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com size_t word_length = UTF16codeunits(utf16_data, max_word_length); 3343562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com if (word_length > 0) { 3353562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com size_t byte_length = word_length * sizeof(utf16_data[0]); 3363562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com vector<uint16_t> utf16_vector(word_length); 3373562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com memcpy(&utf16_vector[0], &utf16_data[0], byte_length); 3383562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com scoped_ptr<string> temp(UTF16ToUTF8(utf16_vector, swap)); 3393562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com if (temp.get()) { 3403562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com utf8_result->assign(*temp); 3413562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 3423562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } else { 3433562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com utf8_result->clear(); 3443562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 3453562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com} 3463261e8b6eac44a41341f112821482bee6c940c98mmentovai 3471b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org 3481b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org// For fields that may or may not be valid, PrintValueOrInvalid will print the 3491b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org// string "(invalid)" if the field is not valid, and will print the value if 3501b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org// the field is valid. The value is printed as hexadecimal or decimal. 3511b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org 3521b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.orgenum NumberFormat { 3531b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, 3541b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatHexadecimal, 3551b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org}; 3561b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org 3571b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.orgstatic void PrintValueOrInvalid(bool valid, 3581b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org NumberFormat number_format, 3591b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org uint32_t value) { 3601b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org if (!valid) { 3611b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf("(invalid)\n"); 3621b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } else if (number_format == kNumberFormatDecimal) { 3631b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf("%d\n", value); 3641b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } else { 3651b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf("0x%x\n", value); 3661b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } 3671b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org} 3681b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org 3699c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org// Converts a time_t to a string showing the time in UTC. 3709c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.orgstring TimeTToUTCString(time_t tt) { 3719c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org struct tm timestruct; 3729c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org#ifdef _WIN32 3739c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org gmtime_s(×truct, &tt); 3749c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org#else 3759c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org gmtime_r(&tt, ×truct); 3769c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org#endif 3779c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org 3789c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org char timestr[20]; 3799c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org int rv = strftime(timestr, 20, "%Y-%m-%d %H:%M:%S", ×truct); 3809c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org if (rv == 0) { 3819c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org return string(); 3829c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org } 3839c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org 3849c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org return string(timestr); 3859c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org} 3869c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org 3871b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org 3883261e8b6eac44a41341f112821482bee6c940c98mmentovai// 3893261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpObject 3903261e8b6eac44a41341f112821482bee6c940c98mmentovai// 3913261e8b6eac44a41341f112821482bee6c940c98mmentovai 3923261e8b6eac44a41341f112821482bee6c940c98mmentovai 3933261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpObject::MinidumpObject(Minidump* minidump) 394c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org : DumpObject(), 395c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org minidump_(minidump) { 3963261e8b6eac44a41341f112821482bee6c940c98mmentovai} 3973261e8b6eac44a41341f112821482bee6c940c98mmentovai 3983261e8b6eac44a41341f112821482bee6c940c98mmentovai 3993261e8b6eac44a41341f112821482bee6c940c98mmentovai// 4003261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpStream 4013261e8b6eac44a41341f112821482bee6c940c98mmentovai// 4023261e8b6eac44a41341f112821482bee6c940c98mmentovai 4033261e8b6eac44a41341f112821482bee6c940c98mmentovai 4043261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpStream::MinidumpStream(Minidump* minidump) 4053261e8b6eac44a41341f112821482bee6c940c98mmentovai : MinidumpObject(minidump) { 4063261e8b6eac44a41341f112821482bee6c940c98mmentovai} 4073261e8b6eac44a41341f112821482bee6c940c98mmentovai 4083261e8b6eac44a41341f112821482bee6c940c98mmentovai 4093261e8b6eac44a41341f112821482bee6c940c98mmentovai// 4103261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpContext 4113261e8b6eac44a41341f112821482bee6c940c98mmentovai// 4123261e8b6eac44a41341f112821482bee6c940c98mmentovai 4133261e8b6eac44a41341f112821482bee6c940c98mmentovai 4143261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpContext::MinidumpContext(Minidump* minidump) 415c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org : DumpContext(), 416c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org minidump_(minidump) { 4173261e8b6eac44a41341f112821482bee6c940c98mmentovai} 4183261e8b6eac44a41341f112821482bee6c940c98mmentovai 4193402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovaiMinidumpContext::~MinidumpContext() { 4203402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai} 4213402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 4226162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpContext::Read(uint32_t expected_size) { 4233261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 4243261e8b6eac44a41341f112821482bee6c940c98mmentovai 42539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // Certain raw context types are currently assumed to have unique sizes. 42639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (!IsContextSizeUnique(sizeof(MDRawContextAMD64))) { 42739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "sizeof(MDRawContextAMD64) cannot match the size of any " 42839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org << "other raw context"; 42939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 43039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 43139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (!IsContextSizeUnique(sizeof(MDRawContextPPC64))) { 43239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "sizeof(MDRawContextPPC64) cannot match the size of any " 43339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org << "other raw context"; 43439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 43539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 43639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (!IsContextSizeUnique(sizeof(MDRawContextARM64))) { 43739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "sizeof(MDRawContextARM64) cannot match the size of any " 43839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org << "other raw context"; 43939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 44039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 44139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 4423402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai FreeContext(); 4433261e8b6eac44a41341f112821482bee6c940c98mmentovai 4443402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai // First, figure out what type of CPU this context structure is for. 4458eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // For some reason, the AMD64 Context doesn't have context_flags 4468eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // at the beginning of the structure, so special case it here. 4478eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (expected_size == sizeof(MDRawContextAMD64)) { 4488eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(INFO) << "MinidumpContext: looks like AMD64 context"; 4498eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 4508eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek scoped_ptr<MDRawContextAMD64> context_amd64(new MDRawContextAMD64()); 4518eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!minidump_->ReadBytes(context_amd64.get(), 4528eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek sizeof(MDRawContextAMD64))) { 4538eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext could not read amd64 context"; 4548eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 4558eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 4563402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 4578eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (minidump_->swap()) 4588eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->context_flags); 4593402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 4606162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t cpu_type = context_amd64->context_flags & MD_CONTEXT_CPU_MASK; 461233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com if (cpu_type == 0) { 462233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com if (minidump_->GetContextCPUFlagsFromSystemInfo(&cpu_type)) { 463233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com context_amd64->context_flags |= cpu_type; 464233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } else { 465233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com BPLOG(ERROR) << "Failed to preserve the current stream position"; 466233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com return false; 467233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } 468233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } 4693402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 4708eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (cpu_type != MD_CONTEXT_AMD64) { 47139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // TODO: Fall through to switch below. 47239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // http://code.google.com/p/google-breakpad/issues/detail?id=550 4738eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext not actually amd64 context"; 4748eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 4758eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 4763402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 4778eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Do this after reading the entire MDRawContext structure because 4788eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // GetSystemInfo may seek minidump to a new position. 4798eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!CheckAgainstSystemInfo(cpu_type)) { 4808eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext amd64 does not match system info"; 4818eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 4828eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 4833261e8b6eac44a41341f112821482bee6c940c98mmentovai 4848eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Normalize the 128-bit types in the dump. 4858eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Since this is AMD64, by definition, the values are little-endian. 4868eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek for (unsigned int vr_index = 0; 4878eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek vr_index < MD_CONTEXT_AMD64_VR_COUNT; 4888eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek ++vr_index) 4898eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Normalize128(&context_amd64->vector_register[vr_index], false); 4908eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 4918eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (minidump_->swap()) { 4928eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->p1_home); 4938eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->p2_home); 4948eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->p3_home); 4958eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->p4_home); 4968eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->p5_home); 4978eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->p6_home); 4988eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // context_flags is already swapped 4998eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->mx_csr); 5008eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->cs); 5018eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->ds); 5028eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->es); 5038eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->fs); 5048eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->ss); 5058eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->eflags); 5068eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->dr0); 5078eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->dr1); 5088eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->dr2); 5098eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->dr3); 5108eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->dr6); 5118eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->dr7); 5128eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rax); 5138eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rcx); 5148eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rdx); 5158eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rbx); 5168eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rsp); 5178eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rbp); 5188eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rsi); 5198eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rdi); 5208eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->r8); 5218eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->r9); 5228eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->r10); 5238eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->r11); 5248eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->r12); 5258eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->r13); 5268eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->r14); 5278eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->r15); 5288eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->rip); 529f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // FIXME: I'm not sure what actually determines 5308eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // which member of the union {flt_save, sse_registers} 5318eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // is valid. We're not currently using either, 5328eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // but it would be good to have them swapped properly. 5333402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 5348eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek for (unsigned int vr_index = 0; 5358eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek vr_index < MD_CONTEXT_AMD64_VR_COUNT; 5368eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek ++vr_index) 5378eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->vector_register[vr_index]); 5388eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->vector_control); 5398eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->debug_control); 5408eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->last_branch_to_rip); 5418eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->last_branch_from_rip); 5428eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->last_exception_to_rip); 5438eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_amd64->last_exception_from_rip); 5448eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 5453402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 546c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextFlags(context_amd64->context_flags); 5473402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 548c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextAMD64(context_amd64.release()); 549f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com } else if (expected_size == sizeof(MDRawContextPPC64)) { 550f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // |context_flags| of MDRawContextPPC64 is 64 bits, but other MDRawContext 551f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // in the else case have 32 bits |context_flags|, so special case it here. 552cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org uint64_t context_flags; 553cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org if (!minidump_->ReadBytes(&context_flags, sizeof(context_flags))) { 554cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org BPLOG(ERROR) << "MinidumpContext could not read context flags"; 555cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org return false; 556cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org } 557cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org if (minidump_->swap()) 558cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_flags); 559cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org 560cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org uint32_t cpu_type = context_flags & MD_CONTEXT_CPU_MASK; 561cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org scoped_ptr<MDRawContextPPC64> context_ppc64(new MDRawContextPPC64()); 562cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org 56339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (cpu_type == 0) { 56439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (minidump_->GetContextCPUFlagsFromSystemInfo(&cpu_type)) { 56539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org context_ppc64->context_flags |= cpu_type; 56639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } else { 56739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "Failed to preserve the current stream position"; 56839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 56939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 57039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 57139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 57239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (cpu_type != MD_CONTEXT_PPC64) { 57339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // TODO: Fall through to switch below. 57439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // http://code.google.com/p/google-breakpad/issues/detail?id=550 57539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "MinidumpContext not actually ppc64 context"; 57639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 57739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 578f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com 579cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // Set the context_flags member, which has already been read, and 580cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // read the rest of the structure beginning with the first member 581cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // after context_flags. 582cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org context_ppc64->context_flags = context_flags; 583cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org 584cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org size_t flags_size = sizeof(context_ppc64->context_flags); 585cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org uint8_t* context_after_flags = 586cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org reinterpret_cast<uint8_t*>(context_ppc64.get()) + flags_size; 587cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org if (!minidump_->ReadBytes(context_after_flags, 588cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org sizeof(MDRawContextPPC64) - flags_size)) { 589cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org BPLOG(ERROR) << "MinidumpContext could not read ppc64 context"; 590cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org return false; 591cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org } 592cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org 593cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // Do this after reading the entire MDRawContext structure because 594cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // GetSystemInfo may seek minidump to a new position. 595cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org if (!CheckAgainstSystemInfo(cpu_type)) { 596cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org BPLOG(ERROR) << "MinidumpContext ppc64 does not match system info"; 597cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org return false; 598cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org } 599cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org if (minidump_->swap()) { 600cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // context_ppc64->context_flags was already swapped. 601cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->srr0); 602cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->srr1); 603cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org for (unsigned int gpr_index = 0; 604cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org gpr_index < MD_CONTEXT_PPC64_GPR_COUNT; 605cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org ++gpr_index) { 606cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->gpr[gpr_index]); 607cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org } 608cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->cr); 609cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->xer); 610cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->lr); 611cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->ctr); 612cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->vrsave); 613cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org for (unsigned int fpr_index = 0; 614cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org fpr_index < MD_FLOATINGSAVEAREA_PPC_FPR_COUNT; 615cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org ++fpr_index) { 616cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->float_save.fpregs[fpr_index]); 617cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org } 618cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // Don't swap context_ppc64->float_save.fpscr_pad because it is only 619cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // used for padding. 620cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->float_save.fpscr); 621cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org for (unsigned int vr_index = 0; 622cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org vr_index < MD_VECTORSAVEAREA_PPC_VR_COUNT; 623cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org ++vr_index) { 624cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Normalize128(&context_ppc64->vector_save.save_vr[vr_index], true); 625cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->vector_save.save_vr[vr_index]); 626cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org } 627cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->vector_save.save_vscr); 628cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org // Don't swap the padding fields in vector_save. 629cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org Swap(&context_ppc64->vector_save.save_vrvalid); 630cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org } 631cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org 632c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextFlags(static_cast<uint32_t>(context_ppc64->context_flags)); 633f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com 634f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // Check for data loss when converting context flags from uint64_t into 635f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // uint32_t 636c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org if (static_cast<uint64_t>(GetContextFlags()) != 637f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com context_ppc64->context_flags) { 638f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "Data loss detected when converting PPC64 context_flags"; 639f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com return false; 640f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com } 641cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org 642c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextPPC64(context_ppc64.release()); 64339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } else if (expected_size == sizeof(MDRawContextARM64)) { 64439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // |context_flags| of MDRawContextARM64 is 64 bits, but other MDRawContext 64539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // in the else case have 32 bits |context_flags|, so special case it here. 64639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org uint64_t context_flags; 64739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 64839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(INFO) << "MinidumpContext: looks like ARM64 context"; 64939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 65039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (!minidump_->ReadBytes(&context_flags, sizeof(context_flags))) { 65139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "MinidumpContext could not read context flags"; 65239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 65339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 65439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (minidump_->swap()) 65539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org Swap(&context_flags); 65639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 65739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org scoped_ptr<MDRawContextARM64> context_arm64(new MDRawContextARM64()); 65839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 65939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org uint32_t cpu_type = context_flags & MD_CONTEXT_CPU_MASK; 66039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (cpu_type == 0) { 66139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (minidump_->GetContextCPUFlagsFromSystemInfo(&cpu_type)) { 66239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org context_arm64->context_flags |= cpu_type; 66339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } else { 66439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "Failed to preserve the current stream position"; 66539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 66639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 66739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 66839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 66939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (cpu_type != MD_CONTEXT_ARM64) { 67039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // TODO: Fall through to switch below. 67139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // http://code.google.com/p/google-breakpad/issues/detail?id=550 67239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "MinidumpContext not actually arm64 context"; 67339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 67439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 67539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 67639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // Set the context_flags member, which has already been read, and 67739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // read the rest of the structure beginning with the first member 67839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // after context_flags. 67939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org context_arm64->context_flags = context_flags; 68039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 68139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org size_t flags_size = sizeof(context_arm64->context_flags); 68239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org uint8_t* context_after_flags = 68339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org reinterpret_cast<uint8_t*>(context_arm64.get()) + flags_size; 68439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (!minidump_->ReadBytes(context_after_flags, 68539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org sizeof(MDRawContextARM64) - flags_size)) { 68639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "MinidumpContext could not read arm64 context"; 68739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 68839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 68939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 69039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // Do this after reading the entire MDRawContext structure because 69139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // GetSystemInfo may seek minidump to a new position. 69239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (!CheckAgainstSystemInfo(cpu_type)) { 69339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org BPLOG(ERROR) << "MinidumpContext arm64 does not match system info"; 69439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return false; 69539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 69639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 69739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (minidump_->swap()) { 69839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // context_arm64->context_flags was already swapped. 69939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org for (unsigned int ireg_index = 0; 70039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org ireg_index < MD_CONTEXT_ARM64_GPR_COUNT; 70139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org ++ireg_index) { 70239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org Swap(&context_arm64->iregs[ireg_index]); 70339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 70439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org Swap(&context_arm64->cpsr); 70539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org Swap(&context_arm64->float_save.fpsr); 70639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org Swap(&context_arm64->float_save.fpcr); 70739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org for (unsigned int fpr_index = 0; 70839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org fpr_index < MD_FLOATINGSAVEAREA_ARM64_FPR_COUNT; 70939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org ++fpr_index) { 71039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // While ARM64 is bi-endian, iOS (currently the only platform 71139d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // for which ARM64 support has been brought up) uses ARM64 exclusively 71239d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org // in little-endian mode. 71339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org Normalize128(&context_arm64->float_save.regs[fpr_index], false); 71439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org Swap(&context_arm64->float_save.regs[fpr_index]); 71539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 71639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org } 717c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextFlags(static_cast<uint32_t>(context_arm64->context_flags)); 718ce1a45e28cad26f9c0f34ae329b0fd694d004f0fthestig@chromium.org 719ce1a45e28cad26f9c0f34ae329b0fd694d004f0fthestig@chromium.org // Check for data loss when converting context flags from uint64_t into 720ce1a45e28cad26f9c0f34ae329b0fd694d004f0fthestig@chromium.org // uint32_t 721c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org if (static_cast<uint64_t>(GetContextFlags()) != 722ce1a45e28cad26f9c0f34ae329b0fd694d004f0fthestig@chromium.org context_arm64->context_flags) { 723ce1a45e28cad26f9c0f34ae329b0fd694d004f0fthestig@chromium.org BPLOG(ERROR) << "Data loss detected when converting ARM64 context_flags"; 724ce1a45e28cad26f9c0f34ae329b0fd694d004f0fthestig@chromium.org return false; 725ce1a45e28cad26f9c0f34ae329b0fd694d004f0fthestig@chromium.org } 726ce1a45e28cad26f9c0f34ae329b0fd694d004f0fthestig@chromium.org 727c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextARM64(context_arm64.release()); 728f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com } else { 7296162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t context_flags; 7308eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!minidump_->ReadBytes(&context_flags, sizeof(context_flags))) { 7318eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext could not read context flags"; 7328eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 7333402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } 7348eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (minidump_->swap()) 7358eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_flags); 7368eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 7376162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t cpu_type = context_flags & MD_CONTEXT_CPU_MASK; 7381a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek if (cpu_type == 0) { 7391a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek // Unfortunately the flag for MD_CONTEXT_ARM that was taken 7401a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek // from a Windows CE SDK header conflicts in practice with 7411a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek // the CONTEXT_XSTATE flag. MD_CONTEXT_ARM has been renumbered, 7421a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek // but handle dumps with the legacy value gracefully here. 7431a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek if (context_flags & MD_CONTEXT_ARM_OLD) { 7441a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek context_flags |= MD_CONTEXT_ARM; 7451a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek context_flags &= ~MD_CONTEXT_ARM_OLD; 7461a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek cpu_type = MD_CONTEXT_ARM; 7471a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek } 7481a1890a52aaf8bfbea34a8d918423e6c81f3ea80ted.mielczarek } 7498eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 750233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com if (cpu_type == 0) { 751233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com if (minidump_->GetContextCPUFlagsFromSystemInfo(&cpu_type)) { 752233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com context_flags |= cpu_type; 753233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } else { 754233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com BPLOG(ERROR) << "Failed to preserve the current stream position"; 755233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com return false; 756233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } 757233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } 758233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com 7598eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Allocate the context structure for the correct CPU and fill it. The 7608eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // casts are slightly unorthodox, but it seems better to do that than to 7618eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // maintain a separate pointer for each type of CPU context structure 7628eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // when only one of them will be used. 7638eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek switch (cpu_type) { 7648eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek case MD_CONTEXT_X86: { 7658eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (expected_size != sizeof(MDRawContextX86)) { 7668eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext x86 size mismatch, " << 7678eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek expected_size << " != " << sizeof(MDRawContextX86); 7688eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 7698eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 7703402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 7718eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek scoped_ptr<MDRawContextX86> context_x86(new MDRawContextX86()); 7723402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 7738eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Set the context_flags member, which has already been read, and 7748eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // read the rest of the structure beginning with the first member 7758eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // after context_flags. 7768eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek context_x86->context_flags = context_flags; 7773402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 7788eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek size_t flags_size = sizeof(context_x86->context_flags); 7796162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint8_t* context_after_flags = 7806162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com reinterpret_cast<uint8_t*>(context_x86.get()) + flags_size; 7818eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!minidump_->ReadBytes(context_after_flags, 7828eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek sizeof(MDRawContextX86) - flags_size)) { 7838eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext could not read x86 context"; 7848eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 7858eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 7863402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 7878eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Do this after reading the entire MDRawContext structure because 7888eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // GetSystemInfo may seek minidump to a new position. 7898eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!CheckAgainstSystemInfo(cpu_type)) { 7908eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext x86 does not match system info"; 7918eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 7928eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 7933402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 7948eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (minidump_->swap()) { 7958eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // context_x86->context_flags was already swapped. 7968eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->dr0); 7978eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->dr1); 7988eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->dr2); 7998eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->dr3); 8008eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->dr6); 8018eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->dr7); 8028eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->float_save.control_word); 8038eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->float_save.status_word); 8048eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->float_save.tag_word); 8058eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->float_save.error_offset); 8068eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->float_save.error_selector); 8078eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->float_save.data_offset); 8088eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->float_save.data_selector); 8098eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // context_x86->float_save.register_area[] contains 8-bit quantities 8108eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // and does not need to be swapped. 8118eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->float_save.cr0_npx_state); 8128eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->gs); 8138eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->fs); 8148eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->es); 8158eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->ds); 8168eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->edi); 8178eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->esi); 8188eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->ebx); 8198eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->edx); 8208eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->ecx); 8218eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->eax); 8228eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->ebp); 8238eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->eip); 8248eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->cs); 8258eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->eflags); 8268eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->esp); 8278eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_x86->ss); 8288eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // context_x86->extended_registers[] contains 8-bit quantities and 8298eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // does not need to be swapped. 8308eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 8313402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 832c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextX86(context_x86.release()); 8338eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 8348eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek break; 8351d78cad82e3c7aa2315ed7438211a1901a91ed34bryner } 8361d78cad82e3c7aa2315ed7438211a1901a91ed34bryner 8378eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek case MD_CONTEXT_PPC: { 8388eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (expected_size != sizeof(MDRawContextPPC)) { 8398eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext ppc size mismatch, " << 8408eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek expected_size << " != " << sizeof(MDRawContextPPC); 8418eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 8428eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 8438eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 8448eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek scoped_ptr<MDRawContextPPC> context_ppc(new MDRawContextPPC()); 8458eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 8468eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Set the context_flags member, which has already been read, and 8478eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // read the rest of the structure beginning with the first member 8488eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // after context_flags. 8498eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek context_ppc->context_flags = context_flags; 8508eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 8518eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek size_t flags_size = sizeof(context_ppc->context_flags); 8526162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint8_t* context_after_flags = 8536162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com reinterpret_cast<uint8_t*>(context_ppc.get()) + flags_size; 8548eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!minidump_->ReadBytes(context_after_flags, 8558eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek sizeof(MDRawContextPPC) - flags_size)) { 8568eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext could not read ppc context"; 8578eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 8583402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } 8598eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 8608eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Do this after reading the entire MDRawContext structure because 8618eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // GetSystemInfo may seek minidump to a new position. 8628eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!CheckAgainstSystemInfo(cpu_type)) { 8638eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext ppc does not match system info"; 8648eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 8653402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } 8668eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 8678eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Normalize the 128-bit types in the dump. 8688eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Since this is PowerPC, by definition, the values are big-endian. 8693402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai for (unsigned int vr_index = 0; 8703402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai vr_index < MD_VECTORSAVEAREA_PPC_VR_COUNT; 8713402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai ++vr_index) { 8728eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Normalize128(&context_ppc->vector_save.save_vr[vr_index], true); 8733402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } 8743402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 8758eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (minidump_->swap()) { 8768eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // context_ppc->context_flags was already swapped. 8778eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->srr0); 8788eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->srr1); 8798eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek for (unsigned int gpr_index = 0; 8808eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek gpr_index < MD_CONTEXT_PPC_GPR_COUNT; 8818eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek ++gpr_index) { 8828eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->gpr[gpr_index]); 8838eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 8848eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->cr); 8858eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->xer); 8868eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->lr); 8878eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->ctr); 8888eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->mq); 8898eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->vrsave); 8908eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek for (unsigned int fpr_index = 0; 8918eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek fpr_index < MD_FLOATINGSAVEAREA_PPC_FPR_COUNT; 8928eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek ++fpr_index) { 8938eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->float_save.fpregs[fpr_index]); 8948eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 8958eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Don't swap context_ppc->float_save.fpscr_pad because it is only 8968eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // used for padding. 8978eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->float_save.fpscr); 8988eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek for (unsigned int vr_index = 0; 8998eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek vr_index < MD_VECTORSAVEAREA_PPC_VR_COUNT; 9008eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek ++vr_index) { 9018eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->vector_save.save_vr[vr_index]); 9028eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 9038eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->vector_save.save_vscr); 9048eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Don't swap the padding fields in vector_save. 9058eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_ppc->vector_save.save_vrvalid); 9068eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 9073402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 908c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextPPC(context_ppc.release()); 9093402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 9108eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek break; 911ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 912ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 9138eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek case MD_CONTEXT_SPARC: { 9148eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (expected_size != sizeof(MDRawContextSPARC)) { 9158eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext sparc size mismatch, " << 9168eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek expected_size << " != " << sizeof(MDRawContextSPARC); 9178eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 9188eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 919ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 9208eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek scoped_ptr<MDRawContextSPARC> context_sparc(new MDRawContextSPARC()); 921ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 9228eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Set the context_flags member, which has already been read, and 9238eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // read the rest of the structure beginning with the first member 9248eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // after context_flags. 9258eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek context_sparc->context_flags = context_flags; 926ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 9278eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek size_t flags_size = sizeof(context_sparc->context_flags); 9286162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint8_t* context_after_flags = 9296162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com reinterpret_cast<uint8_t*>(context_sparc.get()) + flags_size; 9308eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!minidump_->ReadBytes(context_after_flags, 9318eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek sizeof(MDRawContextSPARC) - flags_size)) { 9328eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext could not read sparc context"; 9338eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 9348eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 935ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 9368eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // Do this after reading the entire MDRawContext structure because 9378eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // GetSystemInfo may seek minidump to a new position. 9388eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (!CheckAgainstSystemInfo(cpu_type)) { 9398eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek BPLOG(ERROR) << "MinidumpContext sparc does not match system info"; 9408eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 941ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 9428eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 9438eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (minidump_->swap()) { 9448eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek // context_sparc->context_flags was already swapped. 9458eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek for (unsigned int gpr_index = 0; 9468eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek gpr_index < MD_CONTEXT_SPARC_GPR_COUNT; 9478eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek ++gpr_index) { 9488eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->g_r[gpr_index]); 9498eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 9508eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->ccr); 9518eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->pc); 9528eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->npc); 9538eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->y); 9548eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->asi); 9558eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->fprs); 9568eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek for (unsigned int fpr_index = 0; 9578eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek fpr_index < MD_FLOATINGSAVEAREA_SPARC_FPR_COUNT; 9588eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek ++fpr_index) { 9598eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->float_save.regs[fpr_index]); 9608eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 9618eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->float_save.filler); 9628eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek Swap(&context_sparc->float_save.fsr); 963ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 964c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextSPARC(context_sparc.release()); 965ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 9668eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek break; 9678eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 968ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 9699276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek case MD_CONTEXT_ARM: { 9709276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek if (expected_size != sizeof(MDRawContextARM)) { 9719276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek BPLOG(ERROR) << "MinidumpContext arm size mismatch, " << 9729276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek expected_size << " != " << sizeof(MDRawContextARM); 9739276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek return false; 9749276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek } 9759276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 9769276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek scoped_ptr<MDRawContextARM> context_arm(new MDRawContextARM()); 9779276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 9789276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek // Set the context_flags member, which has already been read, and 9799276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek // read the rest of the structure beginning with the first member 9809276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek // after context_flags. 9819276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek context_arm->context_flags = context_flags; 9829276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 9839276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek size_t flags_size = sizeof(context_arm->context_flags); 9846162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint8_t* context_after_flags = 9856162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com reinterpret_cast<uint8_t*>(context_arm.get()) + flags_size; 9869276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek if (!minidump_->ReadBytes(context_after_flags, 9879276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek sizeof(MDRawContextARM) - flags_size)) { 9889276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek BPLOG(ERROR) << "MinidumpContext could not read arm context"; 9899276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek return false; 9909276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek } 9919276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 9929276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek // Do this after reading the entire MDRawContext structure because 9939276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek // GetSystemInfo may seek minidump to a new position. 9949276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek if (!CheckAgainstSystemInfo(cpu_type)) { 9959276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek BPLOG(ERROR) << "MinidumpContext arm does not match system info"; 9969276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek return false; 9979276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek } 9989276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 9999276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek if (minidump_->swap()) { 10009276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek // context_arm->context_flags was already swapped. 10019276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek for (unsigned int ireg_index = 0; 10029276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek ireg_index < MD_CONTEXT_ARM_GPR_COUNT; 10039276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek ++ireg_index) { 10049276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek Swap(&context_arm->iregs[ireg_index]); 10059276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek } 10069276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek Swap(&context_arm->cpsr); 10079276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek Swap(&context_arm->float_save.fpscr); 10089276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek for (unsigned int fpr_index = 0; 10099276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek fpr_index < MD_FLOATINGSAVEAREA_ARM_FPR_COUNT; 10109276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek ++fpr_index) { 10119276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek Swap(&context_arm->float_save.regs[fpr_index]); 10129276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek } 10139276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek for (unsigned int fpe_index = 0; 10149276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek fpe_index < MD_FLOATINGSAVEAREA_ARM_FPEXTRA_COUNT; 10159276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek ++fpe_index) { 10169276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek Swap(&context_arm->float_save.extra[fpe_index]); 10179276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek } 10189276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek } 1019c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextARM(context_arm.release()); 10209276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 10219276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek break; 10229276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek } 10239276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 10245f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com case MD_CONTEXT_MIPS: { 10255f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com if (expected_size != sizeof(MDRawContextMIPS)) { 1026c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org BPLOG(ERROR) << "MinidumpContext MIPS size mismatch, " 1027c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org << expected_size 1028c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org << " != " 10295f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com << sizeof(MDRawContextMIPS); 10305f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com return false; 10315f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com } 10325f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com 10335f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com scoped_ptr<MDRawContextMIPS> context_mips(new MDRawContextMIPS()); 10345f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com 10355f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com // Set the context_flags member, which has already been read, and 10365f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com // read the rest of the structure beginning with the first member 10375f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com // after context_flags. 10385f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com context_mips->context_flags = context_flags; 10395f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com 10405f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com size_t flags_size = sizeof(context_mips->context_flags); 10415f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com uint8_t* context_after_flags = 10425f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com reinterpret_cast<uint8_t*>(context_mips.get()) + flags_size; 10435f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com if (!minidump_->ReadBytes(context_after_flags, 10445f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com sizeof(MDRawContextMIPS) - flags_size)) { 10455f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com BPLOG(ERROR) << "MinidumpContext could not read MIPS context"; 10465f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com return false; 10475f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com } 10485f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com 10495f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com // Do this after reading the entire MDRawContext structure because 10505f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com // GetSystemInfo may seek minidump to a new position. 10515f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com if (!CheckAgainstSystemInfo(cpu_type)) { 10525f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com BPLOG(ERROR) << "MinidumpContext MIPS does not match system info"; 10535f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com return false; 10545f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com } 10555f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com 10565f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com if (minidump_->swap()) { 10575f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com // context_mips->context_flags was already swapped. 10585f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com for (int ireg_index = 0; 10595f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com ireg_index < MD_CONTEXT_MIPS_GPR_COUNT; 10605f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com ++ireg_index) { 10615f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->iregs[ireg_index]); 10625f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com } 10635f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->mdhi); 10645f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->mdlo); 10655f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com for (int dsp_index = 0; 10665f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com dsp_index < MD_CONTEXT_MIPS_DSP_COUNT; 10675f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com ++dsp_index) { 10685f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->hi[dsp_index]); 10695f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->lo[dsp_index]); 10705f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com } 10715f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->dsp_control); 10725f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->epc); 10735f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->badvaddr); 10745f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->status); 10755f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->cause); 10765f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com for (int fpr_index = 0; 10775f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com fpr_index < MD_FLOATINGSAVEAREA_MIPS_FPR_COUNT; 10785f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com ++fpr_index) { 10795f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->float_save.regs[fpr_index]); 10805f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com } 10815f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->float_save.fpcsr); 10825f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com Swap(&context_mips->float_save.fir); 10835f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com } 1084c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextMIPS(context_mips.release()); 10855f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com 10865f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com break; 10875f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com } 10885f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com 10898eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek default: { 1090bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com // Unknown context type - Don't log as an error yet. Let the 10915f4fa55598baf92785223debe7d3787c6b29cf3ejschuh@chromium.org // caller work that out. 10925f4fa55598baf92785223debe7d3787c6b29cf3ejschuh@chromium.org BPLOG(INFO) << "MinidumpContext unknown context type " << 10938eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek HexString(cpu_type); 10948eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return false; 10958eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek break; 10968eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek } 10973402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } 1098c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.org SetContextFlags(context_flags); 10993261e8b6eac44a41341f112821482bee6c940c98mmentovai } 11003261e8b6eac44a41341f112821482bee6c940c98mmentovai 11013261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 11023261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 11033261e8b6eac44a41341f112821482bee6c940c98mmentovai} 11043261e8b6eac44a41341f112821482bee6c940c98mmentovai 11056162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpContext::CheckAgainstSystemInfo(uint32_t context_cpu_type) { 1106e47047b3835dcbb5da7fe7f5f9b6d78a5307122awaylonis // It's OK if the minidump doesn't contain an MD_SYSTEM_INFO_STREAM, 11073402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai // as this function just implements a sanity check. 11083402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai MinidumpSystemInfo* system_info = minidump_->GetSystemInfo(); 1109af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!system_info) { 1110af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "MinidumpContext could not be compared against " 1111af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "MinidumpSystemInfo"; 11123402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai return true; 1113af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 11143402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 1115e47047b3835dcbb5da7fe7f5f9b6d78a5307122awaylonis // If there is an MD_SYSTEM_INFO_STREAM, it should contain valid system info. 11163402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai const MDRawSystemInfo* raw_system_info = system_info->system_info(); 1117af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!raw_system_info) { 1118af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "MinidumpContext could not be compared against " 1119af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "MDRawSystemInfo"; 11203402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai return false; 1121af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 11223402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 11233402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai MDCPUArchitecture system_info_cpu_type = static_cast<MDCPUArchitecture>( 11243402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai raw_system_info->processor_architecture); 11253402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 11263402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai // Compare the CPU type of the context record to the CPU type in the 11273402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai // minidump's system info stream. 1128af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai bool return_value = false; 11293402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai switch (context_cpu_type) { 11303402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai case MD_CONTEXT_X86: 1131af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (system_info_cpu_type == MD_CPU_ARCHITECTURE_X86 || 1132299c816021c5d3e61106d48c1dd2d52fbf4d6cd4luly system_info_cpu_type == MD_CPU_ARCHITECTURE_X86_WIN64 || 1133299c816021c5d3e61106d48c1dd2d52fbf4d6cd4luly system_info_cpu_type == MD_CPU_ARCHITECTURE_AMD64) { 1134af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return_value = true; 11353402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } 11363402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai break; 11373402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 11383402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai case MD_CONTEXT_PPC: 1139af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (system_info_cpu_type == MD_CPU_ARCHITECTURE_PPC) 1140af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return_value = true; 11413402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai break; 1142ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 1143cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org case MD_CONTEXT_PPC64: 1144cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org if (system_info_cpu_type == MD_CPU_ARCHITECTURE_PPC64) 1145cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org return_value = true; 1146cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org break; 1147cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org 11488eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek case MD_CONTEXT_AMD64: 11498eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek if (system_info_cpu_type == MD_CPU_ARCHITECTURE_AMD64) 11508eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek return_value = true; 11518eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek break; 11528eb7111814953cb64ec0569b91ea99804b2d5b85ted.mielczarek 1153ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai case MD_CONTEXT_SPARC: 1154ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai if (system_info_cpu_type == MD_CPU_ARCHITECTURE_SPARC) 1155ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai return_value = true; 1156ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai break; 11579276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 11589276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek case MD_CONTEXT_ARM: 11599276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek if (system_info_cpu_type == MD_CPU_ARCHITECTURE_ARM) 11609276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek return_value = true; 11619276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek break; 11625f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com 116339d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org case MD_CONTEXT_ARM64: 116439d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org if (system_info_cpu_type == MD_CPU_ARCHITECTURE_ARM64) 116539d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org return_value = true; 116639d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org break; 116739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 11685f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com case MD_CONTEXT_MIPS: 11695f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com if (system_info_cpu_type == MD_CPU_ARCHITECTURE_MIPS) 11705f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com return_value = true; 11715f22d6a7f471f2352d394c188560fd06830e14f3gordana.cmiljanovic@imgtec.com break; 11723402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } 11733402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 1174af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !return_value) << "MinidumpContext CPU " << 1175af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(context_cpu_type) << 11760344a368deac6abaa280a298bcea9bb00a90df3fted.mielczarek@gmail.com " wrong for MinidumpSystemInfo CPU " << 1177af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(system_info_cpu_type); 1178af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 1179af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return return_value; 11803402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai} 11813402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 11823402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 11833261e8b6eac44a41341f112821482bee6c940c98mmentovai// 11843261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpMemoryRegion 11853261e8b6eac44a41341f112821482bee6c940c98mmentovai// 11863261e8b6eac44a41341f112821482bee6c940c98mmentovai 11873261e8b6eac44a41341f112821482bee6c940c98mmentovai 11886162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint32_t MinidumpMemoryRegion::max_bytes_ = 1024 * 1024; // 1MB 1189e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 1190e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 11913261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMemoryRegion::MinidumpMemoryRegion(Minidump* minidump) 119253d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpObject(minidump), 119353d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai descriptor_(NULL), 119453d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai memory_(NULL) { 11953261e8b6eac44a41341f112821482bee6c940c98mmentovai} 11963261e8b6eac44a41341f112821482bee6c940c98mmentovai 11973261e8b6eac44a41341f112821482bee6c940c98mmentovai 11983261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMemoryRegion::~MinidumpMemoryRegion() { 11993261e8b6eac44a41341f112821482bee6c940c98mmentovai delete memory_; 12003261e8b6eac44a41341f112821482bee6c940c98mmentovai} 12013261e8b6eac44a41341f112821482bee6c940c98mmentovai 12023261e8b6eac44a41341f112821482bee6c940c98mmentovai 12033261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpMemoryRegion::SetDescriptor(MDMemoryDescriptor* descriptor) { 12043261e8b6eac44a41341f112821482bee6c940c98mmentovai descriptor_ = descriptor; 12053261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = descriptor && 1206fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai descriptor_->memory.data_size <= 12076162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com numeric_limits<uint64_t>::max() - 1208fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai descriptor_->start_of_memory_range; 12093261e8b6eac44a41341f112821482bee6c940c98mmentovai} 12103261e8b6eac44a41341f112821482bee6c940c98mmentovai 12113261e8b6eac44a41341f112821482bee6c940c98mmentovai 12126162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comconst uint8_t* MinidumpMemoryRegion::GetMemory() const { 1213af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1214af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpMemoryRegion for GetMemory"; 12153261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 1216af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 12173261e8b6eac44a41341f112821482bee6c940c98mmentovai 12183261e8b6eac44a41341f112821482bee6c940c98mmentovai if (!memory_) { 1219af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (descriptor_->memory.data_size == 0) { 1220af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryRegion is empty"; 1221373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return NULL; 1222af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1223373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai 1224af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->SeekSet(descriptor_->memory.rva)) { 1225af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryRegion could not seek to memory region"; 12263261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 1227af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 12283261e8b6eac44a41341f112821482bee6c940c98mmentovai 1229e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (descriptor_->memory.data_size > max_bytes_) { 1230e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "MinidumpMemoryRegion size " << 1231e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai descriptor_->memory.data_size << " exceeds maximum " << 1232e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai max_bytes_; 1233e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai return NULL; 1234e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai } 1235e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 12366162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com scoped_ptr< vector<uint8_t> > memory( 12376162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com new vector<uint8_t>(descriptor_->memory.data_size)); 12383261e8b6eac44a41341f112821482bee6c940c98mmentovai 1239af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&(*memory)[0], descriptor_->memory.data_size)) { 1240af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryRegion could not read memory region"; 12413261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 1242af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 12433261e8b6eac44a41341f112821482bee6c940c98mmentovai 12443261e8b6eac44a41341f112821482bee6c940c98mmentovai memory_ = memory.release(); 12453261e8b6eac44a41341f112821482bee6c940c98mmentovai } 12463261e8b6eac44a41341f112821482bee6c940c98mmentovai 12473261e8b6eac44a41341f112821482bee6c940c98mmentovai return &(*memory_)[0]; 12483261e8b6eac44a41341f112821482bee6c940c98mmentovai} 12493261e8b6eac44a41341f112821482bee6c940c98mmentovai 12503261e8b6eac44a41341f112821482bee6c940c98mmentovai 12516162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint64_t MinidumpMemoryRegion::GetBase() const { 1252af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1253af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpMemoryRegion for GetBase"; 12546162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com return static_cast<uint64_t>(-1); 1255af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1256af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 1257af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return descriptor_->start_of_memory_range; 12583261e8b6eac44a41341f112821482bee6c940c98mmentovai} 12593261e8b6eac44a41341f112821482bee6c940c98mmentovai 12603261e8b6eac44a41341f112821482bee6c940c98mmentovai 12616162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint32_t MinidumpMemoryRegion::GetSize() const { 1262af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1263af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpMemoryRegion for GetSize"; 1264af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return 0; 1265af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1266af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 1267af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return descriptor_->memory.data_size; 12683261e8b6eac44a41341f112821482bee6c940c98mmentovai} 12693261e8b6eac44a41341f112821482bee6c940c98mmentovai 12703261e8b6eac44a41341f112821482bee6c940c98mmentovai 12713261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpMemoryRegion::FreeMemory() { 12723261e8b6eac44a41341f112821482bee6c940c98mmentovai delete memory_; 12733261e8b6eac44a41341f112821482bee6c940c98mmentovai memory_ = NULL; 12743261e8b6eac44a41341f112821482bee6c940c98mmentovai} 12753261e8b6eac44a41341f112821482bee6c940c98mmentovai 12763261e8b6eac44a41341f112821482bee6c940c98mmentovai 12773261e8b6eac44a41341f112821482bee6c940c98mmentovaitemplate<typename T> 12786162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpMemoryRegion::GetMemoryAtAddressInternal(uint64_t address, 12792214cb9bc1872cafae9127778c0cba556c89e43djimblandy T* value) const { 1280af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !value) << "MinidumpMemoryRegion::GetMemoryAtAddressInternal " 1281af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "requires |value|"; 1282af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai assert(value); 1283af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai *value = 0; 1284af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 1285af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1286af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpMemoryRegion for " 1287af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "GetMemoryAtAddressInternal"; 12883261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 1289af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 12903261e8b6eac44a41341f112821482bee6c940c98mmentovai 1291bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com // Common failure case 12923261e8b6eac44a41341f112821482bee6c940c98mmentovai if (address < descriptor_->start_of_memory_range || 12936162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com sizeof(T) > numeric_limits<uint64_t>::max() - address || 12943261e8b6eac44a41341f112821482bee6c940c98mmentovai address + sizeof(T) > descriptor_->start_of_memory_range + 12953261e8b6eac44a41341f112821482bee6c940c98mmentovai descriptor_->memory.data_size) { 1296bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com BPLOG(INFO) << "MinidumpMemoryRegion request out of range: " << 1297af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(address) << "+" << sizeof(T) << "/" << 1298af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(descriptor_->start_of_memory_range) << "+" << 1299af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(descriptor_->memory.data_size); 13003261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 13013261e8b6eac44a41341f112821482bee6c940c98mmentovai } 13023261e8b6eac44a41341f112821482bee6c940c98mmentovai 13036162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com const uint8_t* memory = GetMemory(); 1304af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!memory) { 1305af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai // GetMemory already logged a perfectly good message. 13063261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 1307af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 13083261e8b6eac44a41341f112821482bee6c940c98mmentovai 13093261e8b6eac44a41341f112821482bee6c940c98mmentovai // If the CPU requires memory accesses to be aligned, this can crash. 13103261e8b6eac44a41341f112821482bee6c940c98mmentovai // x86 and ppc are able to cope, though. 13113261e8b6eac44a41341f112821482bee6c940c98mmentovai *value = *reinterpret_cast<const T*>( 13123261e8b6eac44a41341f112821482bee6c940c98mmentovai &memory[address - descriptor_->start_of_memory_range]); 13133261e8b6eac44a41341f112821482bee6c940c98mmentovai 13143261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) 13153261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(value); 13163261e8b6eac44a41341f112821482bee6c940c98mmentovai 13173261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 13183261e8b6eac44a41341f112821482bee6c940c98mmentovai} 13193261e8b6eac44a41341f112821482bee6c940c98mmentovai 13203261e8b6eac44a41341f112821482bee6c940c98mmentovai 13216162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpMemoryRegion::GetMemoryAtAddress(uint64_t address, 13226162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint8_t* value) const { 13233261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetMemoryAtAddressInternal(address, value); 13243261e8b6eac44a41341f112821482bee6c940c98mmentovai} 13253261e8b6eac44a41341f112821482bee6c940c98mmentovai 13263261e8b6eac44a41341f112821482bee6c940c98mmentovai 13276162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpMemoryRegion::GetMemoryAtAddress(uint64_t address, 13286162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint16_t* value) const { 13293261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetMemoryAtAddressInternal(address, value); 13303261e8b6eac44a41341f112821482bee6c940c98mmentovai} 13313261e8b6eac44a41341f112821482bee6c940c98mmentovai 13323261e8b6eac44a41341f112821482bee6c940c98mmentovai 13336162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpMemoryRegion::GetMemoryAtAddress(uint64_t address, 13346162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t* value) const { 13353261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetMemoryAtAddressInternal(address, value); 13363261e8b6eac44a41341f112821482bee6c940c98mmentovai} 13373261e8b6eac44a41341f112821482bee6c940c98mmentovai 13383261e8b6eac44a41341f112821482bee6c940c98mmentovai 13396162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpMemoryRegion::GetMemoryAtAddress(uint64_t address, 13406162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t* value) const { 13413261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetMemoryAtAddressInternal(address, value); 13423261e8b6eac44a41341f112821482bee6c940c98mmentovai} 13433261e8b6eac44a41341f112821482bee6c940c98mmentovai 13443261e8b6eac44a41341f112821482bee6c940c98mmentovai 1345c5e242b8cd4280db5162e5a3084f2dc9e16e8ffbmmandlis@chromium.orgvoid MinidumpMemoryRegion::Print() const { 1346af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1347af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryRegion cannot print invalid data"; 13483261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 1349af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 13503261e8b6eac44a41341f112821482bee6c940c98mmentovai 13516162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com const uint8_t* memory = GetMemory(); 13523261e8b6eac44a41341f112821482bee6c940c98mmentovai if (memory) { 13533261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("0x"); 13543261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int byte_index = 0; 13553261e8b6eac44a41341f112821482bee6c940c98mmentovai byte_index < descriptor_->memory.data_size; 13563261e8b6eac44a41341f112821482bee6c940c98mmentovai byte_index++) { 13573261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("%02x", memory[byte_index]); 13583261e8b6eac44a41341f112821482bee6c940c98mmentovai } 13593261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 13603261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 13613261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("No memory\n"); 13623261e8b6eac44a41341f112821482bee6c940c98mmentovai } 13633261e8b6eac44a41341f112821482bee6c940c98mmentovai} 13643261e8b6eac44a41341f112821482bee6c940c98mmentovai 13653261e8b6eac44a41341f112821482bee6c940c98mmentovai 13663261e8b6eac44a41341f112821482bee6c940c98mmentovai// 13673261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpThread 13683261e8b6eac44a41341f112821482bee6c940c98mmentovai// 13693261e8b6eac44a41341f112821482bee6c940c98mmentovai 13703261e8b6eac44a41341f112821482bee6c940c98mmentovai 13713261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpThread::MinidumpThread(Minidump* minidump) 137253d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpObject(minidump), 137353d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai thread_(), 137453d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai memory_(NULL), 137553d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai context_(NULL) { 13763261e8b6eac44a41341f112821482bee6c940c98mmentovai} 13773261e8b6eac44a41341f112821482bee6c940c98mmentovai 13783261e8b6eac44a41341f112821482bee6c940c98mmentovai 13793261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpThread::~MinidumpThread() { 13803261e8b6eac44a41341f112821482bee6c940c98mmentovai delete memory_; 13813261e8b6eac44a41341f112821482bee6c940c98mmentovai delete context_; 13823261e8b6eac44a41341f112821482bee6c940c98mmentovai} 13833261e8b6eac44a41341f112821482bee6c940c98mmentovai 13843261e8b6eac44a41341f112821482bee6c940c98mmentovai 13853261e8b6eac44a41341f112821482bee6c940c98mmentovaibool MinidumpThread::Read() { 13863261e8b6eac44a41341f112821482bee6c940c98mmentovai // Invalidate cached data. 13873261e8b6eac44a41341f112821482bee6c940c98mmentovai delete memory_; 13883261e8b6eac44a41341f112821482bee6c940c98mmentovai memory_ = NULL; 13893261e8b6eac44a41341f112821482bee6c940c98mmentovai delete context_; 13903261e8b6eac44a41341f112821482bee6c940c98mmentovai context_ = NULL; 13913261e8b6eac44a41341f112821482bee6c940c98mmentovai 13923261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 13933261e8b6eac44a41341f112821482bee6c940c98mmentovai 1394af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&thread_, sizeof(thread_))) { 1395af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThread cannot read thread"; 13963261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 1397af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 13983261e8b6eac44a41341f112821482bee6c940c98mmentovai 13993261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) { 14003261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&thread_.thread_id); 14013261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&thread_.suspend_count); 14023261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&thread_.priority_class); 14033261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&thread_.priority); 14043261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&thread_.teb); 14053261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&thread_.stack); 14063261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&thread_.thread_context); 14073261e8b6eac44a41341f112821482bee6c940c98mmentovai } 14083261e8b6eac44a41341f112821482bee6c940c98mmentovai 1409fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai // Check for base + size overflow or undersize. 141002ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com if (thread_.stack.memory.rva == 0 || 141102ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com thread_.stack.memory.data_size == 0 || 14126162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com thread_.stack.memory.data_size > numeric_limits<uint64_t>::max() - 1413fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai thread_.stack.start_of_memory_range) { 1414e721e628ec10381c96e38cfc82c1816983097165ted.mielczarek@gmail.com // This is ok, but log an error anyway. 1415af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThread has a memory region problem, " << 1416af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(thread_.stack.start_of_memory_range) << "+" << 141702ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com HexString(thread_.stack.memory.data_size) << 141802ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com ", RVA 0x" << HexString(thread_.stack.memory.rva); 1419e721e628ec10381c96e38cfc82c1816983097165ted.mielczarek@gmail.com } else { 1420e721e628ec10381c96e38cfc82c1816983097165ted.mielczarek@gmail.com memory_ = new MinidumpMemoryRegion(minidump_); 1421e721e628ec10381c96e38cfc82c1816983097165ted.mielczarek@gmail.com memory_->SetDescriptor(&thread_.stack); 1422af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 14233261e8b6eac44a41341f112821482bee6c940c98mmentovai 14243261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 14253261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 14263261e8b6eac44a41341f112821482bee6c940c98mmentovai} 14273261e8b6eac44a41341f112821482bee6c940c98mmentovai 142802ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.comuint64_t MinidumpThread::GetStartOfStackMemoryRange() const { 142902ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com if (!valid_) { 143002ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com BPLOG(ERROR) << "GetStartOfStackMemoryRange: Invalid MinidumpThread"; 143102ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com return 0; 143202ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com } 143302ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com 143402ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com return thread_.stack.start_of_memory_range; 143502ee17f7cf2b669ec1ee688243bcf78f8f58397bivan.penkov@gmail.com} 14363261e8b6eac44a41341f112821482bee6c940c98mmentovai 14373261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMemoryRegion* MinidumpThread::GetMemory() { 1438af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1439af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpThread for GetMemory"; 1440af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return NULL; 1441af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1442af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 1443af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return memory_; 14443261e8b6eac44a41341f112821482bee6c940c98mmentovai} 14453261e8b6eac44a41341f112821482bee6c940c98mmentovai 14463261e8b6eac44a41341f112821482bee6c940c98mmentovai 14473261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpContext* MinidumpThread::GetContext() { 1448af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1449af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpThread for GetContext"; 14503261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 1451af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 14523261e8b6eac44a41341f112821482bee6c940c98mmentovai 14533261e8b6eac44a41341f112821482bee6c940c98mmentovai if (!context_) { 1454af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->SeekSet(thread_.thread_context.rva)) { 1455af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThread cannot seek to context"; 14563261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 1457af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 14583261e8b6eac44a41341f112821482bee6c940c98mmentovai 14592466d8e993a800a17e00deda2f3a27e0505140e1mmentovai scoped_ptr<MinidumpContext> context(new MinidumpContext(minidump_)); 14603261e8b6eac44a41341f112821482bee6c940c98mmentovai 1461af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!context->Read(thread_.thread_context.data_size)) { 1462af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThread cannot read context"; 14633261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 1464af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 14653261e8b6eac44a41341f112821482bee6c940c98mmentovai 14663261e8b6eac44a41341f112821482bee6c940c98mmentovai context_ = context.release(); 14673261e8b6eac44a41341f112821482bee6c940c98mmentovai } 14683261e8b6eac44a41341f112821482bee6c940c98mmentovai 14693261e8b6eac44a41341f112821482bee6c940c98mmentovai return context_; 14703261e8b6eac44a41341f112821482bee6c940c98mmentovai} 14713261e8b6eac44a41341f112821482bee6c940c98mmentovai 14723261e8b6eac44a41341f112821482bee6c940c98mmentovai 14736162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpThread::GetThreadID(uint32_t *thread_id) const { 1474af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !thread_id) << "MinidumpThread::GetThreadID requires " 1475af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "|thread_id|"; 1476af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai assert(thread_id); 1477af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai *thread_id = 0; 1478af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 1479af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1480af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpThread for GetThreadID"; 148176f052f8fbf8864dee5992b857229d06560a766ammentovai return false; 1482af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 148376f052f8fbf8864dee5992b857229d06560a766ammentovai 148476f052f8fbf8864dee5992b857229d06560a766ammentovai *thread_id = thread_.thread_id; 148576f052f8fbf8864dee5992b857229d06560a766ammentovai return true; 14863261e8b6eac44a41341f112821482bee6c940c98mmentovai} 14873261e8b6eac44a41341f112821482bee6c940c98mmentovai 14883261e8b6eac44a41341f112821482bee6c940c98mmentovai 14893261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpThread::Print() { 1490af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1491af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThread cannot print invalid data"; 14923261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 1493af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 14943261e8b6eac44a41341f112821482bee6c940c98mmentovai 14953261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MDRawThread\n"); 14963261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" thread_id = 0x%x\n", thread_.thread_id); 14973261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" suspend_count = %d\n", thread_.suspend_count); 14983261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" priority_class = 0x%x\n", thread_.priority_class); 14993261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" priority = 0x%x\n", thread_.priority); 1500c27cf3e3959189f78fe2de40405987c3f33488cemmentovai printf(" teb = 0x%" PRIx64 "\n", thread_.teb); 1501c27cf3e3959189f78fe2de40405987c3f33488cemmentovai printf(" stack.start_of_memory_range = 0x%" PRIx64 "\n", 15023261e8b6eac44a41341f112821482bee6c940c98mmentovai thread_.stack.start_of_memory_range); 15033261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" stack.memory.data_size = 0x%x\n", 15043261e8b6eac44a41341f112821482bee6c940c98mmentovai thread_.stack.memory.data_size); 15053261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" stack.memory.rva = 0x%x\n", thread_.stack.memory.rva); 15063261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" thread_context.data_size = 0x%x\n", 15073261e8b6eac44a41341f112821482bee6c940c98mmentovai thread_.thread_context.data_size); 15083261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" thread_context.rva = 0x%x\n", 15093261e8b6eac44a41341f112821482bee6c940c98mmentovai thread_.thread_context.rva); 15103261e8b6eac44a41341f112821482bee6c940c98mmentovai 15113261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpContext* context = GetContext(); 15123261e8b6eac44a41341f112821482bee6c940c98mmentovai if (context) { 15133261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 15143261e8b6eac44a41341f112821482bee6c940c98mmentovai context->Print(); 15153261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 15163261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (no context)\n"); 15173261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 15183261e8b6eac44a41341f112821482bee6c940c98mmentovai } 15193261e8b6eac44a41341f112821482bee6c940c98mmentovai 15203261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpMemoryRegion* memory = GetMemory(); 15213261e8b6eac44a41341f112821482bee6c940c98mmentovai if (memory) { 15223261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("Stack\n"); 15233261e8b6eac44a41341f112821482bee6c940c98mmentovai memory->Print(); 15243261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 15253261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("No stack\n"); 15263261e8b6eac44a41341f112821482bee6c940c98mmentovai } 15273261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 15283261e8b6eac44a41341f112821482bee6c940c98mmentovai} 15293261e8b6eac44a41341f112821482bee6c940c98mmentovai 15303261e8b6eac44a41341f112821482bee6c940c98mmentovai 15313261e8b6eac44a41341f112821482bee6c940c98mmentovai// 15323261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpThreadList 15333261e8b6eac44a41341f112821482bee6c940c98mmentovai// 15343261e8b6eac44a41341f112821482bee6c940c98mmentovai 15353261e8b6eac44a41341f112821482bee6c940c98mmentovai 15366162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint32_t MinidumpThreadList::max_threads_ = 4096; 1537e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 1538e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 15393261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpThreadList::MinidumpThreadList(Minidump* minidump) 154053d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpStream(minidump), 154153d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai id_to_thread_map_(), 154253d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai threads_(NULL), 154353d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai thread_count_(0) { 15443261e8b6eac44a41341f112821482bee6c940c98mmentovai} 15453261e8b6eac44a41341f112821482bee6c940c98mmentovai 15463261e8b6eac44a41341f112821482bee6c940c98mmentovai 15473261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpThreadList::~MinidumpThreadList() { 15483261e8b6eac44a41341f112821482bee6c940c98mmentovai delete threads_; 15493261e8b6eac44a41341f112821482bee6c940c98mmentovai} 15503261e8b6eac44a41341f112821482bee6c940c98mmentovai 15513261e8b6eac44a41341f112821482bee6c940c98mmentovai 15526162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpThreadList::Read(uint32_t expected_size) { 15533261e8b6eac44a41341f112821482bee6c940c98mmentovai // Invalidate cached data. 15543261e8b6eac44a41341f112821482bee6c940c98mmentovai id_to_thread_map_.clear(); 15553261e8b6eac44a41341f112821482bee6c940c98mmentovai delete threads_; 15563261e8b6eac44a41341f112821482bee6c940c98mmentovai threads_ = NULL; 15573261e8b6eac44a41341f112821482bee6c940c98mmentovai thread_count_ = 0; 15583261e8b6eac44a41341f112821482bee6c940c98mmentovai 15593261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 15603261e8b6eac44a41341f112821482bee6c940c98mmentovai 15616162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t thread_count; 1562af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (expected_size < sizeof(thread_count)) { 1563af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThreadList count size mismatch, " << 1564af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai expected_size << " < " << sizeof(thread_count); 15653261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 1566af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1567af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&thread_count, sizeof(thread_count))) { 1568af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThreadList cannot read thread count"; 15693261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 1570af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 15713261e8b6eac44a41341f112821482bee6c940c98mmentovai 15723261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) 15733261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&thread_count); 15743261e8b6eac44a41341f112821482bee6c940c98mmentovai 15756162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com if (thread_count > numeric_limits<uint32_t>::max() / sizeof(MDRawThread)) { 1576fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai BPLOG(ERROR) << "MinidumpThreadList thread count " << thread_count << 1577fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai " would cause multiplication overflow"; 1578fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai return false; 1579fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai } 1580fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai 15813261e8b6eac44a41341f112821482bee6c940c98mmentovai if (expected_size != sizeof(thread_count) + 15823261e8b6eac44a41341f112821482bee6c940c98mmentovai thread_count * sizeof(MDRawThread)) { 1583ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai // may be padded with 4 bytes on 64bit ABIs for alignment 1584ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai if (expected_size == sizeof(thread_count) + 4 + 1585ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai thread_count * sizeof(MDRawThread)) { 15866162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t useless; 1587ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai if (!minidump_->ReadBytes(&useless, 4)) { 1588f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "MinidumpThreadList cannot read threadlist padded " 1589f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com "bytes"; 1590ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai return false; 1591ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 1592ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } else { 1593ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai BPLOG(ERROR) << "MinidumpThreadList size mismatch, " << expected_size << 1594ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai " != " << sizeof(thread_count) + 1595ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai thread_count * sizeof(MDRawThread); 1596ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai return false; 1597ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 15983261e8b6eac44a41341f112821482bee6c940c98mmentovai } 15993261e8b6eac44a41341f112821482bee6c940c98mmentovai 1600bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com 1601e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (thread_count > max_threads_) { 1602e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "MinidumpThreadList count " << thread_count << 1603e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai " exceeds maximum " << max_threads_; 1604e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai return false; 1605e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai } 1606e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 1607e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (thread_count != 0) { 1608373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai scoped_ptr<MinidumpThreads> threads( 1609373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai new MinidumpThreads(thread_count, MinidumpThread(minidump_))); 16103261e8b6eac44a41341f112821482bee6c940c98mmentovai 1611373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai for (unsigned int thread_index = 0; 1612373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai thread_index < thread_count; 1613373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai ++thread_index) { 1614373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai MinidumpThread* thread = &(*threads)[thread_index]; 16153261e8b6eac44a41341f112821482bee6c940c98mmentovai 1616373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Assume that the file offset is correct after the last read. 1617af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!thread->Read()) { 1618af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThreadList cannot read thread " << 1619af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai thread_index << "/" << thread_count; 1620373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 1621af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 16223261e8b6eac44a41341f112821482bee6c940c98mmentovai 16236162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t thread_id; 1624af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!thread->GetThreadID(&thread_id)) { 1625af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThreadList cannot get thread ID for thread " << 1626af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai thread_index << "/" << thread_count; 1627373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 1628af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 162976f052f8fbf8864dee5992b857229d06560a766ammentovai 1630373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai if (GetThreadByID(thread_id)) { 1631373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Another thread with this ID is already in the list. Data error. 1632af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThreadList found multiple threads with ID " << 1633af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(thread_id) << " at thread " << 1634af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai thread_index << "/" << thread_count; 1635373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 1636373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 1637373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai id_to_thread_map_[thread_id] = thread; 16383261e8b6eac44a41341f112821482bee6c940c98mmentovai } 1639373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai 1640373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai threads_ = threads.release(); 16413261e8b6eac44a41341f112821482bee6c940c98mmentovai } 16423261e8b6eac44a41341f112821482bee6c940c98mmentovai 16433261e8b6eac44a41341f112821482bee6c940c98mmentovai thread_count_ = thread_count; 16443261e8b6eac44a41341f112821482bee6c940c98mmentovai 16453261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 16463261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 16473261e8b6eac44a41341f112821482bee6c940c98mmentovai} 16483261e8b6eac44a41341f112821482bee6c940c98mmentovai 16493261e8b6eac44a41341f112821482bee6c940c98mmentovai 16503261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpThread* MinidumpThreadList::GetThreadAtIndex(unsigned int index) 16513261e8b6eac44a41341f112821482bee6c940c98mmentovai const { 1652af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1653af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpThreadList for GetThreadAtIndex"; 1654af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return NULL; 1655af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1656af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 1657af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (index >= thread_count_) { 1658af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThreadList index out of range: " << 1659af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai index << "/" << thread_count_; 16603261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 1661af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 16623261e8b6eac44a41341f112821482bee6c940c98mmentovai 16633261e8b6eac44a41341f112821482bee6c940c98mmentovai return &(*threads_)[index]; 16643261e8b6eac44a41341f112821482bee6c940c98mmentovai} 16653261e8b6eac44a41341f112821482bee6c940c98mmentovai 16663261e8b6eac44a41341f112821482bee6c940c98mmentovai 16676162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comMinidumpThread* MinidumpThreadList::GetThreadByID(uint32_t thread_id) { 16683261e8b6eac44a41341f112821482bee6c940c98mmentovai // Don't check valid_. Read calls this method before everything is 16693261e8b6eac44a41341f112821482bee6c940c98mmentovai // validated. It is safe to not check valid_ here. 16703261e8b6eac44a41341f112821482bee6c940c98mmentovai return id_to_thread_map_[thread_id]; 16713261e8b6eac44a41341f112821482bee6c940c98mmentovai} 16723261e8b6eac44a41341f112821482bee6c940c98mmentovai 16733261e8b6eac44a41341f112821482bee6c940c98mmentovai 16743261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpThreadList::Print() { 1675af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1676af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpThreadList cannot print invalid data"; 16773261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 1678af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 16793261e8b6eac44a41341f112821482bee6c940c98mmentovai 16803261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MinidumpThreadList\n"); 16813261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" thread_count = %d\n", thread_count_); 16823261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 16833261e8b6eac44a41341f112821482bee6c940c98mmentovai 16843261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int thread_index = 0; 16853261e8b6eac44a41341f112821482bee6c940c98mmentovai thread_index < thread_count_; 16863261e8b6eac44a41341f112821482bee6c940c98mmentovai ++thread_index) { 16873261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("thread[%d]\n", thread_index); 16883261e8b6eac44a41341f112821482bee6c940c98mmentovai 16893261e8b6eac44a41341f112821482bee6c940c98mmentovai (*threads_)[thread_index].Print(); 16903261e8b6eac44a41341f112821482bee6c940c98mmentovai } 16913261e8b6eac44a41341f112821482bee6c940c98mmentovai} 16923261e8b6eac44a41341f112821482bee6c940c98mmentovai 16933261e8b6eac44a41341f112821482bee6c940c98mmentovai 16943261e8b6eac44a41341f112821482bee6c940c98mmentovai// 16953261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpModule 16963261e8b6eac44a41341f112821482bee6c940c98mmentovai// 16973261e8b6eac44a41341f112821482bee6c940c98mmentovai 16983261e8b6eac44a41341f112821482bee6c940c98mmentovai 16996162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint32_t MinidumpModule::max_cv_bytes_ = 32768; 17006162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint32_t MinidumpModule::max_misc_bytes_ = 32768; 1701e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 1702e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 17033261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpModule::MinidumpModule(Minidump* minidump) 170453d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpObject(minidump), 1705db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_valid_(false), 170611e180cd3e855796aee4239aa4b22dbda5de9c00mmentovai has_debug_info_(false), 170753d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai module_(), 170853d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai name_(NULL), 170953d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai cv_record_(NULL), 171048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_signature_(MD_CVINFOUNKNOWN_SIGNATURE), 1711db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai misc_record_(NULL) { 17123261e8b6eac44a41341f112821482bee6c940c98mmentovai} 17133261e8b6eac44a41341f112821482bee6c940c98mmentovai 17143261e8b6eac44a41341f112821482bee6c940c98mmentovai 17153261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpModule::~MinidumpModule() { 17163261e8b6eac44a41341f112821482bee6c940c98mmentovai delete name_; 17173261e8b6eac44a41341f112821482bee6c940c98mmentovai delete cv_record_; 17183261e8b6eac44a41341f112821482bee6c940c98mmentovai delete misc_record_; 17193261e8b6eac44a41341f112821482bee6c940c98mmentovai} 17203261e8b6eac44a41341f112821482bee6c940c98mmentovai 17213261e8b6eac44a41341f112821482bee6c940c98mmentovai 17223261e8b6eac44a41341f112821482bee6c940c98mmentovaibool MinidumpModule::Read() { 17233261e8b6eac44a41341f112821482bee6c940c98mmentovai // Invalidate cached data. 17243261e8b6eac44a41341f112821482bee6c940c98mmentovai delete name_; 17253261e8b6eac44a41341f112821482bee6c940c98mmentovai name_ = NULL; 17263261e8b6eac44a41341f112821482bee6c940c98mmentovai delete cv_record_; 17273261e8b6eac44a41341f112821482bee6c940c98mmentovai cv_record_ = NULL; 172848dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_signature_ = MD_CVINFOUNKNOWN_SIGNATURE; 17293261e8b6eac44a41341f112821482bee6c940c98mmentovai delete misc_record_; 17303261e8b6eac44a41341f112821482bee6c940c98mmentovai misc_record_ = NULL; 17313261e8b6eac44a41341f112821482bee6c940c98mmentovai 1732db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_valid_ = false; 1733d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek has_debug_info_ = false; 17343261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 17353261e8b6eac44a41341f112821482bee6c940c98mmentovai 1736af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&module_, MD_MODULE_SIZE)) { 1737af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule cannot read module"; 17383261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 1739af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 17403261e8b6eac44a41341f112821482bee6c940c98mmentovai 17413261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) { 17423261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.base_of_image); 17433261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.size_of_image); 17443261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.checksum); 17453261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.time_date_stamp); 17463261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.module_name_rva); 17473261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.signature); 17483261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.struct_version); 17493261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_version_hi); 17503261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_version_lo); 17513261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.product_version_hi); 17523261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.product_version_lo); 17533261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_flags_mask); 17543261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_flags); 17553261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_os); 17563261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_type); 17573261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_subtype); 17583261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_date_hi); 17593261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.version_info.file_date_lo); 17603261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.cv_record); 17613261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_.misc_record); 17623261e8b6eac44a41341f112821482bee6c940c98mmentovai // Don't swap reserved fields because their contents are unknown (as 17633261e8b6eac44a41341f112821482bee6c940c98mmentovai // are their proper widths). 17643261e8b6eac44a41341f112821482bee6c940c98mmentovai } 17653261e8b6eac44a41341f112821482bee6c940c98mmentovai 1766fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai // Check for base + size overflow or undersize. 1767fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai if (module_.size_of_image == 0 || 1768fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai module_.size_of_image > 17696162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com numeric_limits<uint64_t>::max() - module_.base_of_image) { 1770af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule has a module problem, " << 1771af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(module_.base_of_image) << "+" << 1772fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai HexString(module_.size_of_image); 17733261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 1774af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 17753261e8b6eac44a41341f112821482bee6c940c98mmentovai 1776db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_valid_ = true; 1777db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return true; 1778db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 1779db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1780db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1781db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaibool MinidumpModule::ReadAuxiliaryData() { 1782af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!module_valid_) { 1783af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModule for ReadAuxiliaryData"; 1784db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return false; 1785af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1786db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1787db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // Each module must have a name. 1788db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai name_ = minidump_->ReadString(module_.module_name_rva); 1789af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!name_) { 1790af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule could not read name"; 1791db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return false; 1792af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1793db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1794d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek // At this point, we have enough info for the module to be valid. 1795d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek valid_ = true; 1796d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek 1797db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // CodeView and miscellaneous debug records are only required if the 1798db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // module indicates that they exist. 1799af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (module_.cv_record.data_size && !GetCVRecord(NULL)) { 1800af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule has no CodeView record, " 1801af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "but one was expected"; 1802db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return false; 1803af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1804db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1805af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (module_.misc_record.data_size && !GetMiscRecord(NULL)) { 1806af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule has no miscellaneous debug record, " 1807af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "but one was expected"; 1808db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return false; 1809af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1810db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1811d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek has_debug_info_ = true; 18123261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 18133261e8b6eac44a41341f112821482bee6c940c98mmentovai} 18143261e8b6eac44a41341f112821482bee6c940c98mmentovai 18153261e8b6eac44a41341f112821482bee6c940c98mmentovai 1816db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaistring MinidumpModule::code_file() const { 1817af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1818af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModule for code_file"; 1819db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return ""; 1820af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1821db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1822db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return *name_; 1823db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 18243261e8b6eac44a41341f112821482bee6c940c98mmentovai 18253261e8b6eac44a41341f112821482bee6c940c98mmentovai 1826db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaistring MinidumpModule::code_identifier() const { 1827af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1828af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModule for code_identifier"; 1829db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return ""; 1830af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1831db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1832d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek if (!has_debug_info_) 1833d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek return ""; 1834d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek 1835db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai MinidumpSystemInfo *minidump_system_info = minidump_->GetSystemInfo(); 1836af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_system_info) { 1837af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule code_identifier requires " 1838af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "MinidumpSystemInfo"; 1839db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return ""; 1840af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1841db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1842db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai const MDRawSystemInfo *raw_system_info = minidump_system_info->system_info(); 1843af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!raw_system_info) { 1844af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule code_identifier requires MDRawSystemInfo"; 1845db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return ""; 1846af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1847db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1848db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai string identifier; 1849db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1850db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai switch (raw_system_info->platform_id) { 1851db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai case MD_OS_WIN32_NT: 1852db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai case MD_OS_WIN32_WINDOWS: { 1853c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai // Use the same format that the MS symbol server uses in filesystem 1854c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai // hierarchies. 1855db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai char identifier_string[17]; 1856c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai snprintf(identifier_string, sizeof(identifier_string), "%08X%x", 1857db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_.time_date_stamp, module_.size_of_image); 1858db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai identifier = identifier_string; 1859db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai break; 1860db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1861db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 18620e94332f7c615d2b734e840bef233f3ee1188801ted.mielczarek case MD_OS_MAC_OS_X: 186363f97ad134db3fa175c489b50b66a79bf0c95025qsr@chromium.org case MD_OS_IOS: 1864ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai case MD_OS_SOLARIS: 18655187de1ae5cb7df2343b650416aa805084bdf8dadigit@chromium.org case MD_OS_ANDROID: 1866f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com case MD_OS_LINUX: 1867a29f376a8bb9122f29ea1c53c02a188683fd5a72bradnelson@chromium.org case MD_OS_NACL: 1868d9b40724ed6a6c243eaef196e617071ec36d4282thestig@chromium.org case MD_OS_PS3: { 1869db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // TODO(mmentovai): support uuid extension if present, otherwise fall 1870db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // back to version (from LC_ID_DYLIB?), otherwise fall back to something 1871db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // else. 1872db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai identifier = "id"; 1873db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai break; 1874db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1875db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1876db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai default: { 1877db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // Without knowing what OS generated the dump, we can't generate a good 1878db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // identifier. Return an empty string, signalling failure. 1879af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule code_identifier requires known platform, " 1880af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "found " << HexString(raw_system_info->platform_id); 1881db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai break; 1882db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1883db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1884db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1885db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return identifier; 18863261e8b6eac44a41341f112821482bee6c940c98mmentovai} 18873261e8b6eac44a41341f112821482bee6c940c98mmentovai 18883261e8b6eac44a41341f112821482bee6c940c98mmentovai 1889db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaistring MinidumpModule::debug_file() const { 1890af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1891af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModule for debug_file"; 1892db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return ""; 1893af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1894db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1895d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek if (!has_debug_info_) 1896d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek return ""; 1897d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek 1898db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai string file; 1899db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // Prefer the CodeView record if present. 190028e5990b579f69aaeded6ab4c2b68d841424b4cdmmentovai if (cv_record_) { 190148dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai if (cv_record_signature_ == MD_CVINFOPDB70_SIGNATURE) { 190248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // It's actually an MDCVInfoPDB70 structure. 190348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai const MDCVInfoPDB70* cv_record_70 = 190448dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai reinterpret_cast<const MDCVInfoPDB70*>(&(*cv_record_)[0]); 190548dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai assert(cv_record_70->cv_signature == MD_CVINFOPDB70_SIGNATURE); 190648dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai 1907db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // GetCVRecord guarantees pdb_file_name is null-terminated. 1908db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai file = reinterpret_cast<const char*>(cv_record_70->pdb_file_name); 190948dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai } else if (cv_record_signature_ == MD_CVINFOPDB20_SIGNATURE) { 191048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // It's actually an MDCVInfoPDB20 structure. 1911db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai const MDCVInfoPDB20* cv_record_20 = 1912db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai reinterpret_cast<const MDCVInfoPDB20*>(&(*cv_record_)[0]); 191348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai assert(cv_record_20->cv_header.signature == MD_CVINFOPDB20_SIGNATURE); 1914db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1915db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // GetCVRecord guarantees pdb_file_name is null-terminated. 1916db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai file = reinterpret_cast<const char*>(cv_record_20->pdb_file_name); 1917db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1918db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1919db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // If there's a CodeView record but it doesn't match a known signature, 192048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // try the miscellaneous record. 1921db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1922db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1923db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai if (file.empty()) { 1924db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // No usable CodeView record. Try the miscellaneous debug record. 192528e5990b579f69aaeded6ab4c2b68d841424b4cdmmentovai if (misc_record_) { 192628e5990b579f69aaeded6ab4c2b68d841424b4cdmmentovai const MDImageDebugMisc* misc_record = 192728e5990b579f69aaeded6ab4c2b68d841424b4cdmmentovai reinterpret_cast<const MDImageDebugMisc *>(&(*misc_record_)[0]); 1928db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai if (!misc_record->unicode) { 1929db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // If it's not Unicode, just stuff it into the string. It's unclear 1930db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // if misc_record->data is 0-terminated, so use an explicit size. 1931db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai file = string( 1932db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai reinterpret_cast<const char*>(misc_record->data), 19332e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai module_.misc_record.data_size - MDImageDebugMisc_minsize); 1934db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } else { 1935db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // There's a misc_record but it encodes the debug filename in UTF-16. 1936db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // (Actually, because miscellaneous records are so old, it's probably 1937db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // UCS-2.) Convert it to UTF-8 for congruity with the other strings 1938db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // that this method (and all other methods in the Minidump family) 1939db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // return. 1940db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1941db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai unsigned int bytes = 19422e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai module_.misc_record.data_size - MDImageDebugMisc_minsize; 1943db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai if (bytes % 2 == 0) { 1944db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai unsigned int utf16_words = bytes / 2; 1945db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 19466162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com // UTF16ToUTF8 expects a vector<uint16_t>, so create a temporary one 1947db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // and copy the UTF-16 data into it. 19486162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com vector<uint16_t> string_utf16(utf16_words); 1949db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai if (utf16_words) 1950db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai memcpy(&string_utf16[0], &misc_record->data, bytes); 1951db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1952db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // GetMiscRecord already byte-swapped the data[] field if it contains 1953db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // UTF-16, so pass false as the swap argument. 1954db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai scoped_ptr<string> new_file(UTF16ToUTF8(string_utf16, false)); 1955db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai file = *new_file; 1956db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1957db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1958db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1959db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 1960db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1961bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com // Relatively common case 1962bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com BPLOG_IF(INFO, file.empty()) << "MinidumpModule could not determine " 1963bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com "debug_file for " << *name_; 1964af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 1965db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return file; 1966db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 1967db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1968db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1969db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaistring MinidumpModule::debug_identifier() const { 1970af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 1971af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModule for debug_identifier"; 1972db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return ""; 1973af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 1974db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1975d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek if (!has_debug_info_) 1976d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek return ""; 1977d732add382b95d1460b281f191f0f0e4397eaf51ted.mielczarek 1978db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai string identifier; 1979db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 1980db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // Use the CodeView record if present. 198128e5990b579f69aaeded6ab4c2b68d841424b4cdmmentovai if (cv_record_) { 198248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai if (cv_record_signature_ == MD_CVINFOPDB70_SIGNATURE) { 198348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // It's actually an MDCVInfoPDB70 structure. 198448dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai const MDCVInfoPDB70* cv_record_70 = 198548dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai reinterpret_cast<const MDCVInfoPDB70*>(&(*cv_record_)[0]); 198648dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai assert(cv_record_70->cv_signature == MD_CVINFOPDB70_SIGNATURE); 198748dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai 1988c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai // Use the same format that the MS symbol server uses in filesystem 1989c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai // hierarchies. 1990db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai char identifier_string[41]; 1991db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai snprintf(identifier_string, sizeof(identifier_string), 1992c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai "%08X%04X%04X%02X%02X%02X%02X%02X%02X%02X%02X%x", 1993db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data1, 1994db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data2, 1995db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data3, 1996db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data4[0], 1997db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data4[1], 1998db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data4[2], 1999db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data4[3], 2000db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data4[4], 2001db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data4[5], 2002db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data4[6], 2003db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->signature.data4[7], 2004db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai cv_record_70->age); 2005db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai identifier = identifier_string; 200648dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai } else if (cv_record_signature_ == MD_CVINFOPDB20_SIGNATURE) { 200748dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // It's actually an MDCVInfoPDB20 structure. 2008db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai const MDCVInfoPDB20* cv_record_20 = 2009db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai reinterpret_cast<const MDCVInfoPDB20*>(&(*cv_record_)[0]); 201048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai assert(cv_record_20->cv_header.signature == MD_CVINFOPDB20_SIGNATURE); 2011db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2012c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai // Use the same format that the MS symbol server uses in filesystem 2013c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai // hierarchies. 2014db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai char identifier_string[17]; 2015db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai snprintf(identifier_string, sizeof(identifier_string), 2016c7b6c11f3287ec4fe52e6ad915ed9c906ed8d301mmentovai "%08X%x", cv_record_20->signature, cv_record_20->age); 2017db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai identifier = identifier_string; 2018db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 2019db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 2020db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 202148dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // TODO(mmentovai): if there's no usable CodeView record, there might be a 2022db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // miscellaneous debug record. It only carries a filename, though, and no 2023db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // identifier. I'm not sure what the right thing to do for the identifier 2024db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // is in that case, but I don't expect to find many modules without a 2025e5dc60822e5938fea2ae892ccddb906641ba174emmentovai // CodeView record (or some other Breakpad extension structure in place of 2026db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // a CodeView record). Treat it as an error (empty identifier) for now. 2027db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2028db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // TODO(mmentovai): on the Mac, provide fallbacks as in code_identifier(). 2029db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2030bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com // Relatively common case 2031bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com BPLOG_IF(INFO, identifier.empty()) << "MinidumpModule could not determine " 2032bb87ebd809e4437c953e9008a73927cd7e1c05efjessicag.feedback@gmail.com "debug_identifier for " << *name_; 2033af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 2034db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return identifier; 2035db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 2036db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2037db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2038db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaistring MinidumpModule::version() const { 2039af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2040af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModule for version"; 2041db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return ""; 2042af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2043db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2044db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai string version; 2045db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2046db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai if (module_.version_info.signature == MD_VSFIXEDFILEINFO_SIGNATURE && 2047db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_.version_info.struct_version & MD_VSFIXEDFILEINFO_VERSION) { 2048db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai char version_string[24]; 2049db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai snprintf(version_string, sizeof(version_string), "%u.%u.%u.%u", 2050db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_.version_info.file_version_hi >> 16, 2051db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_.version_info.file_version_hi & 0xffff, 2052db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_.version_info.file_version_lo >> 16, 2053db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_.version_info.file_version_lo & 0xffff); 2054db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai version = version_string; 2055db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 2056db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2057db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // TODO(mmentovai): possibly support other struct types in place of 2058db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // the one used with MD_VSFIXEDFILEINFO_SIGNATURE. We can possibly use 2059db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // a different structure that better represents versioning facilities on 2060db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // Mac OS X and Linux, instead of forcing them to adhere to the dotted 2061db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // quad of 16-bit ints that Windows uses. 2062db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2063af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(INFO, version.empty()) << "MinidumpModule could not determine " 2064af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "version for " << *name_; 2065af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 2066db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return version; 2067db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 2068db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2069db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2070db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaiconst CodeModule* MinidumpModule::Copy() const { 2071db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return new BasicCodeModule(this); 2072db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 2073db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2074db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 20756162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comconst uint8_t* MinidumpModule::GetCVRecord(uint32_t* size) { 2076af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!module_valid_) { 2077af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModule for GetCVRecord"; 20783261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2079af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 20803261e8b6eac44a41341f112821482bee6c940c98mmentovai 20813261e8b6eac44a41341f112821482bee6c940c98mmentovai if (!cv_record_) { 208248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // This just guards against 0-sized CodeView records; more specific checks 208348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // are used when the signature is checked against various structure types. 2084af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (module_.cv_record.data_size == 0) { 20853261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2086af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 20873261e8b6eac44a41341f112821482bee6c940c98mmentovai 2088af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->SeekSet(module_.cv_record.rva)) { 2089af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule could not seek to CodeView record"; 20903261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2091af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 20923261e8b6eac44a41341f112821482bee6c940c98mmentovai 2093e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (module_.cv_record.data_size > max_cv_bytes_) { 2094e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "MinidumpModule CodeView record size " << 2095e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai module_.cv_record.data_size << " exceeds maximum " << 2096e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai max_cv_bytes_; 2097e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai return NULL; 2098e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai } 20993261e8b6eac44a41341f112821482bee6c940c98mmentovai 21003261e8b6eac44a41341f112821482bee6c940c98mmentovai // Allocating something that will be accessed as MDCVInfoPDB70 or 21016162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com // MDCVInfoPDB20 but is allocated as uint8_t[] can cause alignment 21023261e8b6eac44a41341f112821482bee6c940c98mmentovai // problems. x86 and ppc are able to cope, though. This allocation 21033261e8b6eac44a41341f112821482bee6c940c98mmentovai // style is needed because the MDCVInfoPDB70 or MDCVInfoPDB20 are 21043261e8b6eac44a41341f112821482bee6c940c98mmentovai // variable-sized due to their pdb_file_name fields; these structures 21052e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai // are not MDCVInfoPDB70_minsize or MDCVInfoPDB20_minsize and treating 21063261e8b6eac44a41341f112821482bee6c940c98mmentovai // them as such would result in incomplete structures or overruns. 21076162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com scoped_ptr< vector<uint8_t> > cv_record( 21086162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com new vector<uint8_t>(module_.cv_record.data_size)); 21093261e8b6eac44a41341f112821482bee6c940c98mmentovai 2110af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&(*cv_record)[0], module_.cv_record.data_size)) { 2111af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule could not read CodeView record"; 21123261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2113af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 21143261e8b6eac44a41341f112821482bee6c940c98mmentovai 21156162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t signature = MD_CVINFOUNKNOWN_SIGNATURE; 211648dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai if (module_.cv_record.data_size > sizeof(signature)) { 211748dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai MDCVInfoPDB70* cv_record_signature = 211848dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai reinterpret_cast<MDCVInfoPDB70*>(&(*cv_record)[0]); 211948dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai signature = cv_record_signature->cv_signature; 212048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai if (minidump_->swap()) 212148dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai Swap(&signature); 212248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai } 21233261e8b6eac44a41341f112821482bee6c940c98mmentovai 21243261e8b6eac44a41341f112821482bee6c940c98mmentovai if (signature == MD_CVINFOPDB70_SIGNATURE) { 21253261e8b6eac44a41341f112821482bee6c940c98mmentovai // Now that the structure type is known, recheck the size. 21262e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai if (MDCVInfoPDB70_minsize > module_.cv_record.data_size) { 2127af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule CodeView7 record size mismatch, " << 21282e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai MDCVInfoPDB70_minsize << " > " << 2129af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai module_.cv_record.data_size; 21303261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2131af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 21323261e8b6eac44a41341f112821482bee6c940c98mmentovai 21333261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) { 213448dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai MDCVInfoPDB70* cv_record_70 = 213548dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai reinterpret_cast<MDCVInfoPDB70*>(&(*cv_record)[0]); 21363261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&cv_record_70->cv_signature); 21373261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&cv_record_70->signature); 21383261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&cv_record_70->age); 21393261e8b6eac44a41341f112821482bee6c940c98mmentovai // Don't swap cv_record_70.pdb_file_name because it's an array of 8-bit 214048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // quantities. (It's a path, is it UTF-8?) 21413261e8b6eac44a41341f112821482bee6c940c98mmentovai } 214248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai 214348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // The last field of either structure is null-terminated 8-bit character 214448dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // data. Ensure that it's null-terminated. 2145af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if ((*cv_record)[module_.cv_record.data_size - 1] != '\0') { 2146af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule CodeView7 record string is not " 2147af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "0-terminated"; 214848dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai return NULL; 2149af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 21503261e8b6eac44a41341f112821482bee6c940c98mmentovai } else if (signature == MD_CVINFOPDB20_SIGNATURE) { 215148dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // Now that the structure type is known, recheck the size. 21522e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai if (MDCVInfoPDB20_minsize > module_.cv_record.data_size) { 2153af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule CodeView2 record size mismatch, " << 21542e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai MDCVInfoPDB20_minsize << " > " << 2155af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai module_.cv_record.data_size; 215648dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai return NULL; 2157af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 21583261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) { 21593261e8b6eac44a41341f112821482bee6c940c98mmentovai MDCVInfoPDB20* cv_record_20 = 216048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai reinterpret_cast<MDCVInfoPDB20*>(&(*cv_record)[0]); 21613261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&cv_record_20->cv_header.signature); 21623261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&cv_record_20->cv_header.offset); 21633261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&cv_record_20->signature); 21643261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&cv_record_20->age); 21653261e8b6eac44a41341f112821482bee6c940c98mmentovai // Don't swap cv_record_20.pdb_file_name because it's an array of 8-bit 21663261e8b6eac44a41341f112821482bee6c940c98mmentovai // quantities. (It's a path, is it UTF-8?) 21673261e8b6eac44a41341f112821482bee6c940c98mmentovai } 216848dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai 216948dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // The last field of either structure is null-terminated 8-bit character 217048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // data. Ensure that it's null-terminated. 2171af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if ((*cv_record)[module_.cv_record.data_size - 1] != '\0') { 2172af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MindumpModule CodeView2 record string is not " 2173af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "0-terminated"; 217448dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai return NULL; 2175af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 21763261e8b6eac44a41341f112821482bee6c940c98mmentovai } 21773261e8b6eac44a41341f112821482bee6c940c98mmentovai 217848dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // If the signature doesn't match something above, it's not something 2179e5dc60822e5938fea2ae892ccddb906641ba174emmentovai // that Breakpad can presently handle directly. Because some modules in 218048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // the wild contain such CodeView records as MD_CVINFOCV50_SIGNATURE, 218148dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // don't bail out here - allow the data to be returned to the user, 218248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai // although byte-swapping can't be done. 21833261e8b6eac44a41341f112821482bee6c940c98mmentovai 21843261e8b6eac44a41341f112821482bee6c940c98mmentovai // Store the vector type because that's how storage was allocated, but 21856162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com // return it casted to uint8_t*. 21863261e8b6eac44a41341f112821482bee6c940c98mmentovai cv_record_ = cv_record.release(); 218748dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_signature_ = signature; 21883261e8b6eac44a41341f112821482bee6c940c98mmentovai } 21893261e8b6eac44a41341f112821482bee6c940c98mmentovai 219048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai if (size) 219148dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai *size = module_.cv_record.data_size; 219248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai 21933261e8b6eac44a41341f112821482bee6c940c98mmentovai return &(*cv_record_)[0]; 21943261e8b6eac44a41341f112821482bee6c940c98mmentovai} 21953261e8b6eac44a41341f112821482bee6c940c98mmentovai 21963261e8b6eac44a41341f112821482bee6c940c98mmentovai 21976162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comconst MDImageDebugMisc* MinidumpModule::GetMiscRecord(uint32_t* size) { 2198af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!module_valid_) { 2199af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModule for GetMiscRecord"; 22003261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2201af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 22023261e8b6eac44a41341f112821482bee6c940c98mmentovai 22033261e8b6eac44a41341f112821482bee6c940c98mmentovai if (!misc_record_) { 2204af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (module_.misc_record.data_size == 0) { 22053261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2206af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 22073261e8b6eac44a41341f112821482bee6c940c98mmentovai 22082e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai if (MDImageDebugMisc_minsize > module_.misc_record.data_size) { 2209af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule miscellaneous debugging record " 22102e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai "size mismatch, " << MDImageDebugMisc_minsize << " > " << 2211af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai module_.misc_record.data_size; 22123261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2213af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2214af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 2215af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->SeekSet(module_.misc_record.rva)) { 2216af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule could not seek to miscellaneous " 2217af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "debugging record"; 2218af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return NULL; 2219af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 22203261e8b6eac44a41341f112821482bee6c940c98mmentovai 2221e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (module_.misc_record.data_size > max_misc_bytes_) { 2222e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "MinidumpModule miscellaneous debugging record size " << 2223e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai module_.misc_record.data_size << " exceeds maximum " << 2224e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai max_misc_bytes_; 2225e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai return NULL; 2226e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai } 22273261e8b6eac44a41341f112821482bee6c940c98mmentovai 22283261e8b6eac44a41341f112821482bee6c940c98mmentovai // Allocating something that will be accessed as MDImageDebugMisc but 22296162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com // is allocated as uint8_t[] can cause alignment problems. x86 and 22303261e8b6eac44a41341f112821482bee6c940c98mmentovai // ppc are able to cope, though. This allocation style is needed 22313261e8b6eac44a41341f112821482bee6c940c98mmentovai // because the MDImageDebugMisc is variable-sized due to its data field; 22322e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai // this structure is not MDImageDebugMisc_minsize and treating it as such 22333261e8b6eac44a41341f112821482bee6c940c98mmentovai // would result in an incomplete structure or an overrun. 22346162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com scoped_ptr< vector<uint8_t> > misc_record_mem( 22356162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com new vector<uint8_t>(module_.misc_record.data_size)); 22363261e8b6eac44a41341f112821482bee6c940c98mmentovai MDImageDebugMisc* misc_record = 22373261e8b6eac44a41341f112821482bee6c940c98mmentovai reinterpret_cast<MDImageDebugMisc*>(&(*misc_record_mem)[0]); 22383261e8b6eac44a41341f112821482bee6c940c98mmentovai 2239af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(misc_record, module_.misc_record.data_size)) { 2240af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule could not read miscellaneous debugging " 2241af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "record"; 22423261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2243af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 22443261e8b6eac44a41341f112821482bee6c940c98mmentovai 22453261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) { 22463261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_record->data_type); 22473261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_record->length); 22483261e8b6eac44a41341f112821482bee6c940c98mmentovai // Don't swap misc_record.unicode because it's an 8-bit quantity. 22493261e8b6eac44a41341f112821482bee6c940c98mmentovai // Don't swap the reserved fields for the same reason, and because 22503261e8b6eac44a41341f112821482bee6c940c98mmentovai // they don't contain any valid data. 22513261e8b6eac44a41341f112821482bee6c940c98mmentovai if (misc_record->unicode) { 22523261e8b6eac44a41341f112821482bee6c940c98mmentovai // There is a potential alignment problem, but shouldn't be a problem 22533261e8b6eac44a41341f112821482bee6c940c98mmentovai // in practice due to the layout of MDImageDebugMisc. 22546162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint16_t* data16 = reinterpret_cast<uint16_t*>(&(misc_record->data)); 22553261e8b6eac44a41341f112821482bee6c940c98mmentovai unsigned int dataBytes = module_.misc_record.data_size - 22562e0e2234b9e9d7d82c4c3c20396bdf8f18007e6cmmentovai MDImageDebugMisc_minsize; 22573562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(data16, dataBytes); 22583261e8b6eac44a41341f112821482bee6c940c98mmentovai } 22593261e8b6eac44a41341f112821482bee6c940c98mmentovai } 22603261e8b6eac44a41341f112821482bee6c940c98mmentovai 2261af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (module_.misc_record.data_size != misc_record->length) { 2262af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule miscellaneous debugging record data " 2263af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "size mismatch, " << module_.misc_record.data_size << 2264af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai " != " << misc_record->length; 22653261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2266af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 22673261e8b6eac44a41341f112821482bee6c940c98mmentovai 22683261e8b6eac44a41341f112821482bee6c940c98mmentovai // Store the vector type because that's how storage was allocated, but 22693261e8b6eac44a41341f112821482bee6c940c98mmentovai // return it casted to MDImageDebugMisc*. 22703261e8b6eac44a41341f112821482bee6c940c98mmentovai misc_record_ = misc_record_mem.release(); 22713261e8b6eac44a41341f112821482bee6c940c98mmentovai } 22723261e8b6eac44a41341f112821482bee6c940c98mmentovai 227348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai if (size) 227448dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai *size = module_.misc_record.data_size; 227548dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai 22763261e8b6eac44a41341f112821482bee6c940c98mmentovai return reinterpret_cast<MDImageDebugMisc*>(&(*misc_record_)[0]); 22773261e8b6eac44a41341f112821482bee6c940c98mmentovai} 22783261e8b6eac44a41341f112821482bee6c940c98mmentovai 22793261e8b6eac44a41341f112821482bee6c940c98mmentovai 22803261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpModule::Print() { 2281af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2282af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModule cannot print invalid data"; 22833261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 2284af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 22853261e8b6eac44a41341f112821482bee6c940c98mmentovai 22863261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MDRawModule\n"); 2287c27cf3e3959189f78fe2de40405987c3f33488cemmentovai printf(" base_of_image = 0x%" PRIx64 "\n", 22883261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.base_of_image); 22893261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" size_of_image = 0x%x\n", 22903261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.size_of_image); 22913261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" checksum = 0x%x\n", 22923261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.checksum); 22939c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org printf(" time_date_stamp = 0x%x %s\n", 22949c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org module_.time_date_stamp, 22959c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org TimeTToUTCString(module_.time_date_stamp).c_str()); 22963261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" module_name_rva = 0x%x\n", 22973261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.module_name_rva); 22983261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.signature = 0x%x\n", 22993261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.signature); 23003261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.struct_version = 0x%x\n", 23013261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.struct_version); 23023261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.file_version = 0x%x:0x%x\n", 23033261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_version_hi, 23043261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_version_lo); 23053261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.product_version = 0x%x:0x%x\n", 23063261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.product_version_hi, 23073261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.product_version_lo); 23083261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.file_flags_mask = 0x%x\n", 23093261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_flags_mask); 23103261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.file_flags = 0x%x\n", 23113261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_flags); 23123261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.file_os = 0x%x\n", 23133261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_os); 23143261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.file_type = 0x%x\n", 23153261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_type); 23163261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.file_subtype = 0x%x\n", 23173261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_subtype); 23183261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version_info.file_date = 0x%x:0x%x\n", 23193261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_date_hi, 23203261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.version_info.file_date_lo); 23213261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" cv_record.data_size = %d\n", 23223261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.cv_record.data_size); 23233261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" cv_record.rva = 0x%x\n", 23243261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.cv_record.rva); 23253261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" misc_record.data_size = %d\n", 23263261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.misc_record.data_size); 23273261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" misc_record.rva = 0x%x\n", 23283261e8b6eac44a41341f112821482bee6c940c98mmentovai module_.misc_record.rva); 23293261e8b6eac44a41341f112821482bee6c940c98mmentovai 2330db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai printf(" (code_file) = \"%s\"\n", code_file().c_str()); 2331db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai printf(" (code_identifier) = \"%s\"\n", 2332db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai code_identifier().c_str()); 23333261e8b6eac44a41341f112821482bee6c940c98mmentovai 23346162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t cv_record_size; 23356162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com const uint8_t *cv_record = GetCVRecord(&cv_record_size); 23363261e8b6eac44a41341f112821482bee6c940c98mmentovai if (cv_record) { 233748dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai if (cv_record_signature_ == MD_CVINFOPDB70_SIGNATURE) { 233848dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai const MDCVInfoPDB70* cv_record_70 = 233948dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai reinterpret_cast<const MDCVInfoPDB70*>(cv_record); 234048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai assert(cv_record_70->cv_signature == MD_CVINFOPDB70_SIGNATURE); 234148dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai 23423261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record).cv_signature = 0x%x\n", 234348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_70->cv_signature); 23443261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record).signature = %08x-%04x-%04x-%02x%02x-", 234548dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_70->signature.data1, 234648dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_70->signature.data2, 234748dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_70->signature.data3, 234848dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_70->signature.data4[0], 234948dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_70->signature.data4[1]); 23503261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int guidIndex = 2; 23513261e8b6eac44a41341f112821482bee6c940c98mmentovai guidIndex < 8; 23523261e8b6eac44a41341f112821482bee6c940c98mmentovai ++guidIndex) { 235348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai printf("%02x", cv_record_70->signature.data4[guidIndex]); 23543261e8b6eac44a41341f112821482bee6c940c98mmentovai } 23553261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 23563261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record).age = %d\n", 235748dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_70->age); 23583261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record).pdb_file_name = \"%s\"\n", 235948dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_record_70->pdb_file_name); 236048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai } else if (cv_record_signature_ == MD_CVINFOPDB20_SIGNATURE) { 23613261e8b6eac44a41341f112821482bee6c940c98mmentovai const MDCVInfoPDB20* cv_record_20 = 236248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai reinterpret_cast<const MDCVInfoPDB20*>(cv_record); 236348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai assert(cv_record_20->cv_header.signature == MD_CVINFOPDB20_SIGNATURE); 236448dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai 23653261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record).cv_header.signature = 0x%x\n", 23663261e8b6eac44a41341f112821482bee6c940c98mmentovai cv_record_20->cv_header.signature); 23673261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record).cv_header.offset = 0x%x\n", 23683261e8b6eac44a41341f112821482bee6c940c98mmentovai cv_record_20->cv_header.offset); 23699c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org printf(" (cv_record).signature = 0x%x %s\n", 23709c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org cv_record_20->signature, 23719c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org TimeTToUTCString(cv_record_20->signature).c_str()); 23723261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record).age = %d\n", 23733261e8b6eac44a41341f112821482bee6c940c98mmentovai cv_record_20->age); 23743261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record).pdb_file_name = \"%s\"\n", 23753261e8b6eac44a41341f112821482bee6c940c98mmentovai cv_record_20->pdb_file_name); 237648dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai } else { 237748dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai printf(" (cv_record) = "); 237848dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai for (unsigned int cv_byte_index = 0; 237948dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai cv_byte_index < cv_record_size; 238048dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai ++cv_byte_index) { 238148dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai printf("%02x", cv_record[cv_byte_index]); 238248dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai } 238348dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai printf("\n"); 23843261e8b6eac44a41341f112821482bee6c940c98mmentovai } 23853261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 23863261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (cv_record) = (null)\n"); 23873261e8b6eac44a41341f112821482bee6c940c98mmentovai } 23883261e8b6eac44a41341f112821482bee6c940c98mmentovai 238948dab62c2df1621db0e5b57f400e1fc38c1b37eemmentovai const MDImageDebugMisc* misc_record = GetMiscRecord(NULL); 23903261e8b6eac44a41341f112821482bee6c940c98mmentovai if (misc_record) { 23913261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (misc_record).data_type = 0x%x\n", 23923261e8b6eac44a41341f112821482bee6c940c98mmentovai misc_record->data_type); 23933261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (misc_record).length = 0x%x\n", 23943261e8b6eac44a41341f112821482bee6c940c98mmentovai misc_record->length); 23953261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (misc_record).unicode = %d\n", 23963261e8b6eac44a41341f112821482bee6c940c98mmentovai misc_record->unicode); 23979c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org if (misc_record->unicode) { 23989c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org string misc_record_data_utf8; 23999c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org ConvertUTF16BufferToUTF8String( 24009c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org reinterpret_cast<const uint16_t*>(misc_record->data), 24019c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org misc_record->length - offsetof(MDImageDebugMisc, data), 24029c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org &misc_record_data_utf8, 24039c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org false); // already swapped 24049c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org printf(" (misc_record).data = \"%s\"\n", 24059c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org misc_record_data_utf8.c_str()); 24069c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org } else { 24073261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (misc_record).data = \"%s\"\n", 24083261e8b6eac44a41341f112821482bee6c940c98mmentovai misc_record->data); 24099c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org } 24103261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 24113261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (misc_record) = (null)\n"); 24123261e8b6eac44a41341f112821482bee6c940c98mmentovai } 24133261e8b6eac44a41341f112821482bee6c940c98mmentovai 2414db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai printf(" (debug_file) = \"%s\"\n", debug_file().c_str()); 2415db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai printf(" (debug_identifier) = \"%s\"\n", 2416db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai debug_identifier().c_str()); 2417db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai printf(" (version) = \"%s\"\n", version().c_str()); 24183261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 24193261e8b6eac44a41341f112821482bee6c940c98mmentovai} 24203261e8b6eac44a41341f112821482bee6c940c98mmentovai 24213261e8b6eac44a41341f112821482bee6c940c98mmentovai 24223261e8b6eac44a41341f112821482bee6c940c98mmentovai// 24233261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpModuleList 24243261e8b6eac44a41341f112821482bee6c940c98mmentovai// 24253261e8b6eac44a41341f112821482bee6c940c98mmentovai 24263261e8b6eac44a41341f112821482bee6c940c98mmentovai 24276162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint32_t MinidumpModuleList::max_modules_ = 1024; 2428e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 2429e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 24303261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpModuleList::MinidumpModuleList(Minidump* minidump) 243153d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpStream(minidump), 24326162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com range_map_(new RangeMap<uint64_t, unsigned int>()), 243353d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai modules_(NULL), 243453d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai module_count_(0) { 24353261e8b6eac44a41341f112821482bee6c940c98mmentovai} 24363261e8b6eac44a41341f112821482bee6c940c98mmentovai 24373261e8b6eac44a41341f112821482bee6c940c98mmentovai 24383261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpModuleList::~MinidumpModuleList() { 2439fe82bf24a93d9d3affd614aaa23f2f018a733d8emmentovai delete range_map_; 24403261e8b6eac44a41341f112821482bee6c940c98mmentovai delete modules_; 24413261e8b6eac44a41341f112821482bee6c940c98mmentovai} 24423261e8b6eac44a41341f112821482bee6c940c98mmentovai 24433261e8b6eac44a41341f112821482bee6c940c98mmentovai 24446162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpModuleList::Read(uint32_t expected_size) { 24453261e8b6eac44a41341f112821482bee6c940c98mmentovai // Invalidate cached data. 2446fe82bf24a93d9d3affd614aaa23f2f018a733d8emmentovai range_map_->Clear(); 24473261e8b6eac44a41341f112821482bee6c940c98mmentovai delete modules_; 24483261e8b6eac44a41341f112821482bee6c940c98mmentovai modules_ = NULL; 24493261e8b6eac44a41341f112821482bee6c940c98mmentovai module_count_ = 0; 24503261e8b6eac44a41341f112821482bee6c940c98mmentovai 24513261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 24523261e8b6eac44a41341f112821482bee6c940c98mmentovai 24536162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t module_count; 2454af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (expected_size < sizeof(module_count)) { 2455af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModuleList count size mismatch, " << 2456af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai expected_size << " < " << sizeof(module_count); 24573261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 2458af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2459af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&module_count, sizeof(module_count))) { 2460af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModuleList could not read module count"; 24613261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 2462af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 24633261e8b6eac44a41341f112821482bee6c940c98mmentovai 24643261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) 24653261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&module_count); 24663261e8b6eac44a41341f112821482bee6c940c98mmentovai 24676162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com if (module_count > numeric_limits<uint32_t>::max() / MD_MODULE_SIZE) { 2468fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai BPLOG(ERROR) << "MinidumpModuleList module count " << module_count << 2469fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai " would cause multiplication overflow"; 2470fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai return false; 2471fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai } 2472fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai 24733261e8b6eac44a41341f112821482bee6c940c98mmentovai if (expected_size != sizeof(module_count) + 24743261e8b6eac44a41341f112821482bee6c940c98mmentovai module_count * MD_MODULE_SIZE) { 2475ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai // may be padded with 4 bytes on 64bit ABIs for alignment 2476ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai if (expected_size == sizeof(module_count) + 4 + 2477ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai module_count * MD_MODULE_SIZE) { 24786162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t useless; 2479ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai if (!minidump_->ReadBytes(&useless, 4)) { 2480f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "MinidumpModuleList cannot read modulelist padded " 2481f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com "bytes"; 2482ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai return false; 2483ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 2484ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } else { 2485ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai BPLOG(ERROR) << "MinidumpModuleList size mismatch, " << expected_size << 2486ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai " != " << sizeof(module_count) + 2487ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai module_count * MD_MODULE_SIZE; 2488ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai return false; 2489ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 24903261e8b6eac44a41341f112821482bee6c940c98mmentovai } 24913261e8b6eac44a41341f112821482bee6c940c98mmentovai 2492e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (module_count > max_modules_) { 2493e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "MinidumpModuleList count " << module_count_ << 2494e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai " exceeds maximum " << max_modules_; 2495e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai return false; 2496e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai } 2497e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 2498e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (module_count != 0) { 2499373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai scoped_ptr<MinidumpModules> modules( 2500373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai new MinidumpModules(module_count, MinidumpModule(minidump_))); 25013261e8b6eac44a41341f112821482bee6c940c98mmentovai 2502373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai for (unsigned int module_index = 0; 2503373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai module_index < module_count; 2504373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai ++module_index) { 2505373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai MinidumpModule* module = &(*modules)[module_index]; 25063261e8b6eac44a41341f112821482bee6c940c98mmentovai 2507373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Assume that the file offset is correct after the last read. 2508af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!module->Read()) { 2509af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModuleList could not read module " << 2510af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai module_index << "/" << module_count; 2511373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 2512af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2513db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai } 2514db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2515db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // Loop through the module list once more to read additional data and 2516db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // build the range map. This is done in a second pass because 2517db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // MinidumpModule::ReadAuxiliaryData seeks around, and if it were 2518db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // included in the loop above, additional seeks would be needed where 2519db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // none are now to read contiguous data. 2520db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai for (unsigned int module_index = 0; 2521db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai module_index < module_count; 2522db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai ++module_index) { 2523db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai MinidumpModule* module = &(*modules)[module_index]; 2524db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 252561ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai // ReadAuxiliaryData fails if any data that the module indicates should 252661ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai // exist is missing, but we treat some such cases as valid anyway. See 252761ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai // issue #222: if a debugging record is of a format that's too large to 252861ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai // handle, it shouldn't render the entire dump invalid. Check module 252961ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai // validity before giving up. 253061ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai if (!module->ReadAuxiliaryData() && !module->valid()) { 253161ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai BPLOG(ERROR) << "MinidumpModuleList could not read required module " 253261ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai "auxiliary data for module " << 253361ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai module_index << "/" << module_count; 253461ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai return false; 2535af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2536af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 2537af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai // It is safe to use module->code_file() after successfully calling 253861ea8bf0d5c2cf652e8d75605f770d0f9733acfemmentovai // module->ReadAuxiliaryData or noting that the module is valid. 25393261e8b6eac44a41341f112821482bee6c940c98mmentovai 25406162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t base_address = module->base_address(); 25416162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t module_size = module->size(); 25426162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com if (base_address == static_cast<uint64_t>(-1)) { 2543af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModuleList found bad base address " 2544af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "for module " << module_index << "/" << module_count << 2545af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai ", " << module->code_file(); 2546373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 2547af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 25483261e8b6eac44a41341f112821482bee6c940c98mmentovai 2549af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!range_map_->StoreRange(base_address, module_size, module_index)) { 2550746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org // Android's shared memory implementation /dev/ashmem can contain 2551746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org // duplicate entries for JITted code, so ignore these. 2552746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org // TODO(wfh): Remove this code when Android is fixed. 2553746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org // See https://crbug.com/439531 2554746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org const string kDevAshmem("/dev/ashmem/"); 2555746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org if (module->code_file().compare( 2556746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org 0, kDevAshmem.length(), kDevAshmem) != 0) { 2557746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org BPLOG(ERROR) << "MinidumpModuleList could not store module " << 2558746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org module_index << "/" << module_count << ", " << 2559746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org module->code_file() << ", " << 2560746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org HexString(base_address) << "+" << 2561746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org HexString(module_size); 2562746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org return false; 2563746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org } else { 2564746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org BPLOG(INFO) << "MinidumpModuleList ignoring overlapping module " << 2565746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org module_index << "/" << module_count << ", " << 2566746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org module->code_file() << ", " << 2567746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org HexString(base_address) << "+" << 2568746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org HexString(module_size); 2569746f65a45e1cf52110ed677da19f87fa2eb6aefawfh@chromium.org } 2570af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2571373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 2572373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai 2573373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai modules_ = modules.release(); 25743261e8b6eac44a41341f112821482bee6c940c98mmentovai } 25753261e8b6eac44a41341f112821482bee6c940c98mmentovai 25763261e8b6eac44a41341f112821482bee6c940c98mmentovai module_count_ = module_count; 25773261e8b6eac44a41341f112821482bee6c940c98mmentovai 25783261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 25793261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 25803261e8b6eac44a41341f112821482bee6c940c98mmentovai} 25813261e8b6eac44a41341f112821482bee6c940c98mmentovai 25823261e8b6eac44a41341f112821482bee6c940c98mmentovai 2583db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaiconst MinidumpModule* MinidumpModuleList::GetModuleForAddress( 25846162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t address) const { 2585af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2586af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModuleList for GetModuleForAddress"; 25873261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2588af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 25893261e8b6eac44a41341f112821482bee6c940c98mmentovai 2590db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai unsigned int module_index; 2591af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!range_map_->RetrieveRange(address, &module_index, NULL, NULL)) { 2592af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "MinidumpModuleList has no module at " << 2593af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(address); 2594db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return NULL; 2595af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2596db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2597db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return GetModuleAtIndex(module_index); 25983261e8b6eac44a41341f112821482bee6c940c98mmentovai} 25993261e8b6eac44a41341f112821482bee6c940c98mmentovai 26003261e8b6eac44a41341f112821482bee6c940c98mmentovai 2601db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaiconst MinidumpModule* MinidumpModuleList::GetMainModule() const { 2602af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2603af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModuleList for GetMainModule"; 26043261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2605af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 26063261e8b6eac44a41341f112821482bee6c940c98mmentovai 2607db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // The main code module is the first one present in a minidump file's 2608db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai // MDRawModuleList. 2609327783c42fcc2062bfe6c118c54c431ac6b5ffcfmkrebs@chromium.org return GetModuleAtIndex(0); 2610db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 2611db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2612db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2613db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaiconst MinidumpModule* MinidumpModuleList::GetModuleAtSequence( 2614db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai unsigned int sequence) const { 2615af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2616af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModuleList for GetModuleAtSequence"; 2617af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return NULL; 2618af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2619af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 2620af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (sequence >= module_count_) { 2621af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModuleList sequence out of range: " << 2622af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai sequence << "/" << module_count_; 2623db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return NULL; 2624af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2625db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 26263261e8b6eac44a41341f112821482bee6c940c98mmentovai unsigned int module_index; 2627af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!range_map_->RetrieveRangeAtIndex(sequence, &module_index, NULL, NULL)) { 2628af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModuleList has no module at sequence " << sequence; 26293261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2630af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 26313261e8b6eac44a41341f112821482bee6c940c98mmentovai 26323261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetModuleAtIndex(module_index); 26333261e8b6eac44a41341f112821482bee6c940c98mmentovai} 26343261e8b6eac44a41341f112821482bee6c940c98mmentovai 26353261e8b6eac44a41341f112821482bee6c940c98mmentovai 2636db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaiconst MinidumpModule* MinidumpModuleList::GetModuleAtIndex( 2637db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai unsigned int index) const { 2638af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2639af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpModuleList for GetModuleAtIndex"; 2640af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return NULL; 2641af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2642af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 2643af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (index >= module_count_) { 2644af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModuleList index out of range: " << 2645af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai index << "/" << module_count_; 2646db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return NULL; 2647af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2648db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2649db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return &(*modules_)[index]; 2650db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 2651db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2652db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2653db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovaiconst CodeModules* MinidumpModuleList::Copy() const { 2654db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai return new BasicCodeModules(this); 2655db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai} 2656db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 2657db3342a10ec30902aa9018b80e1d9a40bd01c487mmentovai 26583261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpModuleList::Print() { 2659af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2660af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpModuleList cannot print invalid data"; 26613261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 2662af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 26633261e8b6eac44a41341f112821482bee6c940c98mmentovai 26643261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MinidumpModuleList\n"); 26653261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" module_count = %d\n", module_count_); 26663261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 26673261e8b6eac44a41341f112821482bee6c940c98mmentovai 26683261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int module_index = 0; 26693261e8b6eac44a41341f112821482bee6c940c98mmentovai module_index < module_count_; 26703261e8b6eac44a41341f112821482bee6c940c98mmentovai ++module_index) { 26713261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("module[%d]\n", module_index); 26723261e8b6eac44a41341f112821482bee6c940c98mmentovai 26733261e8b6eac44a41341f112821482bee6c940c98mmentovai (*modules_)[module_index].Print(); 26743261e8b6eac44a41341f112821482bee6c940c98mmentovai } 26753261e8b6eac44a41341f112821482bee6c940c98mmentovai} 26763261e8b6eac44a41341f112821482bee6c940c98mmentovai 26773261e8b6eac44a41341f112821482bee6c940c98mmentovai 26783261e8b6eac44a41341f112821482bee6c940c98mmentovai// 26793261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpMemoryList 26803261e8b6eac44a41341f112821482bee6c940c98mmentovai// 26813261e8b6eac44a41341f112821482bee6c940c98mmentovai 26823261e8b6eac44a41341f112821482bee6c940c98mmentovai 26836162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint32_t MinidumpMemoryList::max_regions_ = 4096; 2684e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 2685e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 26863261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMemoryList::MinidumpMemoryList(Minidump* minidump) 268753d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpStream(minidump), 26886162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com range_map_(new RangeMap<uint64_t, unsigned int>()), 268953d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai descriptors_(NULL), 269053d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai regions_(NULL), 269153d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai region_count_(0) { 26923261e8b6eac44a41341f112821482bee6c940c98mmentovai} 26933261e8b6eac44a41341f112821482bee6c940c98mmentovai 26943261e8b6eac44a41341f112821482bee6c940c98mmentovai 26953261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMemoryList::~MinidumpMemoryList() { 2696fe82bf24a93d9d3affd614aaa23f2f018a733d8emmentovai delete range_map_; 26973261e8b6eac44a41341f112821482bee6c940c98mmentovai delete descriptors_; 26983261e8b6eac44a41341f112821482bee6c940c98mmentovai delete regions_; 26993261e8b6eac44a41341f112821482bee6c940c98mmentovai} 27003261e8b6eac44a41341f112821482bee6c940c98mmentovai 27013261e8b6eac44a41341f112821482bee6c940c98mmentovai 27026162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpMemoryList::Read(uint32_t expected_size) { 27033261e8b6eac44a41341f112821482bee6c940c98mmentovai // Invalidate cached data. 27043261e8b6eac44a41341f112821482bee6c940c98mmentovai delete descriptors_; 27053261e8b6eac44a41341f112821482bee6c940c98mmentovai descriptors_ = NULL; 27063261e8b6eac44a41341f112821482bee6c940c98mmentovai delete regions_; 27073261e8b6eac44a41341f112821482bee6c940c98mmentovai regions_ = NULL; 2708fe82bf24a93d9d3affd614aaa23f2f018a733d8emmentovai range_map_->Clear(); 27093261e8b6eac44a41341f112821482bee6c940c98mmentovai region_count_ = 0; 27103261e8b6eac44a41341f112821482bee6c940c98mmentovai 27113261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 27123261e8b6eac44a41341f112821482bee6c940c98mmentovai 27136162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t region_count; 2714af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (expected_size < sizeof(region_count)) { 2715af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryList count size mismatch, " << 2716af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai expected_size << " < " << sizeof(region_count); 27173261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 2718af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2719af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(®ion_count, sizeof(region_count))) { 2720af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryList could not read memory region count"; 27213261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 2722af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 27233261e8b6eac44a41341f112821482bee6c940c98mmentovai 27243261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) 27253261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(®ion_count); 27263261e8b6eac44a41341f112821482bee6c940c98mmentovai 2727fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai if (region_count > 27286162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com numeric_limits<uint32_t>::max() / sizeof(MDMemoryDescriptor)) { 2729fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai BPLOG(ERROR) << "MinidumpMemoryList region count " << region_count << 2730fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai " would cause multiplication overflow"; 2731fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai return false; 2732fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai } 2733fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai 27343261e8b6eac44a41341f112821482bee6c940c98mmentovai if (expected_size != sizeof(region_count) + 27353261e8b6eac44a41341f112821482bee6c940c98mmentovai region_count * sizeof(MDMemoryDescriptor)) { 2736ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai // may be padded with 4 bytes on 64bit ABIs for alignment 2737ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai if (expected_size == sizeof(region_count) + 4 + 2738ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai region_count * sizeof(MDMemoryDescriptor)) { 27396162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t useless; 2740ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai if (!minidump_->ReadBytes(&useless, 4)) { 2741f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "MinidumpMemoryList cannot read memorylist padded " 2742f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com "bytes"; 2743ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai return false; 2744ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 2745ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } else { 2746ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai BPLOG(ERROR) << "MinidumpMemoryList size mismatch, " << expected_size << 2747f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com " != " << sizeof(region_count) + 2748ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai region_count * sizeof(MDMemoryDescriptor); 2749ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai return false; 2750ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai } 27513261e8b6eac44a41341f112821482bee6c940c98mmentovai } 27523261e8b6eac44a41341f112821482bee6c940c98mmentovai 2753e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (region_count > max_regions_) { 2754e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "MinidumpMemoryList count " << region_count << 2755e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai " exceeds maximum " << max_regions_; 2756e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai return false; 2757e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai } 2758e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 2759e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (region_count != 0) { 2760373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai scoped_ptr<MemoryDescriptors> descriptors( 2761373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai new MemoryDescriptors(region_count)); 27623261e8b6eac44a41341f112821482bee6c940c98mmentovai 2763373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Read the entire array in one fell swoop, instead of reading one entry 2764373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // at a time in the loop. 2765373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai if (!minidump_->ReadBytes(&(*descriptors)[0], 2766373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai sizeof(MDMemoryDescriptor) * region_count)) { 2767af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryList could not read memory region list"; 2768373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 2769373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 27703261e8b6eac44a41341f112821482bee6c940c98mmentovai 2771373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai scoped_ptr<MemoryRegions> regions( 2772373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai new MemoryRegions(region_count, MinidumpMemoryRegion(minidump_))); 27733261e8b6eac44a41341f112821482bee6c940c98mmentovai 2774373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai for (unsigned int region_index = 0; 2775373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai region_index < region_count; 2776373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai ++region_index) { 2777373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai MDMemoryDescriptor* descriptor = &(*descriptors)[region_index]; 27783261e8b6eac44a41341f112821482bee6c940c98mmentovai 2779373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai if (minidump_->swap()) 2780373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai Swap(descriptor); 27813261e8b6eac44a41341f112821482bee6c940c98mmentovai 27826162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t base_address = descriptor->start_of_memory_range; 27836162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t region_size = descriptor->memory.data_size; 27843261e8b6eac44a41341f112821482bee6c940c98mmentovai 2785fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai // Check for base + size overflow or undersize. 2786fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai if (region_size == 0 || 27876162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com region_size > numeric_limits<uint64_t>::max() - base_address) { 2788af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryList has a memory region problem, " << 2789af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai " region " << region_index << "/" << region_count << 2790af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai ", " << HexString(base_address) << "+" << 2791fabb8714a75b9e857bd216ac49b55a15656d7b8emmentovai HexString(region_size); 2792373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 2793af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 27943261e8b6eac44a41341f112821482bee6c940c98mmentovai 2795af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!range_map_->StoreRange(base_address, region_size, region_index)) { 2796af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryList could not store memory region " << 2797af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai region_index << "/" << region_count << ", " << 2798af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(base_address) << "+" << 2799af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(region_size); 2800373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 2801af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2802373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai 2803373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai (*regions)[region_index].SetDescriptor(descriptor); 2804373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 28053261e8b6eac44a41341f112821482bee6c940c98mmentovai 2806373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai descriptors_ = descriptors.release(); 2807373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai regions_ = regions.release(); 28083261e8b6eac44a41341f112821482bee6c940c98mmentovai } 28093261e8b6eac44a41341f112821482bee6c940c98mmentovai 28103261e8b6eac44a41341f112821482bee6c940c98mmentovai region_count_ = region_count; 28113261e8b6eac44a41341f112821482bee6c940c98mmentovai 28123261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 28133261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 28143261e8b6eac44a41341f112821482bee6c940c98mmentovai} 28153261e8b6eac44a41341f112821482bee6c940c98mmentovai 28163261e8b6eac44a41341f112821482bee6c940c98mmentovai 28173261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMemoryRegion* MinidumpMemoryList::GetMemoryRegionAtIndex( 28183261e8b6eac44a41341f112821482bee6c940c98mmentovai unsigned int index) { 2819af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2820af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpMemoryList for GetMemoryRegionAtIndex"; 2821af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return NULL; 2822af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 2823af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 2824af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (index >= region_count_) { 2825af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryList index out of range: " << 2826af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai index << "/" << region_count_; 28273261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2828af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 28293261e8b6eac44a41341f112821482bee6c940c98mmentovai 28303261e8b6eac44a41341f112821482bee6c940c98mmentovai return &(*regions_)[index]; 28313261e8b6eac44a41341f112821482bee6c940c98mmentovai} 28323261e8b6eac44a41341f112821482bee6c940c98mmentovai 28333261e8b6eac44a41341f112821482bee6c940c98mmentovai 28343261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMemoryRegion* MinidumpMemoryList::GetMemoryRegionForAddress( 28356162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t address) { 2836af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2837af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpMemoryList for GetMemoryRegionForAddress"; 28383261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2839af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 28403261e8b6eac44a41341f112821482bee6c940c98mmentovai 28413261e8b6eac44a41341f112821482bee6c940c98mmentovai unsigned int region_index; 2842af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!range_map_->RetrieveRange(address, ®ion_index, NULL, NULL)) { 2843af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "MinidumpMemoryList has no memory region at " << 2844af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(address); 28453261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2846af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 28473261e8b6eac44a41341f112821482bee6c940c98mmentovai 28483261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetMemoryRegionAtIndex(region_index); 28493261e8b6eac44a41341f112821482bee6c940c98mmentovai} 28503261e8b6eac44a41341f112821482bee6c940c98mmentovai 28513261e8b6eac44a41341f112821482bee6c940c98mmentovai 28523261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpMemoryList::Print() { 2853af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2854af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMemoryList cannot print invalid data"; 28553261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 2856af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 28573261e8b6eac44a41341f112821482bee6c940c98mmentovai 28583261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MinidumpMemoryList\n"); 28593261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" region_count = %d\n", region_count_); 28603261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 28613261e8b6eac44a41341f112821482bee6c940c98mmentovai 28623261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int region_index = 0; 28633261e8b6eac44a41341f112821482bee6c940c98mmentovai region_index < region_count_; 28643261e8b6eac44a41341f112821482bee6c940c98mmentovai ++region_index) { 28653261e8b6eac44a41341f112821482bee6c940c98mmentovai MDMemoryDescriptor* descriptor = &(*descriptors_)[region_index]; 28663261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("region[%d]\n", region_index); 28673261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MDMemoryDescriptor\n"); 2868c27cf3e3959189f78fe2de40405987c3f33488cemmentovai printf(" start_of_memory_range = 0x%" PRIx64 "\n", 28693261e8b6eac44a41341f112821482bee6c940c98mmentovai descriptor->start_of_memory_range); 28703261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" memory.data_size = 0x%x\n", descriptor->memory.data_size); 28713261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" memory.rva = 0x%x\n", descriptor->memory.rva); 28723261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpMemoryRegion* region = GetMemoryRegionAtIndex(region_index); 28733261e8b6eac44a41341f112821482bee6c940c98mmentovai if (region) { 28743261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("Memory\n"); 28753261e8b6eac44a41341f112821482bee6c940c98mmentovai region->Print(); 28763261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 28773261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("No memory\n"); 28783261e8b6eac44a41341f112821482bee6c940c98mmentovai } 28793261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 28803261e8b6eac44a41341f112821482bee6c940c98mmentovai } 28813261e8b6eac44a41341f112821482bee6c940c98mmentovai} 28823261e8b6eac44a41341f112821482bee6c940c98mmentovai 28833261e8b6eac44a41341f112821482bee6c940c98mmentovai 28843261e8b6eac44a41341f112821482bee6c940c98mmentovai// 28853261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpException 28863261e8b6eac44a41341f112821482bee6c940c98mmentovai// 28873261e8b6eac44a41341f112821482bee6c940c98mmentovai 28883261e8b6eac44a41341f112821482bee6c940c98mmentovai 28893261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpException::MinidumpException(Minidump* minidump) 289053d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpStream(minidump), 289153d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai exception_(), 289253d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai context_(NULL) { 28933261e8b6eac44a41341f112821482bee6c940c98mmentovai} 28943261e8b6eac44a41341f112821482bee6c940c98mmentovai 28953261e8b6eac44a41341f112821482bee6c940c98mmentovai 28963261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpException::~MinidumpException() { 28973261e8b6eac44a41341f112821482bee6c940c98mmentovai delete context_; 28983261e8b6eac44a41341f112821482bee6c940c98mmentovai} 28993261e8b6eac44a41341f112821482bee6c940c98mmentovai 29003261e8b6eac44a41341f112821482bee6c940c98mmentovai 29016162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpException::Read(uint32_t expected_size) { 29023261e8b6eac44a41341f112821482bee6c940c98mmentovai // Invalidate cached data. 29033261e8b6eac44a41341f112821482bee6c940c98mmentovai delete context_; 29043261e8b6eac44a41341f112821482bee6c940c98mmentovai context_ = NULL; 29053261e8b6eac44a41341f112821482bee6c940c98mmentovai 29063261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 29073261e8b6eac44a41341f112821482bee6c940c98mmentovai 2908af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (expected_size != sizeof(exception_)) { 2909af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpException size mismatch, " << expected_size << 2910af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai " != " << sizeof(exception_); 29113261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 2912af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 29133261e8b6eac44a41341f112821482bee6c940c98mmentovai 2914af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&exception_, sizeof(exception_))) { 2915af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpException cannot read exception"; 29163261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 2917af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 29183261e8b6eac44a41341f112821482bee6c940c98mmentovai 29193261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) { 29203261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&exception_.thread_id); 29213261e8b6eac44a41341f112821482bee6c940c98mmentovai // exception_.__align is for alignment only and does not need to be 29223261e8b6eac44a41341f112821482bee6c940c98mmentovai // swapped. 29233261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&exception_.exception_record.exception_code); 29243261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&exception_.exception_record.exception_flags); 29253261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&exception_.exception_record.exception_record); 29263261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&exception_.exception_record.exception_address); 29273261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&exception_.exception_record.number_parameters); 29283261e8b6eac44a41341f112821482bee6c940c98mmentovai // exception_.exception_record.__align is for alignment only and does not 29293261e8b6eac44a41341f112821482bee6c940c98mmentovai // need to be swapped. 29303261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int parameter_index = 0; 29313261e8b6eac44a41341f112821482bee6c940c98mmentovai parameter_index < MD_EXCEPTION_MAXIMUM_PARAMETERS; 29323261e8b6eac44a41341f112821482bee6c940c98mmentovai ++parameter_index) { 29333261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&exception_.exception_record.exception_information[parameter_index]); 29343261e8b6eac44a41341f112821482bee6c940c98mmentovai } 29353261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&exception_.thread_context); 29363261e8b6eac44a41341f112821482bee6c940c98mmentovai } 29373261e8b6eac44a41341f112821482bee6c940c98mmentovai 29383261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 29393261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 29403261e8b6eac44a41341f112821482bee6c940c98mmentovai} 29413261e8b6eac44a41341f112821482bee6c940c98mmentovai 29423261e8b6eac44a41341f112821482bee6c940c98mmentovai 29436162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpException::GetThreadID(uint32_t *thread_id) const { 2944af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !thread_id) << "MinidumpException::GetThreadID requires " 2945af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "|thread_id|"; 2946af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai assert(thread_id); 2947af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai *thread_id = 0; 2948af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 2949af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2950af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpException for GetThreadID"; 295176f052f8fbf8864dee5992b857229d06560a766ammentovai return false; 2952af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 295376f052f8fbf8864dee5992b857229d06560a766ammentovai 295476f052f8fbf8864dee5992b857229d06560a766ammentovai *thread_id = exception_.thread_id; 295576f052f8fbf8864dee5992b857229d06560a766ammentovai return true; 29563261e8b6eac44a41341f112821482bee6c940c98mmentovai} 29573261e8b6eac44a41341f112821482bee6c940c98mmentovai 29583261e8b6eac44a41341f112821482bee6c940c98mmentovai 29593261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpContext* MinidumpException::GetContext() { 2960af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2961af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpException for GetContext"; 29623261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2963af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 29643261e8b6eac44a41341f112821482bee6c940c98mmentovai 29653261e8b6eac44a41341f112821482bee6c940c98mmentovai if (!context_) { 2966af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->SeekSet(exception_.thread_context.rva)) { 2967af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpException cannot seek to context"; 29683261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2969af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 29703261e8b6eac44a41341f112821482bee6c940c98mmentovai 29712466d8e993a800a17e00deda2f3a27e0505140e1mmentovai scoped_ptr<MinidumpContext> context(new MinidumpContext(minidump_)); 29723261e8b6eac44a41341f112821482bee6c940c98mmentovai 29739276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek // Don't log as an error if we can still fall back on the thread's context 29749276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek // (which must be possible if we got this far.) 2975af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!context->Read(exception_.thread_context.data_size)) { 29765f4fa55598baf92785223debe7d3787c6b29cf3ejschuh@chromium.org BPLOG(INFO) << "MinidumpException cannot read context"; 29773261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 2978af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 29793261e8b6eac44a41341f112821482bee6c940c98mmentovai 29803261e8b6eac44a41341f112821482bee6c940c98mmentovai context_ = context.release(); 29813261e8b6eac44a41341f112821482bee6c940c98mmentovai } 29823261e8b6eac44a41341f112821482bee6c940c98mmentovai 29833261e8b6eac44a41341f112821482bee6c940c98mmentovai return context_; 29843261e8b6eac44a41341f112821482bee6c940c98mmentovai} 29853261e8b6eac44a41341f112821482bee6c940c98mmentovai 29863261e8b6eac44a41341f112821482bee6c940c98mmentovai 29873261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpException::Print() { 2988af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 2989af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpException cannot print invalid data"; 29903261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 2991af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 29923261e8b6eac44a41341f112821482bee6c940c98mmentovai 29933261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MDException\n"); 29943261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" thread_id = 0x%x\n", 29953261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.thread_id); 29963261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" exception_record.exception_code = 0x%x\n", 29973261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.exception_record.exception_code); 29983261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" exception_record.exception_flags = 0x%x\n", 29993261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.exception_record.exception_flags); 3000c27cf3e3959189f78fe2de40405987c3f33488cemmentovai printf(" exception_record.exception_record = 0x%" PRIx64 "\n", 30013261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.exception_record.exception_record); 3002c27cf3e3959189f78fe2de40405987c3f33488cemmentovai printf(" exception_record.exception_address = 0x%" PRIx64 "\n", 30033261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.exception_record.exception_address); 30043261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" exception_record.number_parameters = %d\n", 30053261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.exception_record.number_parameters); 30063261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int parameterIndex = 0; 30073261e8b6eac44a41341f112821482bee6c940c98mmentovai parameterIndex < exception_.exception_record.number_parameters; 30083261e8b6eac44a41341f112821482bee6c940c98mmentovai ++parameterIndex) { 3009c27cf3e3959189f78fe2de40405987c3f33488cemmentovai printf(" exception_record.exception_information[%2d] = 0x%" PRIx64 "\n", 30103261e8b6eac44a41341f112821482bee6c940c98mmentovai parameterIndex, 30113261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.exception_record.exception_information[parameterIndex]); 30123261e8b6eac44a41341f112821482bee6c940c98mmentovai } 30133261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" thread_context.data_size = %d\n", 30143261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.thread_context.data_size); 30153261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" thread_context.rva = 0x%x\n", 30163261e8b6eac44a41341f112821482bee6c940c98mmentovai exception_.thread_context.rva); 30173261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpContext* context = GetContext(); 30183261e8b6eac44a41341f112821482bee6c940c98mmentovai if (context) { 30193261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 30203261e8b6eac44a41341f112821482bee6c940c98mmentovai context->Print(); 30213261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 30223261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (no context)\n"); 30233261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 30243261e8b6eac44a41341f112821482bee6c940c98mmentovai } 30253261e8b6eac44a41341f112821482bee6c940c98mmentovai} 30263261e8b6eac44a41341f112821482bee6c940c98mmentovai 30270314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek// 30280314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek// MinidumpAssertion 30290314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek// 30300314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30310314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30320314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarekMinidumpAssertion::MinidumpAssertion(Minidump* minidump) 30330314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek : MinidumpStream(minidump), 30340314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek assertion_(), 30350314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek expression_(), 30360314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek function_(), 30370314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek file_() { 30380314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek} 30390314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30400314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30410314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarekMinidumpAssertion::~MinidumpAssertion() { 30420314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek} 30430314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30440314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30456162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpAssertion::Read(uint32_t expected_size) { 30460314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek // Invalidate cached data. 30470314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek valid_ = false; 30480314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30490314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek if (expected_size != sizeof(assertion_)) { 30500314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek BPLOG(ERROR) << "MinidumpAssertion size mismatch, " << expected_size << 30510314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek " != " << sizeof(assertion_); 30520314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek return false; 30530314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek } 30540314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30550314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek if (!minidump_->ReadBytes(&assertion_, sizeof(assertion_))) { 30560314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek BPLOG(ERROR) << "MinidumpAssertion cannot read assertion"; 30570314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek return false; 30580314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek } 30590314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30600314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek // Each of {expression, function, file} is a UTF-16 string, 30610314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek // we'll convert them to UTF-8 for ease of use. 30623562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com ConvertUTF16BufferToUTF8String(assertion_.expression, 30633562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com sizeof(assertion_.expression), &expression_, 30643562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com minidump_->swap()); 30653562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com ConvertUTF16BufferToUTF8String(assertion_.function, 30663562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com sizeof(assertion_.function), &function_, 30673562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com minidump_->swap()); 30683562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com ConvertUTF16BufferToUTF8String(assertion_.file, sizeof(assertion_.file), 30693562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com &file_, minidump_->swap()); 30700314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30710314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek if (minidump_->swap()) { 30720314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek Swap(&assertion_.line); 30730314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek Swap(&assertion_.type); 30740314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek } 30750314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30760314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek valid_ = true; 30770314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek return true; 30780314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek} 30790314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30800314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarekvoid MinidumpAssertion::Print() { 30810314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek if (!valid_) { 30820314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek BPLOG(ERROR) << "MinidumpAssertion cannot print invalid data"; 30830314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek return; 30840314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek } 30850314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 30860314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek printf("MDAssertion\n"); 30870314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek printf(" expression = %s\n", 30880314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek expression_.c_str()); 30890314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek printf(" function = %s\n", 30900314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek function_.c_str()); 30910314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek printf(" file = %s\n", 30920314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek file_.c_str()); 30930314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek printf(" line = %u\n", 30940314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek assertion_.line); 30950314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek printf(" type = %u\n", 30960314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek assertion_.type); 30970314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek printf("\n"); 30980314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek} 30993261e8b6eac44a41341f112821482bee6c940c98mmentovai 31003261e8b6eac44a41341f112821482bee6c940c98mmentovai// 31013261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpSystemInfo 31023261e8b6eac44a41341f112821482bee6c940c98mmentovai// 31033261e8b6eac44a41341f112821482bee6c940c98mmentovai 31043261e8b6eac44a41341f112821482bee6c940c98mmentovai 31053261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpSystemInfo::MinidumpSystemInfo(Minidump* minidump) 310653d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpStream(minidump), 310753d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai system_info_(), 3108e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai csd_version_(NULL), 3109e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai cpu_vendor_(NULL) { 31103261e8b6eac44a41341f112821482bee6c940c98mmentovai} 31113261e8b6eac44a41341f112821482bee6c940c98mmentovai 31123261e8b6eac44a41341f112821482bee6c940c98mmentovai 31133261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpSystemInfo::~MinidumpSystemInfo() { 31143261e8b6eac44a41341f112821482bee6c940c98mmentovai delete csd_version_; 3115e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai delete cpu_vendor_; 31163261e8b6eac44a41341f112821482bee6c940c98mmentovai} 31173261e8b6eac44a41341f112821482bee6c940c98mmentovai 31183261e8b6eac44a41341f112821482bee6c940c98mmentovai 31196162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpSystemInfo::Read(uint32_t expected_size) { 31203261e8b6eac44a41341f112821482bee6c940c98mmentovai // Invalidate cached data. 31213261e8b6eac44a41341f112821482bee6c940c98mmentovai delete csd_version_; 31223261e8b6eac44a41341f112821482bee6c940c98mmentovai csd_version_ = NULL; 3123e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai delete cpu_vendor_; 3124e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai cpu_vendor_ = NULL; 31253261e8b6eac44a41341f112821482bee6c940c98mmentovai 31263261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 31273261e8b6eac44a41341f112821482bee6c940c98mmentovai 3128af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (expected_size != sizeof(system_info_)) { 3129af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpSystemInfo size mismatch, " << expected_size << 3130af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai " != " << sizeof(system_info_); 31313261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 3132af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 31333261e8b6eac44a41341f112821482bee6c940c98mmentovai 3134af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&system_info_, sizeof(system_info_))) { 3135af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpSystemInfo cannot read system info"; 31363261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 3137af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 31383261e8b6eac44a41341f112821482bee6c940c98mmentovai 31393261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) { 31403261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.processor_architecture); 31413261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.processor_level); 31423261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.processor_revision); 31433261e8b6eac44a41341f112821482bee6c940c98mmentovai // number_of_processors and product_type are 8-bit quantities and need no 31443261e8b6eac44a41341f112821482bee6c940c98mmentovai // swapping. 31453261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.major_version); 31463261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.minor_version); 31473261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.build_number); 31483261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.platform_id); 31493261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.csd_version_rva); 31503261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&system_info_.suite_mask); 31513402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai // Don't swap the reserved2 field because its contents are unknown. 31523402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai 31533402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai if (system_info_.processor_architecture == MD_CPU_ARCHITECTURE_X86 || 31543402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai system_info_.processor_architecture == MD_CPU_ARCHITECTURE_X86_WIN64) { 31553402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai for (unsigned int i = 0; i < 3; ++i) 31563402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai Swap(&system_info_.cpu.x86_cpu_info.vendor_id[i]); 31573402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai Swap(&system_info_.cpu.x86_cpu_info.version_information); 31583402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai Swap(&system_info_.cpu.x86_cpu_info.feature_information); 31593402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai Swap(&system_info_.cpu.x86_cpu_info.amd_extended_cpu_features); 31603402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } else { 31613402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai for (unsigned int i = 0; i < 2; ++i) 31623402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai Swap(&system_info_.cpu.other_cpu_info.processor_features[i]); 31633402cae5e58f7503adc4d9de6d9ea69e725ddcb2mmentovai } 31643261e8b6eac44a41341f112821482bee6c940c98mmentovai } 31653261e8b6eac44a41341f112821482bee6c940c98mmentovai 31663261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 31673261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 31683261e8b6eac44a41341f112821482bee6c940c98mmentovai} 31693261e8b6eac44a41341f112821482bee6c940c98mmentovai 31703261e8b6eac44a41341f112821482bee6c940c98mmentovai 317197d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovaistring MinidumpSystemInfo::GetOS() { 31724e518a4357a2d1c379d4a91df6d4e153ee791101ivan.penkov@gmail.com string os; 31734e518a4357a2d1c379d4a91df6d4e153ee791101ivan.penkov@gmail.com 3174af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 3175af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpSystemInfo for GetOS"; 31764e518a4357a2d1c379d4a91df6d4e153ee791101ivan.penkov@gmail.com return os; 3177af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 317897d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 317997d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai switch (system_info_.platform_id) { 318097d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai case MD_OS_WIN32_NT: 318197d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai case MD_OS_WIN32_WINDOWS: 318297d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai os = "windows"; 318397d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai break; 318497d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 318597d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai case MD_OS_MAC_OS_X: 318697d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai os = "mac"; 318797d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai break; 318897d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 318963f97ad134db3fa175c489b50b66a79bf0c95025qsr@chromium.org case MD_OS_IOS: 319063f97ad134db3fa175c489b50b66a79bf0c95025qsr@chromium.org os = "ios"; 319163f97ad134db3fa175c489b50b66a79bf0c95025qsr@chromium.org break; 319263f97ad134db3fa175c489b50b66a79bf0c95025qsr@chromium.org 319397d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai case MD_OS_LINUX: 319497d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai os = "linux"; 319597d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai break; 3196af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 3197ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai case MD_OS_SOLARIS: 3198ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai os = "solaris"; 31995187de1ae5cb7df2343b650416aa805084bdf8dadigit@chromium.org break; 32005187de1ae5cb7df2343b650416aa805084bdf8dadigit@chromium.org 32015187de1ae5cb7df2343b650416aa805084bdf8dadigit@chromium.org case MD_OS_ANDROID: 32025187de1ae5cb7df2343b650416aa805084bdf8dadigit@chromium.org os = "android"; 3203ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai break; 3204ea2bba970675e01f9964f82d3f44960c1aad05dcmmentovai 3205d9b40724ed6a6c243eaef196e617071ec36d4282thestig@chromium.org case MD_OS_PS3: 3206d9b40724ed6a6c243eaef196e617071ec36d4282thestig@chromium.org os = "ps3"; 3207d9b40724ed6a6c243eaef196e617071ec36d4282thestig@chromium.org break; 3208d9b40724ed6a6c243eaef196e617071ec36d4282thestig@chromium.org 320942faddc9c3be48a8a29236ab866013288d25491emseaborn@chromium.org case MD_OS_NACL: 321042faddc9c3be48a8a29236ab866013288d25491emseaborn@chromium.org os = "nacl"; 321142faddc9c3be48a8a29236ab866013288d25491emseaborn@chromium.org break; 321242faddc9c3be48a8a29236ab866013288d25491emseaborn@chromium.org 3213af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai default: 3214af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpSystemInfo unknown OS for platform " << 3215af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(system_info_.platform_id); 3216af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai break; 321797d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai } 321897d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 321997d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai return os; 322097d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai} 322197d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 322297d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 322397d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovaistring MinidumpSystemInfo::GetCPU() { 3224af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 3225af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpSystemInfo for GetCPU"; 322697d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai return ""; 3227af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 322897d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 322997d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai string cpu; 323097d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 323197d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai switch (system_info_.processor_architecture) { 323297d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai case MD_CPU_ARCHITECTURE_X86: 323397d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai case MD_CPU_ARCHITECTURE_X86_WIN64: 323497d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai cpu = "x86"; 323597d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai break; 323697d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 32379276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek case MD_CPU_ARCHITECTURE_AMD64: 32389276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek cpu = "x86-64"; 32399276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek break; 32409276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 324197d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai case MD_CPU_ARCHITECTURE_PPC: 324297d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai cpu = "ppc"; 324397d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai break; 3244af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 3245cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org case MD_CPU_ARCHITECTURE_PPC64: 3246cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org cpu = "ppc64"; 3247cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org break; 3248cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org 3249dd2ff4a21c57672170eb14ccc5142efd7d92f3f1ted.mielczarek case MD_CPU_ARCHITECTURE_SPARC: 3250dd2ff4a21c57672170eb14ccc5142efd7d92f3f1ted.mielczarek cpu = "sparc"; 3251dd2ff4a21c57672170eb14ccc5142efd7d92f3f1ted.mielczarek break; 3252dd2ff4a21c57672170eb14ccc5142efd7d92f3f1ted.mielczarek 32539276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek case MD_CPU_ARCHITECTURE_ARM: 32549276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek cpu = "arm"; 32559276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek break; 32569276b0d3017ad5ca93c8b593cacf317e1eaa114eted.mielczarek 325739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org case MD_CPU_ARCHITECTURE_ARM64: 325839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org cpu = "arm64"; 325939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org break; 326039d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org 3261af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai default: 3262af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpSystemInfo unknown CPU for architecture " << 3263af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(system_info_.processor_architecture); 3264af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai break; 326597d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai } 326697d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 326797d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai return cpu; 326897d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai} 326997d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 327097d392dc4b60f0099cd7ad8c8a5f06581a532392mmentovai 32713261e8b6eac44a41341f112821482bee6c940c98mmentovaiconst string* MinidumpSystemInfo::GetCSDVersion() { 3272af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 3273af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpSystemInfo for GetCSDVersion"; 32743261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 3275af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 32763261e8b6eac44a41341f112821482bee6c940c98mmentovai 32773261e8b6eac44a41341f112821482bee6c940c98mmentovai if (!csd_version_) 32783261e8b6eac44a41341f112821482bee6c940c98mmentovai csd_version_ = minidump_->ReadString(system_info_.csd_version_rva); 32793261e8b6eac44a41341f112821482bee6c940c98mmentovai 3280af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !csd_version_) << "MinidumpSystemInfo could not read " 3281af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "CSD version"; 3282af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 32833261e8b6eac44a41341f112821482bee6c940c98mmentovai return csd_version_; 32843261e8b6eac44a41341f112821482bee6c940c98mmentovai} 32853261e8b6eac44a41341f112821482bee6c940c98mmentovai 32863261e8b6eac44a41341f112821482bee6c940c98mmentovai 3287e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovaiconst string* MinidumpSystemInfo::GetCPUVendor() { 3288af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 3289af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpSystemInfo for GetCPUVendor"; 3290e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai return NULL; 3291af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 3292e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai 3293e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai // CPU vendor information can only be determined from x86 minidumps. 3294e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai if (!cpu_vendor_ && 3295e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.processor_architecture == MD_CPU_ARCHITECTURE_X86 || 3296e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai system_info_.processor_architecture == MD_CPU_ARCHITECTURE_X86_WIN64)) { 3297e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai char cpu_vendor_string[13]; 3298e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai snprintf(cpu_vendor_string, sizeof(cpu_vendor_string), 3299e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai "%c%c%c%c%c%c%c%c%c%c%c%c", 3300e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai system_info_.cpu.x86_cpu_info.vendor_id[0] & 0xff, 3301e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[0] >> 8) & 0xff, 3302e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[0] >> 16) & 0xff, 3303e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[0] >> 24) & 0xff, 3304e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai system_info_.cpu.x86_cpu_info.vendor_id[1] & 0xff, 3305e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[1] >> 8) & 0xff, 3306e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[1] >> 16) & 0xff, 3307e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[1] >> 24) & 0xff, 3308e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai system_info_.cpu.x86_cpu_info.vendor_id[2] & 0xff, 3309e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[2] >> 8) & 0xff, 3310e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[2] >> 16) & 0xff, 3311e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai (system_info_.cpu.x86_cpu_info.vendor_id[2] >> 24) & 0xff); 3312e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai cpu_vendor_ = new string(cpu_vendor_string); 3313e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai } 3314e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai 3315e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai return cpu_vendor_; 3316e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai} 3317e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai 3318e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai 33193261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpSystemInfo::Print() { 3320af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 3321af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpSystemInfo cannot print invalid data"; 33223261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 3323af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 33243261e8b6eac44a41341f112821482bee6c940c98mmentovai 33253261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MDRawSystemInfo\n"); 33261b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" processor_architecture = 0x%x\n", 33273261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.processor_architecture); 33283261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" processor_level = %d\n", 33293261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.processor_level); 33300a7e6bf16cad354710df60929c2ac82f647cb54emmentovai printf(" processor_revision = 0x%x\n", 33310a7e6bf16cad354710df60929c2ac82f647cb54emmentovai system_info_.processor_revision); 33323261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" number_of_processors = %d\n", 33333261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.number_of_processors); 33343261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" product_type = %d\n", 33353261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.product_type); 33363261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" major_version = %d\n", 33373261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.major_version); 33383261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" minor_version = %d\n", 33393261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.minor_version); 33403261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" build_number = %d\n", 33413261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.build_number); 33421b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" platform_id = 0x%x\n", 33433261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.platform_id); 33443261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" csd_version_rva = 0x%x\n", 33453261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.csd_version_rva); 33463261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" suite_mask = 0x%x\n", 33473261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.suite_mask); 33489c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org if (system_info_.processor_architecture == MD_CPU_ARCHITECTURE_X86 || 33499c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org system_info_.processor_architecture == MD_CPU_ARCHITECTURE_X86_WIN64) { 33509c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org printf(" cpu.x86_cpu_info (valid):\n"); 33519c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org } else { 33529c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org printf(" cpu.x86_cpu_info (invalid):\n"); 33539c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org } 33543261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int i = 0; i < 3; ++i) { 33553261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" cpu.x86_cpu_info.vendor_id[%d] = 0x%x\n", 33563261e8b6eac44a41341f112821482bee6c940c98mmentovai i, system_info_.cpu.x86_cpu_info.vendor_id[i]); 33573261e8b6eac44a41341f112821482bee6c940c98mmentovai } 33583261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" cpu.x86_cpu_info.version_information = 0x%x\n", 33593261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.cpu.x86_cpu_info.version_information); 33603261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" cpu.x86_cpu_info.feature_information = 0x%x\n", 33613261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.cpu.x86_cpu_info.feature_information); 33623261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" cpu.x86_cpu_info.amd_extended_cpu_features = 0x%x\n", 33633261e8b6eac44a41341f112821482bee6c940c98mmentovai system_info_.cpu.x86_cpu_info.amd_extended_cpu_features); 33649c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org if (system_info_.processor_architecture != MD_CPU_ARCHITECTURE_X86 && 33659c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org system_info_.processor_architecture != MD_CPU_ARCHITECTURE_X86_WIN64) { 33669c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org printf(" cpu.other_cpu_info (valid):\n"); 33679c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org for (unsigned int i = 0; i < 2; ++i) { 33689c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org printf(" cpu.other_cpu_info.processor_features[%d] = 0x%" PRIx64 "\n", 33699c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org i, system_info_.cpu.other_cpu_info.processor_features[i]); 33709c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org } 33719c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org } 3372e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai const string* csd_version = GetCSDVersion(); 3373e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai if (csd_version) { 33743261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (csd_version) = \"%s\"\n", 3375e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai csd_version->c_str()); 3376e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai } else { 33773261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" (csd_version) = (null)\n"); 3378e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai } 3379e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai const string* cpu_vendor = GetCPUVendor(); 3380e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai if (cpu_vendor) { 3381e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai printf(" (cpu_vendor) = \"%s\"\n", 3382e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai cpu_vendor->c_str()); 3383e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai } else { 3384e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai printf(" (cpu_vendor) = (null)\n"); 3385e5468b8a49ac6a4e5acb9d4003838e3e323f85e4mmentovai } 33863261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 33873261e8b6eac44a41341f112821482bee6c940c98mmentovai} 33883261e8b6eac44a41341f112821482bee6c940c98mmentovai 33893261e8b6eac44a41341f112821482bee6c940c98mmentovai 33903261e8b6eac44a41341f112821482bee6c940c98mmentovai// 33913261e8b6eac44a41341f112821482bee6c940c98mmentovai// MinidumpMiscInfo 33923261e8b6eac44a41341f112821482bee6c940c98mmentovai// 33933261e8b6eac44a41341f112821482bee6c940c98mmentovai 33943261e8b6eac44a41341f112821482bee6c940c98mmentovai 33953261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMiscInfo::MinidumpMiscInfo(Minidump* minidump) 339653d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : MinidumpStream(minidump), 339753d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai misc_info_() { 33983261e8b6eac44a41341f112821482bee6c940c98mmentovai} 33993261e8b6eac44a41341f112821482bee6c940c98mmentovai 34003261e8b6eac44a41341f112821482bee6c940c98mmentovai 34016162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpMiscInfo::Read(uint32_t expected_size) { 34023261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 34033261e8b6eac44a41341f112821482bee6c940c98mmentovai 34043261e8b6eac44a41341f112821482bee6c940c98mmentovai if (expected_size != MD_MISCINFO_SIZE && 34053562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com expected_size != MD_MISCINFO2_SIZE && 34063562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com expected_size != MD_MISCINFO3_SIZE && 34073562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com expected_size != MD_MISCINFO4_SIZE) { 34083562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com BPLOG(ERROR) << "MinidumpMiscInfo size mismatch, " << expected_size 34093562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com << " != " << MD_MISCINFO_SIZE << ", " << MD_MISCINFO2_SIZE 34103562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com << ", " << MD_MISCINFO3_SIZE << ", " << MD_MISCINFO4_SIZE 34113562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com << ")"; 34123261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 34133261e8b6eac44a41341f112821482bee6c940c98mmentovai } 34143261e8b6eac44a41341f112821482bee6c940c98mmentovai 3415af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&misc_info_, expected_size)) { 3416af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMiscInfo cannot read miscellaneous info"; 34173261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 3418af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 34193261e8b6eac44a41341f112821482bee6c940c98mmentovai 34203261e8b6eac44a41341f112821482bee6c940c98mmentovai if (minidump_->swap()) { 34213562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Swap version 1 fields 34223261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.size_of_info); 34233261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.flags1); 34243261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.process_id); 34253261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.process_create_time); 34263261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.process_user_time); 34273261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.process_kernel_time); 34283261e8b6eac44a41341f112821482bee6c940c98mmentovai if (misc_info_.size_of_info > MD_MISCINFO_SIZE) { 34293562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Swap version 2 fields 34303261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.processor_max_mhz); 34313261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.processor_current_mhz); 34323261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.processor_mhz_limit); 34333261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.processor_max_idle_state); 34343261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&misc_info_.processor_current_idle_state); 34353261e8b6eac44a41341f112821482bee6c940c98mmentovai } 34363562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com if (misc_info_.size_of_info > MD_MISCINFO2_SIZE) { 34373562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Swap version 3 fields 34383562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&misc_info_.process_integrity_level); 34393562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&misc_info_.process_execute_flags); 34403562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&misc_info_.protected_process); 34413562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&misc_info_.time_zone_id); 34423562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com Swap(&misc_info_.time_zone); 34433562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 34443562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com if (misc_info_.size_of_info > MD_MISCINFO3_SIZE) { 34453562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Swap version 4 fields. 34463562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Do not swap UTF-16 strings. The swap is done as part of the 34473562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // conversion to UTF-8 (code follows below). 34483562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 34493261e8b6eac44a41341f112821482bee6c940c98mmentovai } 34503261e8b6eac44a41341f112821482bee6c940c98mmentovai 345165571f17edb82d122b5f6dc741bd7d4b9e315e1bmmentovai if (expected_size != misc_info_.size_of_info) { 3452af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMiscInfo size mismatch, " << 345365571f17edb82d122b5f6dc741bd7d4b9e315e1bmmentovai expected_size << " != " << misc_info_.size_of_info; 34543261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 3455af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 34563261e8b6eac44a41341f112821482bee6c940c98mmentovai 34573562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Convert UTF-16 strings 34583562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com if (misc_info_.size_of_info > MD_MISCINFO2_SIZE) { 34593562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Convert UTF-16 strings in version 3 fields 34603562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com ConvertUTF16BufferToUTF8String(misc_info_.time_zone.standard_name, 34613562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com sizeof(misc_info_.time_zone.standard_name), 34623562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com &standard_name_, minidump_->swap()); 34633562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com ConvertUTF16BufferToUTF8String(misc_info_.time_zone.daylight_name, 34643562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com sizeof(misc_info_.time_zone.daylight_name), 34653562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com &daylight_name_, minidump_->swap()); 34663562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 34673562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com if (misc_info_.size_of_info > MD_MISCINFO3_SIZE) { 34683562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Convert UTF-16 strings in version 4 fields 34693562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com ConvertUTF16BufferToUTF8String(misc_info_.build_string, 34703562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com sizeof(misc_info_.build_string), 34713562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com &build_string_, minidump_->swap()); 34723562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com ConvertUTF16BufferToUTF8String(misc_info_.dbg_bld_str, 34733562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com sizeof(misc_info_.dbg_bld_str), 34743562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com &dbg_bld_str_, minidump_->swap()); 34753562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 34763562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com 34773261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 34783261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 34793261e8b6eac44a41341f112821482bee6c940c98mmentovai} 34803261e8b6eac44a41341f112821482bee6c940c98mmentovai 34813261e8b6eac44a41341f112821482bee6c940c98mmentovai 34823261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid MinidumpMiscInfo::Print() { 3483af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 3484af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpMiscInfo cannot print invalid data"; 34853261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 3486af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 34873261e8b6eac44a41341f112821482bee6c940c98mmentovai 34883261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MDRawMiscInfo\n"); 34893562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Print version 1 fields 34903261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" size_of_info = %d\n", misc_info_.size_of_info); 34913261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" flags1 = 0x%x\n", misc_info_.flags1); 34921b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" process_id = "); 34931b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & MD_MISCINFO_FLAGS1_PROCESS_ID, 34941b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, misc_info_.process_id); 34951b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org if (misc_info_.flags1 & MD_MISCINFO_FLAGS1_PROCESS_TIMES) { 34961b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" process_create_time = 0x%x %s\n", 34971b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.process_create_time, 34989c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org TimeTToUTCString(misc_info_.process_create_time).c_str()); 34991b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } else { 35001b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" process_create_time = (invalid)\n"); 35011b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } 35021b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" process_user_time = "); 35031b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & MD_MISCINFO_FLAGS1_PROCESS_TIMES, 35041b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, misc_info_.process_user_time); 35051b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" process_kernel_time = "); 35061b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & MD_MISCINFO_FLAGS1_PROCESS_TIMES, 35071b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, misc_info_.process_kernel_time); 35083261e8b6eac44a41341f112821482bee6c940c98mmentovai if (misc_info_.size_of_info > MD_MISCINFO_SIZE) { 35093562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Print version 2 fields 35101b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" processor_max_mhz = "); 35111b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & 35121b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_MISCINFO_FLAGS1_PROCESSOR_POWER_INFO, 35131b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, misc_info_.processor_max_mhz); 35141b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" processor_current_mhz = "); 35151b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & 35161b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_MISCINFO_FLAGS1_PROCESSOR_POWER_INFO, 35171b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, misc_info_.processor_current_mhz); 35181b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" processor_mhz_limit = "); 35191b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & 35201b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_MISCINFO_FLAGS1_PROCESSOR_POWER_INFO, 35211b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, misc_info_.processor_mhz_limit); 35221b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" processor_max_idle_state = "); 35231b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & 35241b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_MISCINFO_FLAGS1_PROCESSOR_POWER_INFO, 35251b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, 35261b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.processor_max_idle_state); 35271b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" processor_current_idle_state = "); 35281b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & 35291b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_MISCINFO_FLAGS1_PROCESSOR_POWER_INFO, 35301b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, 35311b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.processor_current_idle_state); 35323261e8b6eac44a41341f112821482bee6c940c98mmentovai } 35333562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com if (misc_info_.size_of_info > MD_MISCINFO2_SIZE) { 35343562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Print version 3 fields 35351b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" process_integrity_level = "); 35361b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & 35371b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_MISCINFO_FLAGS1_PROCESS_INTEGRITY, 35381b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatHexadecimal, 35391b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.process_integrity_level); 35401b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" process_execute_flags = "); 35411b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & 35421b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_MISCINFO_FLAGS1_PROCESS_EXECUTE_FLAGS, 35431b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatHexadecimal, 35441b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.process_execute_flags); 35451b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" protected_process = "); 35461b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & 35471b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_MISCINFO_FLAGS1_PROTECTED_PROCESS, 35481b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, misc_info_.protected_process); 35491b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone_id = "); 35501b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(misc_info_.flags1 & MD_MISCINFO_FLAGS1_TIMEZONE, 35511b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatDecimal, misc_info_.time_zone_id); 35521b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org if (misc_info_.flags1 & MD_MISCINFO_FLAGS1_TIMEZONE) { 35531b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.bias = %d\n", 35541b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.bias); 35551b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.standard_name = %s\n", standard_name_.c_str()); 35561b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.standard_date = " 35571b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org "%04d-%02d-%02d (%d) %02d:%02d:%02d.%03d\n", 35581b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_date.year, 35591b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_date.month, 35601b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_date.day, 35611b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_date.day_of_week, 35621b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_date.hour, 35631b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_date.minute, 35641b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_date.second, 35651b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_date.milliseconds); 35661b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.standard_bias = %d\n", 35671b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.standard_bias); 35681b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.daylight_name = %s\n", daylight_name_.c_str()); 35691b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.daylight_date = " 35701b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org "%04d-%02d-%02d (%d) %02d:%02d:%02d.%03d\n", 35711b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_date.year, 35721b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_date.month, 35731b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_date.day, 35741b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_date.day_of_week, 35751b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_date.hour, 35761b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_date.minute, 35771b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_date.second, 35781b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_date.milliseconds); 35791b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.daylight_bias = %d\n", 35801b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org misc_info_.time_zone.daylight_bias); 35811b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } else { 35821b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.bias = (invalid)\n"); 35831b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.standard_name = (invalid)\n"); 35841b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.standard_date = (invalid)\n"); 35851b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.standard_bias = (invalid)\n"); 35861b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.daylight_name = (invalid)\n"); 35871b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.daylight_date = (invalid)\n"); 35881b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" time_zone.daylight_bias = (invalid)\n"); 35891b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } 35903562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 35913562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com if (misc_info_.size_of_info > MD_MISCINFO3_SIZE) { 35923562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com // Print version 4 fields 35931b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org if (misc_info_.flags1 & MD_MISCINFO_FLAGS1_BUILDSTRING) { 35941b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" build_string = %s\n", build_string_.c_str()); 35951b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" dbg_bld_str = %s\n", dbg_bld_str_.c_str()); 35961b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } else { 35971b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" build_string = (invalid)\n"); 35981b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" dbg_bld_str = (invalid)\n"); 35991b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org } 36003562017ff5a65fff770cd798903ee294db912aa2ivan.penkov@gmail.com } 360176f052f8fbf8864dee5992b857229d06560a766ammentovai printf("\n"); 360276f052f8fbf8864dee5992b857229d06560a766ammentovai} 360376f052f8fbf8864dee5992b857229d06560a766ammentovai 360476f052f8fbf8864dee5992b857229d06560a766ammentovai 360576f052f8fbf8864dee5992b857229d06560a766ammentovai// 3606e5dc60822e5938fea2ae892ccddb906641ba174emmentovai// MinidumpBreakpadInfo 360776f052f8fbf8864dee5992b857229d06560a766ammentovai// 360876f052f8fbf8864dee5992b857229d06560a766ammentovai 360976f052f8fbf8864dee5992b857229d06560a766ammentovai 3610e5dc60822e5938fea2ae892ccddb906641ba174emmentovaiMinidumpBreakpadInfo::MinidumpBreakpadInfo(Minidump* minidump) 361176f052f8fbf8864dee5992b857229d06560a766ammentovai : MinidumpStream(minidump), 3612e5dc60822e5938fea2ae892ccddb906641ba174emmentovai breakpad_info_() { 361376f052f8fbf8864dee5992b857229d06560a766ammentovai} 361476f052f8fbf8864dee5992b857229d06560a766ammentovai 361576f052f8fbf8864dee5992b857229d06560a766ammentovai 36166162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpBreakpadInfo::Read(uint32_t expected_size) { 361776f052f8fbf8864dee5992b857229d06560a766ammentovai valid_ = false; 361876f052f8fbf8864dee5992b857229d06560a766ammentovai 3619af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (expected_size != sizeof(breakpad_info_)) { 3620af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpBreakpadInfo size mismatch, " << expected_size << 3621af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai " != " << sizeof(breakpad_info_); 362276f052f8fbf8864dee5992b857229d06560a766ammentovai return false; 3623af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 362476f052f8fbf8864dee5992b857229d06560a766ammentovai 3625af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!minidump_->ReadBytes(&breakpad_info_, sizeof(breakpad_info_))) { 3626af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpBreakpadInfo cannot read Breakpad info"; 362776f052f8fbf8864dee5992b857229d06560a766ammentovai return false; 3628af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 362976f052f8fbf8864dee5992b857229d06560a766ammentovai 363076f052f8fbf8864dee5992b857229d06560a766ammentovai if (minidump_->swap()) { 3631e5dc60822e5938fea2ae892ccddb906641ba174emmentovai Swap(&breakpad_info_.validity); 3632e5dc60822e5938fea2ae892ccddb906641ba174emmentovai Swap(&breakpad_info_.dump_thread_id); 3633e5dc60822e5938fea2ae892ccddb906641ba174emmentovai Swap(&breakpad_info_.requesting_thread_id); 363476f052f8fbf8864dee5992b857229d06560a766ammentovai } 363576f052f8fbf8864dee5992b857229d06560a766ammentovai 363676f052f8fbf8864dee5992b857229d06560a766ammentovai valid_ = true; 363776f052f8fbf8864dee5992b857229d06560a766ammentovai return true; 363876f052f8fbf8864dee5992b857229d06560a766ammentovai} 363976f052f8fbf8864dee5992b857229d06560a766ammentovai 364076f052f8fbf8864dee5992b857229d06560a766ammentovai 36416162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpBreakpadInfo::GetDumpThreadID(uint32_t *thread_id) const { 3642af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !thread_id) << "MinidumpBreakpadInfo::GetDumpThreadID " 3643af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "requires |thread_id|"; 3644af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai assert(thread_id); 3645af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai *thread_id = 0; 3646af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 3647af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 3648af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpBreakpadInfo for GetDumpThreadID"; 3649af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return false; 3650af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 3651af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 3652af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!(breakpad_info_.validity & MD_BREAKPAD_INFO_VALID_DUMP_THREAD_ID)) { 3653af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "MinidumpBreakpadInfo has no dump thread"; 365476f052f8fbf8864dee5992b857229d06560a766ammentovai return false; 365576f052f8fbf8864dee5992b857229d06560a766ammentovai } 365676f052f8fbf8864dee5992b857229d06560a766ammentovai 3657e5dc60822e5938fea2ae892ccddb906641ba174emmentovai *thread_id = breakpad_info_.dump_thread_id; 365876f052f8fbf8864dee5992b857229d06560a766ammentovai return true; 365976f052f8fbf8864dee5992b857229d06560a766ammentovai} 366076f052f8fbf8864dee5992b857229d06560a766ammentovai 366176f052f8fbf8864dee5992b857229d06560a766ammentovai 36626162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpBreakpadInfo::GetRequestingThreadID(uint32_t *thread_id) 366376f052f8fbf8864dee5992b857229d06560a766ammentovai const { 3664af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !thread_id) << "MinidumpBreakpadInfo::GetRequestingThreadID " 3665af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "requires |thread_id|"; 3666af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai assert(thread_id); 3667af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai *thread_id = 0; 3668af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 3669af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!thread_id || !valid_) { 3670af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid MinidumpBreakpadInfo for GetRequestingThreadID"; 3671af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return false; 3672af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 3673af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 3674af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!(breakpad_info_.validity & 3675af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai MD_BREAKPAD_INFO_VALID_REQUESTING_THREAD_ID)) { 3676af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "MinidumpBreakpadInfo has no requesting thread"; 367776f052f8fbf8864dee5992b857229d06560a766ammentovai return false; 367876f052f8fbf8864dee5992b857229d06560a766ammentovai } 367976f052f8fbf8864dee5992b857229d06560a766ammentovai 3680e5dc60822e5938fea2ae892ccddb906641ba174emmentovai *thread_id = breakpad_info_.requesting_thread_id; 368176f052f8fbf8864dee5992b857229d06560a766ammentovai return true; 368276f052f8fbf8864dee5992b857229d06560a766ammentovai} 368376f052f8fbf8864dee5992b857229d06560a766ammentovai 368476f052f8fbf8864dee5992b857229d06560a766ammentovai 3685e5dc60822e5938fea2ae892ccddb906641ba174emmentovaivoid MinidumpBreakpadInfo::Print() { 3686af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 3687af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "MinidumpBreakpadInfo cannot print invalid data"; 368876f052f8fbf8864dee5992b857229d06560a766ammentovai return; 3689af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 369076f052f8fbf8864dee5992b857229d06560a766ammentovai 3691e5dc60822e5938fea2ae892ccddb906641ba174emmentovai printf("MDRawBreakpadInfo\n"); 3692e5dc60822e5938fea2ae892ccddb906641ba174emmentovai printf(" validity = 0x%x\n", breakpad_info_.validity); 36931b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" dump_thread_id = "); 36941b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(breakpad_info_.validity & 36951b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_BREAKPAD_INFO_VALID_DUMP_THREAD_ID, 36961b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatHexadecimal, breakpad_info_.dump_thread_id); 36971b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org printf(" requesting_thread_id = "); 36981b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org PrintValueOrInvalid(breakpad_info_.validity & 36991b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org MD_BREAKPAD_INFO_VALID_REQUESTING_THREAD_ID, 37001b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org kNumberFormatHexadecimal, 37011b9922ced527aeee4b085aa58916af43d86b99eemark@chromium.org breakpad_info_.requesting_thread_id); 370276f052f8fbf8864dee5992b857229d06560a766ammentovai 370376f052f8fbf8864dee5992b857229d06560a766ammentovai printf("\n"); 37043261e8b6eac44a41341f112821482bee6c940c98mmentovai} 37053261e8b6eac44a41341f112821482bee6c940c98mmentovai 37063261e8b6eac44a41341f112821482bee6c940c98mmentovai 37073261e8b6eac44a41341f112821482bee6c940c98mmentovai// 37087b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek// MinidumpMemoryInfo 37097b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek// 37107b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37117b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37127b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekMinidumpMemoryInfo::MinidumpMemoryInfo(Minidump* minidump) 37137b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek : MinidumpObject(minidump), 37147b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek memory_info_() { 37157b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 37167b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37177b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37187b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekbool MinidumpMemoryInfo::IsExecutable() const { 37196162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t protection = 37207b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek memory_info_.protection & MD_MEMORY_PROTECTION_ACCESS_MASK; 37217b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return protection == MD_MEMORY_PROTECT_EXECUTE || 37227b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek protection == MD_MEMORY_PROTECT_EXECUTE_READ || 37237b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek protection == MD_MEMORY_PROTECT_EXECUTE_READWRITE; 37247b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 37257b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37267b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37277b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekbool MinidumpMemoryInfo::IsWritable() const { 37286162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t protection = 37297b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek memory_info_.protection & MD_MEMORY_PROTECTION_ACCESS_MASK; 37307b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return protection == MD_MEMORY_PROTECT_READWRITE || 37317b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek protection == MD_MEMORY_PROTECT_WRITECOPY || 37327b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek protection == MD_MEMORY_PROTECT_EXECUTE_READWRITE || 37337b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek protection == MD_MEMORY_PROTECT_EXECUTE_WRITECOPY; 37347b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 37357b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37367b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37377b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekbool MinidumpMemoryInfo::Read() { 37387b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek valid_ = false; 37397b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37407b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!minidump_->ReadBytes(&memory_info_, sizeof(memory_info_))) { 37417b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfo cannot read memory info"; 37427b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 37437b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 37447b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37457b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (minidump_->swap()) { 37467b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&memory_info_.base_address); 37477b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&memory_info_.allocation_base); 37487b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&memory_info_.allocation_protection); 37497b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&memory_info_.region_size); 37507b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&memory_info_.state); 37517b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&memory_info_.protection); 37527b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&memory_info_.type); 37537b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 37547b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37557b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek // Check for base + size overflow or undersize. 37567b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (memory_info_.region_size == 0 || 37576162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com memory_info_.region_size > numeric_limits<uint64_t>::max() - 37587b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek memory_info_.base_address) { 37597b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfo has a memory region problem, " << 37607b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek HexString(memory_info_.base_address) << "+" << 37617b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek HexString(memory_info_.region_size); 37627b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 37637b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 37647b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37657b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek valid_ = true; 37667b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return true; 37677b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 37687b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37697b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37707b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekvoid MinidumpMemoryInfo::Print() { 37717b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!valid_) { 37727b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfo cannot print invalid data"; 37737b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return; 37747b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 37757b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37767b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf("MDRawMemoryInfo\n"); 37777b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf(" base_address = 0x%" PRIx64 "\n", 37787b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek memory_info_.base_address); 37797b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf(" allocation_base = 0x%" PRIx64 "\n", 37807b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek memory_info_.allocation_base); 37817b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf(" allocation_protection = 0x%x\n", 37827b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek memory_info_.allocation_protection); 37837b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf(" region_size = 0x%" PRIx64 "\n", memory_info_.region_size); 37847b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf(" state = 0x%x\n", memory_info_.state); 37857b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf(" protection = 0x%x\n", memory_info_.protection); 37867b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf(" type = 0x%x\n", memory_info_.type); 37877b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 37887b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37897b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37907b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek// 37917b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek// MinidumpMemoryInfoList 37927b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek// 37937b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37947b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 37957b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekMinidumpMemoryInfoList::MinidumpMemoryInfoList(Minidump* minidump) 37967b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek : MinidumpStream(minidump), 37976162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com range_map_(new RangeMap<uint64_t, unsigned int>()), 37987b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek infos_(NULL), 37997b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek info_count_(0) { 38007b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 38017b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38027b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38037b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekMinidumpMemoryInfoList::~MinidumpMemoryInfoList() { 38047b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek delete range_map_; 38057b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek delete infos_; 38067b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 38077b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38087b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38096162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool MinidumpMemoryInfoList::Read(uint32_t expected_size) { 38107b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek // Invalidate cached data. 38117b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek delete infos_; 38127b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek infos_ = NULL; 38137b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek range_map_->Clear(); 38147b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek info_count_ = 0; 38157b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38167b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek valid_ = false; 38177b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38187b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek MDRawMemoryInfoList header; 38197b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (expected_size < sizeof(MDRawMemoryInfoList)) { 38207b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList header size mismatch, " << 38217b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek expected_size << " < " << sizeof(MDRawMemoryInfoList); 38227b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 38237b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 38247b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!minidump_->ReadBytes(&header, sizeof(header))) { 38257b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList could not read header"; 38267b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 38277b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 38287b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38297b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (minidump_->swap()) { 38307b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&header.size_of_header); 38317b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&header.size_of_entry); 38327b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek Swap(&header.number_of_entries); 38337b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 38347b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38357b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek // Sanity check that the header is the expected size. 3836f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // TODO(ted): could possibly handle this more gracefully, assuming 38377b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek // that future versions of the structs would be backwards-compatible. 38387b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (header.size_of_header != sizeof(MDRawMemoryInfoList)) { 38397b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList header size mismatch, " << 38407b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek header.size_of_header << " != " << 38417b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek sizeof(MDRawMemoryInfoList); 38427b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 38437b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 38447b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38457b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek // Sanity check that the entries are the expected size. 38467b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (header.size_of_entry != sizeof(MDRawMemoryInfo)) { 38477b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList entry size mismatch, " << 38487b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek header.size_of_entry << " != " << 38497b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek sizeof(MDRawMemoryInfo); 38507b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 38517b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 38527b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38537b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (header.number_of_entries > 38546162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com numeric_limits<uint32_t>::max() / sizeof(MDRawMemoryInfo)) { 38557b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList info count " << 38567b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek header.number_of_entries << 38577b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek " would cause multiplication overflow"; 38587b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 38597b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 38607b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38617b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (expected_size != sizeof(MDRawMemoryInfoList) + 38627b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek header.number_of_entries * sizeof(MDRawMemoryInfo)) { 38637b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList size mismatch, " << expected_size << 38647b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek " != " << sizeof(MDRawMemoryInfoList) + 38657b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek header.number_of_entries * sizeof(MDRawMemoryInfo); 38667b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 38677b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 38687b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 3869f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // Check for data loss when converting header.number_of_entries from 3870f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // uint64_t into MinidumpMemoryInfos::size_type (uint32_t) 3871f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com MinidumpMemoryInfos::size_type header_number_of_entries = 3872f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com static_cast<unsigned int>(header.number_of_entries); 3873f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com if (static_cast<uint64_t>(header_number_of_entries) != 3874f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com header.number_of_entries) { 3875f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "Data loss detected when converting " 3876f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com "the header's number_of_entries"; 3877f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com return false; 3878f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com } 3879f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com 38807b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (header.number_of_entries != 0) { 38817b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek scoped_ptr<MinidumpMemoryInfos> infos( 3882f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com new MinidumpMemoryInfos(header_number_of_entries, 38837b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek MinidumpMemoryInfo(minidump_))); 38847b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38857b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek for (unsigned int index = 0; 38867b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek index < header.number_of_entries; 38877b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek ++index) { 38887b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek MinidumpMemoryInfo* info = &(*infos)[index]; 38897b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38907b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek // Assume that the file offset is correct after the last read. 38917b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!info->Read()) { 38927b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList cannot read info " << 38937b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek index << "/" << header.number_of_entries; 38947b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 38957b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 38967b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 38976162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t base_address = info->GetBase(); 3898f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com uint64_t region_size = info->GetSize(); 38997b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39007b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!range_map_->StoreRange(base_address, region_size, index)) { 39017b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList could not store" 39027b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek " memory region " << 39037b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek index << "/" << header.number_of_entries << ", " << 39047b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek HexString(base_address) << "+" << 39057b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek HexString(region_size); 39067b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return false; 39077b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39087b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39097b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39107b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek infos_ = infos.release(); 39117b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39127b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 3913f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com info_count_ = header_number_of_entries; 39147b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39157b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek valid_ = true; 39167b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return true; 39177b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 39187b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39197b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39207b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekconst MinidumpMemoryInfo* MinidumpMemoryInfoList::GetMemoryInfoAtIndex( 39217b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek unsigned int index) const { 39227b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!valid_) { 39237b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "Invalid MinidumpMemoryInfoList for GetMemoryInfoAtIndex"; 39247b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return NULL; 39257b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39267b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39277b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (index >= info_count_) { 39287b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList index out of range: " << 39297b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek index << "/" << info_count_; 39307b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return NULL; 39317b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39327b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39337b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return &(*infos_)[index]; 39347b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 39357b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39367b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39377b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekconst MinidumpMemoryInfo* MinidumpMemoryInfoList::GetMemoryInfoForAddress( 39386162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint64_t address) const { 39397b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!valid_) { 39407b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "Invalid MinidumpMemoryInfoList for" 39417b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek " GetMemoryInfoForAddress"; 39427b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return NULL; 39437b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39447b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39457b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek unsigned int info_index; 39467b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!range_map_->RetrieveRange(address, &info_index, NULL, NULL)) { 39477b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(INFO) << "MinidumpMemoryInfoList has no memory info at " << 39487b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek HexString(address); 39497b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return NULL; 39507b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39517b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39527b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return GetMemoryInfoAtIndex(info_index); 39537b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 39547b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39557b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39567b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekvoid MinidumpMemoryInfoList::Print() { 39577b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek if (!valid_) { 39587b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek BPLOG(ERROR) << "MinidumpMemoryInfoList cannot print invalid data"; 39597b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return; 39607b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39617b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39627b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf("MinidumpMemoryInfoList\n"); 39637b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf(" info_count = %d\n", info_count_); 39647b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf("\n"); 39657b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39667b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek for (unsigned int info_index = 0; 39677b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek info_index < info_count_; 39687b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek ++info_index) { 39697b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf("info[%d]\n", info_index); 39707b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek (*infos_)[info_index].Print(); 39717b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek printf("\n"); 39727b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek } 39737b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 39747b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39757b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 39767b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek// 39773261e8b6eac44a41341f112821482bee6c940c98mmentovai// Minidump 39783261e8b6eac44a41341f112821482bee6c940c98mmentovai// 39793261e8b6eac44a41341f112821482bee6c940c98mmentovai 39803261e8b6eac44a41341f112821482bee6c940c98mmentovai 39816162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.comuint32_t Minidump::max_streams_ = 128; 3982e96a791d9a0886a24ce08afe13207e8e105542e3mmentovaiunsigned int Minidump::max_string_length_ = 1024; 3983e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 3984e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 39856dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovaiMinidump::Minidump(const string& path) 398653d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai : header_(), 398753d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai directory_(NULL), 3988373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai stream_map_(new MinidumpStreamMap()), 39896dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai path_(path), 39900cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek stream_(NULL), 399153d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai swap_(false), 399253d0f69d35375fe2ffd119ac7e4898083c0e071cmmentovai valid_(false) { 39933261e8b6eac44a41341f112821482bee6c940c98mmentovai} 39943261e8b6eac44a41341f112821482bee6c940c98mmentovai 39950cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarekMinidump::Minidump(istream& stream) 39960cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek : header_(), 39970cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek directory_(NULL), 39980cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek stream_map_(new MinidumpStreamMap()), 39990cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek path_(), 40000cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek stream_(&stream), 40010cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek swap_(false), 40020cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek valid_(false) { 40030cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek} 40043261e8b6eac44a41341f112821482bee6c940c98mmentovai 40053261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidump::~Minidump() { 40060cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek if (stream_) { 40070cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek BPLOG(INFO) << "Minidump closing minidump"; 40080cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek } 40090cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek if (!path_.empty()) { 40100cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek delete stream_; 40110cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek } 40123261e8b6eac44a41341f112821482bee6c940c98mmentovai delete directory_; 40133261e8b6eac44a41341f112821482bee6c940c98mmentovai delete stream_map_; 40146dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai} 40156dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai 40166dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai 40176dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovaibool Minidump::Open() { 40180cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek if (stream_ != NULL) { 40190cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek BPLOG(INFO) << "Minidump reopening minidump " << path_; 4020af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 40216dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai // The file is already open. Seek to the beginning, which is the position 40226dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai // the file would be at if it were opened anew. 40236dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai return SeekSet(0); 40246dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai } 40256dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai 40260cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek stream_ = new ifstream(path_.c_str(), std::ios::in | std::ios::binary); 40270cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek if (!stream_ || !stream_->good()) { 4028af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai string error_string; 4029af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai int error_code = ErrnoString(&error_string); 4030af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump could not open minidump " << path_ << 4031af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai ", error " << error_code << ": " << error_string; 40326dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai return false; 4033af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 40346dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai 40350cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek BPLOG(INFO) << "Minidump opened minidump " << path_; 40366dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai return true; 40373261e8b6eac44a41341f112821482bee6c940c98mmentovai} 40383261e8b6eac44a41341f112821482bee6c940c98mmentovai 40396162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool Minidump::GetContextCPUFlagsFromSystemInfo(uint32_t *context_cpu_flags) { 4040233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com // Initialize output parameters 4041233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = 0; 4042233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com 4043233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com // Save the current stream position 4044233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com off_t saved_position = Tell(); 4045233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com if (saved_position == -1) { 4046233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com // Failed to save the current stream position. 4047233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com // Returns true because the current position of the stream is preserved. 4048233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com return true; 4049233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } 4050233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com 4051233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com const MDRawSystemInfo* system_info = 4052233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com GetSystemInfo() ? GetSystemInfo()->system_info() : NULL; 4053233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com 4054233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com if (system_info != NULL) { 4055233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com switch (system_info->processor_architecture) { 4056233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_X86: 4057233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_X86; 4058233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4059233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_MIPS: 4060233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_MIPS; 4061233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4062233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_ALPHA: 4063233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_ALPHA; 4064233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4065233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_PPC: 4066233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_PPC; 4067233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4068cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org case MD_CPU_ARCHITECTURE_PPC64: 4069cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org *context_cpu_flags = MD_CONTEXT_PPC64; 4070cd1f1a6399d5a73ac2bdf5671f8322ba013e8e21thestig@chromium.org break; 4071233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_SHX: 4072233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_SHX; 4073233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4074233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_ARM: 4075233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_ARM; 4076233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 407739d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org case MD_CPU_ARCHITECTURE_ARM64: 407839d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org *context_cpu_flags = MD_CONTEXT_ARM64; 407939d7964df5d3ba7c3889bf19004f6e18eee5cfc6mark@chromium.org break; 4080233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_IA64: 4081233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_IA64; 4082233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4083233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_ALPHA64: 4084233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = 0; 4085233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4086233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_MSIL: 4087233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = 0; 4088233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4089233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_AMD64: 4090233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_AMD64; 4091233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4092233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_X86_WIN64: 4093233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = 0; 4094233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4095233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_SPARC: 4096233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = MD_CONTEXT_SPARC; 4097233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4098233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com case MD_CPU_ARCHITECTURE_UNKNOWN: 4099233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = 0; 4100233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4101233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com default: 4102233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com *context_cpu_flags = 0; 4103233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com break; 4104233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } 4105233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com } 4106233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com 4107233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com // Restore position and return 4108233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com return SeekSet(saved_position); 4109233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com} 4110233501d467e38955ba38065b7c717486b94c1da9ivan.penkov@gmail.com 41113261e8b6eac44a41341f112821482bee6c940c98mmentovai 41123261e8b6eac44a41341f112821482bee6c940c98mmentovaibool Minidump::Read() { 41133261e8b6eac44a41341f112821482bee6c940c98mmentovai // Invalidate cached data. 41143261e8b6eac44a41341f112821482bee6c940c98mmentovai delete directory_; 41153261e8b6eac44a41341f112821482bee6c940c98mmentovai directory_ = NULL; 4116373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai stream_map_->clear(); 41173261e8b6eac44a41341f112821482bee6c940c98mmentovai 41183261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = false; 41193261e8b6eac44a41341f112821482bee6c940c98mmentovai 4120af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!Open()) { 4121af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump cannot open minidump"; 41226dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai return false; 4123af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 41246dd21d3baf490a34af0b1b1f5a48f8443dcb1ea6mmentovai 4125af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!ReadBytes(&header_, sizeof(MDRawHeader))) { 4126af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump cannot read header"; 41273261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 4128af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 41293261e8b6eac44a41341f112821482bee6c940c98mmentovai 41303261e8b6eac44a41341f112821482bee6c940c98mmentovai if (header_.signature != MD_HEADER_SIGNATURE) { 41313261e8b6eac44a41341f112821482bee6c940c98mmentovai // The file may be byte-swapped. Under the present architecture, these 41323261e8b6eac44a41341f112821482bee6c940c98mmentovai // classes don't know or need to know what CPU (or endianness) the 41333261e8b6eac44a41341f112821482bee6c940c98mmentovai // minidump was produced on in order to parse it. Use the signature as 41343261e8b6eac44a41341f112821482bee6c940c98mmentovai // a byte order marker. 41356162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t signature_swapped = header_.signature; 41363261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&signature_swapped); 41373261e8b6eac44a41341f112821482bee6c940c98mmentovai if (signature_swapped != MD_HEADER_SIGNATURE) { 41383261e8b6eac44a41341f112821482bee6c940c98mmentovai // This isn't a minidump or a byte-swapped minidump. 4139af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump header signature mismatch: (" << 4140af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(header_.signature) << ", " << 4141af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(signature_swapped) << ") != " << 4142af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(MD_HEADER_SIGNATURE); 41433261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 41443261e8b6eac44a41341f112821482bee6c940c98mmentovai } 41453261e8b6eac44a41341f112821482bee6c940c98mmentovai swap_ = true; 41463261e8b6eac44a41341f112821482bee6c940c98mmentovai } else { 41473261e8b6eac44a41341f112821482bee6c940c98mmentovai // The file is not byte-swapped. Set swap_ false (it may have been true 41483261e8b6eac44a41341f112821482bee6c940c98mmentovai // if the object is being reused?) 41493261e8b6eac44a41341f112821482bee6c940c98mmentovai swap_ = false; 41503261e8b6eac44a41341f112821482bee6c940c98mmentovai } 41513261e8b6eac44a41341f112821482bee6c940c98mmentovai 4152af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "Minidump " << (swap_ ? "" : "not ") << 4153af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "byte-swapping minidump"; 4154af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 41553261e8b6eac44a41341f112821482bee6c940c98mmentovai if (swap_) { 41563261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&header_.signature); 41573261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&header_.version); 41583261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&header_.stream_count); 41593261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&header_.stream_directory_rva); 41603261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&header_.checksum); 41613261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&header_.time_date_stamp); 41623261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&header_.flags); 41633261e8b6eac44a41341f112821482bee6c940c98mmentovai } 41643261e8b6eac44a41341f112821482bee6c940c98mmentovai 41653261e8b6eac44a41341f112821482bee6c940c98mmentovai // Version check. The high 16 bits of header_.version contain something 41663261e8b6eac44a41341f112821482bee6c940c98mmentovai // else "implementation specific." 41673261e8b6eac44a41341f112821482bee6c940c98mmentovai if ((header_.version & 0x0000ffff) != MD_HEADER_VERSION) { 4168af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump version mismatch: " << 4169af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(header_.version & 0x0000ffff) << " != " << 4170af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai HexString(MD_HEADER_VERSION); 41713261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 41723261e8b6eac44a41341f112821482bee6c940c98mmentovai } 41733261e8b6eac44a41341f112821482bee6c940c98mmentovai 4174af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!SeekSet(header_.stream_directory_rva)) { 4175af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump cannot seek to stream directory"; 41763261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 4177af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 41783261e8b6eac44a41341f112821482bee6c940c98mmentovai 4179e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (header_.stream_count > max_streams_) { 4180e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "Minidump stream count " << header_.stream_count << 4181e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai " exceeds maximum " << max_streams_; 4182e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai return false; 4183e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai } 4184e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 4185e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (header_.stream_count != 0) { 4186373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai scoped_ptr<MinidumpDirectoryEntries> directory( 4187373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai new MinidumpDirectoryEntries(header_.stream_count)); 41883261e8b6eac44a41341f112821482bee6c940c98mmentovai 4189373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Read the entire array in one fell swoop, instead of reading one entry 4190373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // at a time in the loop. 4191373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai if (!ReadBytes(&(*directory)[0], 4192af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai sizeof(MDRawDirectory) * header_.stream_count)) { 4193af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump cannot read stream directory"; 4194373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 4195af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 41963261e8b6eac44a41341f112821482bee6c940c98mmentovai 4197373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai for (unsigned int stream_index = 0; 4198373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai stream_index < header_.stream_count; 4199373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai ++stream_index) { 4200373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai MDRawDirectory* directory_entry = &(*directory)[stream_index]; 42013261e8b6eac44a41341f112821482bee6c940c98mmentovai 4202373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai if (swap_) { 4203373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai Swap(&directory_entry->stream_type); 4204373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai Swap(&directory_entry->location); 4205373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 42063261e8b6eac44a41341f112821482bee6c940c98mmentovai 4207373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Initialize the stream_map_ map, which speeds locating a stream by 4208373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // type. 4209373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai unsigned int stream_type = directory_entry->stream_type; 4210373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai switch (stream_type) { 4211373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai case MD_THREAD_LIST_STREAM: 4212373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai case MD_MODULE_LIST_STREAM: 4213373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai case MD_MEMORY_LIST_STREAM: 4214373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai case MD_EXCEPTION_STREAM: 4215373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai case MD_SYSTEM_INFO_STREAM: 4216373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai case MD_MISC_INFO_STREAM: 4217e5dc60822e5938fea2ae892ccddb906641ba174emmentovai case MD_BREAKPAD_INFO_STREAM: { 4218373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai if (stream_map_->find(stream_type) != stream_map_->end()) { 4219373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Another stream with this type was already found. A minidump 4220373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // file should contain at most one of each of these stream types. 4221af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump found multiple streams of type " << 4222af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai stream_type << ", but can only deal with one"; 4223373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return false; 4224373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 4225373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Fall through to default 42263261e8b6eac44a41341f112821482bee6c940c98mmentovai } 42273261e8b6eac44a41341f112821482bee6c940c98mmentovai 4228373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai default: { 4229373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // Overwrites for stream types other than those above, but it's 4230373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai // expected to be the user's burden in that case. 4231373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai (*stream_map_)[stream_type].stream_index = stream_index; 4232373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 42333261e8b6eac44a41341f112821482bee6c940c98mmentovai } 42343261e8b6eac44a41341f112821482bee6c940c98mmentovai } 42353261e8b6eac44a41341f112821482bee6c940c98mmentovai 4236373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai directory_ = directory.release(); 4237373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 42383261e8b6eac44a41341f112821482bee6c940c98mmentovai 42393261e8b6eac44a41341f112821482bee6c940c98mmentovai valid_ = true; 42403261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 42413261e8b6eac44a41341f112821482bee6c940c98mmentovai} 42423261e8b6eac44a41341f112821482bee6c940c98mmentovai 42433261e8b6eac44a41341f112821482bee6c940c98mmentovai 42443261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpThreadList* Minidump::GetThreadList() { 42453261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpThreadList* thread_list; 42463261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetStream(&thread_list); 42473261e8b6eac44a41341f112821482bee6c940c98mmentovai} 42483261e8b6eac44a41341f112821482bee6c940c98mmentovai 42493261e8b6eac44a41341f112821482bee6c940c98mmentovai 42503261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpModuleList* Minidump::GetModuleList() { 42513261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpModuleList* module_list; 42523261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetStream(&module_list); 42533261e8b6eac44a41341f112821482bee6c940c98mmentovai} 42543261e8b6eac44a41341f112821482bee6c940c98mmentovai 42553261e8b6eac44a41341f112821482bee6c940c98mmentovai 42563261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMemoryList* Minidump::GetMemoryList() { 42573261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpMemoryList* memory_list; 42583261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetStream(&memory_list); 42593261e8b6eac44a41341f112821482bee6c940c98mmentovai} 42603261e8b6eac44a41341f112821482bee6c940c98mmentovai 42613261e8b6eac44a41341f112821482bee6c940c98mmentovai 42623261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpException* Minidump::GetException() { 42633261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpException* exception; 42643261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetStream(&exception); 42653261e8b6eac44a41341f112821482bee6c940c98mmentovai} 42663261e8b6eac44a41341f112821482bee6c940c98mmentovai 42670314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarekMinidumpAssertion* Minidump::GetAssertion() { 42680314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek MinidumpAssertion* assertion; 42690314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek return GetStream(&assertion); 42700314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek} 42710314e487e46a45229e275eb78b09f0538a5a7769ted.mielczarek 42723261e8b6eac44a41341f112821482bee6c940c98mmentovai 42733261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpSystemInfo* Minidump::GetSystemInfo() { 42743261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpSystemInfo* system_info; 42753261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetStream(&system_info); 42763261e8b6eac44a41341f112821482bee6c940c98mmentovai} 42773261e8b6eac44a41341f112821482bee6c940c98mmentovai 42783261e8b6eac44a41341f112821482bee6c940c98mmentovai 42793261e8b6eac44a41341f112821482bee6c940c98mmentovaiMinidumpMiscInfo* Minidump::GetMiscInfo() { 42803261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpMiscInfo* misc_info; 42813261e8b6eac44a41341f112821482bee6c940c98mmentovai return GetStream(&misc_info); 42823261e8b6eac44a41341f112821482bee6c940c98mmentovai} 42833261e8b6eac44a41341f112821482bee6c940c98mmentovai 42843261e8b6eac44a41341f112821482bee6c940c98mmentovai 4285e5dc60822e5938fea2ae892ccddb906641ba174emmentovaiMinidumpBreakpadInfo* Minidump::GetBreakpadInfo() { 4286e5dc60822e5938fea2ae892ccddb906641ba174emmentovai MinidumpBreakpadInfo* breakpad_info; 4287e5dc60822e5938fea2ae892ccddb906641ba174emmentovai return GetStream(&breakpad_info); 428876f052f8fbf8864dee5992b857229d06560a766ammentovai} 428976f052f8fbf8864dee5992b857229d06560a766ammentovai 42907b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarekMinidumpMemoryInfoList* Minidump::GetMemoryInfoList() { 42917b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek MinidumpMemoryInfoList* memory_info_list; 42927b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek return GetStream(&memory_info_list); 42937b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek} 42947b8e2b7e090b2d1223d0944d1e7da1d4c571bb5dted.mielczarek 42956f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.comstatic const char* get_stream_name(uint32_t stream_type) { 42966f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com switch (stream_type) { 42976f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_UNUSED_STREAM: 42986f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_UNUSED_STREAM"; 42996f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_RESERVED_STREAM_0: 43006f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_RESERVED_STREAM_0"; 43016f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_RESERVED_STREAM_1: 43026f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_RESERVED_STREAM_1"; 43036f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_THREAD_LIST_STREAM: 43046f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_THREAD_LIST_STREAM"; 43056f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_MODULE_LIST_STREAM: 43066f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_MODULE_LIST_STREAM"; 43076f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_MEMORY_LIST_STREAM: 43086f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_MEMORY_LIST_STREAM"; 43096f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_EXCEPTION_STREAM: 43106f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_EXCEPTION_STREAM"; 43116f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_SYSTEM_INFO_STREAM: 43126f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_SYSTEM_INFO_STREAM"; 43136f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_THREAD_EX_LIST_STREAM: 43146f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_THREAD_EX_LIST_STREAM"; 43156f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_MEMORY_64_LIST_STREAM: 43166f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_MEMORY_64_LIST_STREAM"; 43176f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_COMMENT_STREAM_A: 43186f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_COMMENT_STREAM_A"; 43196f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_COMMENT_STREAM_W: 43206f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_COMMENT_STREAM_W"; 43216f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_HANDLE_DATA_STREAM: 43226f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_HANDLE_DATA_STREAM"; 43236f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_FUNCTION_TABLE_STREAM: 43246f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_FUNCTION_TABLE_STREAM"; 43256f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_UNLOADED_MODULE_LIST_STREAM: 43266f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_UNLOADED_MODULE_LIST_STREAM"; 43276f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_MISC_INFO_STREAM: 43286f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_MISC_INFO_STREAM"; 43296f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_MEMORY_INFO_LIST_STREAM: 43306f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_MEMORY_INFO_LIST_STREAM"; 43316f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_THREAD_INFO_LIST_STREAM: 43326f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_THREAD_INFO_LIST_STREAM"; 43336f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_HANDLE_OPERATION_LIST_STREAM: 43346f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_HANDLE_OPERATION_LIST_STREAM"; 43356f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LAST_RESERVED_STREAM: 43366f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LAST_RESERVED_STREAM"; 43376f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_BREAKPAD_INFO_STREAM: 43386f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_BREAKPAD_INFO_STREAM"; 43396f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_ASSERTION_INFO_STREAM: 43406f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_ASSERTION_INFO_STREAM"; 43416f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LINUX_CPU_INFO: 43426f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LINUX_CPU_INFO"; 43436f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LINUX_PROC_STATUS: 43446f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LINUX_PROC_STATUS"; 43456f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LINUX_LSB_RELEASE: 43466f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LINUX_LSB_RELEASE"; 43476f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LINUX_CMD_LINE: 43486f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LINUX_CMD_LINE"; 43496f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LINUX_ENVIRON: 43506f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LINUX_ENVIRON"; 43516f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LINUX_AUXV: 43526f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LINUX_AUXV"; 43536f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LINUX_MAPS: 43546f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LINUX_MAPS"; 43556f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com case MD_LINUX_DSO_DEBUG: 43566f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "MD_LINUX_DSO_DEBUG"; 43576f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com default: 43586f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com return "unknown"; 43596f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com } 43606f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com} 436176f052f8fbf8864dee5992b857229d06560a766ammentovai 43623261e8b6eac44a41341f112821482bee6c940c98mmentovaivoid Minidump::Print() { 4363af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 4364af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump cannot print invalid data"; 43653261e8b6eac44a41341f112821482bee6c940c98mmentovai return; 4366af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 43673261e8b6eac44a41341f112821482bee6c940c98mmentovai 43683261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MDRawHeader\n"); 43693261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" signature = 0x%x\n", header_.signature); 43703261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" version = 0x%x\n", header_.version); 43713261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" stream_count = %d\n", header_.stream_count); 43723261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" stream_directory_rva = 0x%x\n", header_.stream_directory_rva); 43733261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" checksum = 0x%x\n", header_.checksum); 43749c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org printf(" time_date_stamp = 0x%x %s\n", 43759c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org header_.time_date_stamp, 43769c98e0ff65fc324ce34f4133b63a1201080f5c02mark@chromium.org TimeTToUTCString(header_.time_date_stamp).c_str()); 4377c27cf3e3959189f78fe2de40405987c3f33488cemmentovai printf(" flags = 0x%" PRIx64 "\n", header_.flags); 43783261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 43793261e8b6eac44a41341f112821482bee6c940c98mmentovai 43803261e8b6eac44a41341f112821482bee6c940c98mmentovai for (unsigned int stream_index = 0; 43813261e8b6eac44a41341f112821482bee6c940c98mmentovai stream_index < header_.stream_count; 43823261e8b6eac44a41341f112821482bee6c940c98mmentovai ++stream_index) { 43833261e8b6eac44a41341f112821482bee6c940c98mmentovai MDRawDirectory* directory_entry = &(*directory_)[stream_index]; 43843261e8b6eac44a41341f112821482bee6c940c98mmentovai 43853261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("mDirectory[%d]\n", stream_index); 43863261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("MDRawDirectory\n"); 43876f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com printf(" stream_type = 0x%x (%s)\n", directory_entry->stream_type, 43886f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com get_stream_name(directory_entry->stream_type)); 43893261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" location.data_size = %d\n", 43903261e8b6eac44a41341f112821482bee6c940c98mmentovai directory_entry->location.data_size); 43913261e8b6eac44a41341f112821482bee6c940c98mmentovai printf(" location.rva = 0x%x\n", directory_entry->location.rva); 43923261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 43933261e8b6eac44a41341f112821482bee6c940c98mmentovai } 43943261e8b6eac44a41341f112821482bee6c940c98mmentovai 43953261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("Streams:\n"); 43963261e8b6eac44a41341f112821482bee6c940c98mmentovai for (MinidumpStreamMap::const_iterator iterator = stream_map_->begin(); 43973261e8b6eac44a41341f112821482bee6c940c98mmentovai iterator != stream_map_->end(); 43983261e8b6eac44a41341f112821482bee6c940c98mmentovai ++iterator) { 43996162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t stream_type = iterator->first; 44003261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpStreamInfo info = iterator->second; 44016f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com printf(" stream type 0x%x (%s) at index %d\n", stream_type, 44026f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com get_stream_name(stream_type), 44036f1bd8c271098d15a63ee8ec3d3d0fb625e6c2c8ted.mielczarek@gmail.com info.stream_index); 44043261e8b6eac44a41341f112821482bee6c940c98mmentovai } 44053261e8b6eac44a41341f112821482bee6c940c98mmentovai printf("\n"); 44063261e8b6eac44a41341f112821482bee6c940c98mmentovai} 44073261e8b6eac44a41341f112821482bee6c940c98mmentovai 44083261e8b6eac44a41341f112821482bee6c940c98mmentovai 44093261e8b6eac44a41341f112821482bee6c940c98mmentovaiconst MDRawDirectory* Minidump::GetDirectoryEntryAtIndex(unsigned int index) 44103261e8b6eac44a41341f112821482bee6c940c98mmentovai const { 4411af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 4412af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid Minidump for GetDirectoryEntryAtIndex"; 44133261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 4414af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 4415af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 4416af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (index >= header_.stream_count) { 4417af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Minidump stream directory index out of range: " << 4418af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai index << "/" << header_.stream_count; 4419af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai return NULL; 4420af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 44213261e8b6eac44a41341f112821482bee6c940c98mmentovai 44223261e8b6eac44a41341f112821482bee6c940c98mmentovai return &(*directory_)[index]; 44233261e8b6eac44a41341f112821482bee6c940c98mmentovai} 44243261e8b6eac44a41341f112821482bee6c940c98mmentovai 44253261e8b6eac44a41341f112821482bee6c940c98mmentovai 44263261e8b6eac44a41341f112821482bee6c940c98mmentovaibool Minidump::ReadBytes(void* bytes, size_t count) { 44273261e8b6eac44a41341f112821482bee6c940c98mmentovai // Can't check valid_ because Read needs to call this method before 44280cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek // validity can be determined. 44290cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek if (!stream_) { 44300cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek return false; 44310cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek } 44320cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek stream_->read(static_cast<char*>(bytes), count); 4433f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com std::streamsize bytes_read = stream_->gcount(); 4434f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com if (bytes_read == -1) { 4435f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com string error_string; 4436f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com int error_code = ErrnoString(&error_string); 4437f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "ReadBytes: error " << error_code << ": " << error_string; 4438f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com return false; 4439f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com } 4440f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com 4441f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // Convert to size_t and check for data loss 4442f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com size_t bytes_read_converted = static_cast<size_t>(bytes_read); 4443f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com if (static_cast<std::streamsize>(bytes_read_converted) != bytes_read) { 4444f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "ReadBytes: conversion data loss detected when converting " 4445f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com << bytes_read << " to " << bytes_read_converted; 44463261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 4447af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 4448f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com 4449f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com if (bytes_read_converted != count) { 4450f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "ReadBytes: read " << bytes_read_converted << "/" << count; 4451f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com return false; 4452f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com } 4453f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com 44543261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 44553261e8b6eac44a41341f112821482bee6c940c98mmentovai} 44563261e8b6eac44a41341f112821482bee6c940c98mmentovai 44573261e8b6eac44a41341f112821482bee6c940c98mmentovai 44583261e8b6eac44a41341f112821482bee6c940c98mmentovaibool Minidump::SeekSet(off_t offset) { 44593261e8b6eac44a41341f112821482bee6c940c98mmentovai // Can't check valid_ because Read needs to call this method before 44600cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek // validity can be determined. 44610cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek if (!stream_) { 44620cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek return false; 44630cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek } 44640cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek stream_->seekg(offset, std::ios_base::beg); 44650cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek if (!stream_->good()) { 44660cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek string error_string; 44670cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek int error_code = ErrnoString(&error_string); 44680cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek BPLOG(ERROR) << "SeekSet: error " << error_code << ": " << error_string; 44693261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 4470af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 44713261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 44723261e8b6eac44a41341f112821482bee6c940c98mmentovai} 44733261e8b6eac44a41341f112821482bee6c940c98mmentovai 44740cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarekoff_t Minidump::Tell() { 44750cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek if (!valid_ || !stream_) { 44760cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek return (off_t)-1; 44770cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek } 44780cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek 4479f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com // Check for conversion data loss 4480f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com std::streamoff std_streamoff = stream_->tellg(); 4481f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com off_t rv = static_cast<off_t>(std_streamoff); 4482f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com if (static_cast<std::streamoff>(rv) == std_streamoff) { 4483f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com return rv; 4484f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com } else { 4485f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com BPLOG(ERROR) << "Data loss detected"; 4486f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com return (off_t)-1; 4487f7838a8665eb5e46f2eb136679b27707d6adb523ivan.penkov@gmail.com } 44880cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek} 44890cbd50c975c1680a6cd0f1ce2760e157e2bfd46fted.mielczarek 44903261e8b6eac44a41341f112821482bee6c940c98mmentovai 44913261e8b6eac44a41341f112821482bee6c940c98mmentovaistring* Minidump::ReadString(off_t offset) { 4492af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 4493af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid Minidump for ReadString"; 44943261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 4495af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 4496af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!SeekSet(offset)) { 4497e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "ReadString could not seek to string at offset " << offset; 44983261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 4499af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 45003261e8b6eac44a41341f112821482bee6c940c98mmentovai 45016162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t bytes; 4502af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!ReadBytes(&bytes, sizeof(bytes))) { 4503e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "ReadString could not read string size at offset " << 4504e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai offset; 45053261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 4506af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 45073261e8b6eac44a41341f112821482bee6c940c98mmentovai if (swap_) 45083261e8b6eac44a41341f112821482bee6c940c98mmentovai Swap(&bytes); 45093261e8b6eac44a41341f112821482bee6c940c98mmentovai 4510af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (bytes % 2 != 0) { 4511e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "ReadString found odd-sized " << bytes << 4512e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai "-byte string at offset " << offset; 45133261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 4514af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 45153261e8b6eac44a41341f112821482bee6c940c98mmentovai unsigned int utf16_words = bytes / 2; 45163261e8b6eac44a41341f112821482bee6c940c98mmentovai 4517e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai if (utf16_words > max_string_length_) { 4518e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "ReadString string length " << utf16_words << 4519e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai " exceeds maximum " << max_string_length_ << 4520e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai " at offset " << offset; 4521e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai return NULL; 4522e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai } 4523e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai 45246162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com vector<uint16_t> string_utf16(utf16_words); 45253261e8b6eac44a41341f112821482bee6c940c98mmentovai 4526373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai if (utf16_words) { 4527373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai if (!ReadBytes(&string_utf16[0], bytes)) { 4528e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai BPLOG(ERROR) << "ReadString could not read " << bytes << 4529e96a791d9a0886a24ce08afe13207e8e105542e3mmentovai "-byte string at offset " << offset; 4530373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai return NULL; 4531373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 4532373c49b4167c8a41bfb3adfa613866f0ca516e1cmmentovai } 45333261e8b6eac44a41341f112821482bee6c940c98mmentovai 45343261e8b6eac44a41341f112821482bee6c940c98mmentovai return UTF16ToUTF8(string_utf16, swap_); 45353261e8b6eac44a41341f112821482bee6c940c98mmentovai} 45363261e8b6eac44a41341f112821482bee6c940c98mmentovai 45373261e8b6eac44a41341f112821482bee6c940c98mmentovai 45386162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.combool Minidump::SeekToStreamType(uint32_t stream_type, 45396162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t* stream_length) { 4540af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !stream_length) << "Minidump::SeekToStreamType requires " 4541af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai "|stream_length|"; 4542af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai assert(stream_length); 4543af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai *stream_length = 0; 4544af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 4545af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 4546af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid Mindump for SeekToStreamType"; 45473261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 4548af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 45493261e8b6eac44a41341f112821482bee6c940c98mmentovai 45503261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpStreamMap::const_iterator iterator = stream_map_->find(stream_type); 45513261e8b6eac44a41341f112821482bee6c940c98mmentovai if (iterator == stream_map_->end()) { 45523261e8b6eac44a41341f112821482bee6c940c98mmentovai // This stream type didn't exist in the directory. 4553af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "SeekToStreamType: type " << stream_type << " not present"; 45543261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 45553261e8b6eac44a41341f112821482bee6c940c98mmentovai } 45563261e8b6eac44a41341f112821482bee6c940c98mmentovai 45573261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpStreamInfo info = iterator->second; 4558af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (info.stream_index >= header_.stream_count) { 4559af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "SeekToStreamType: type " << stream_type << 4560af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai " out of range: " << 4561af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai info.stream_index << "/" << header_.stream_count; 45623261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 4563af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 45643261e8b6eac44a41341f112821482bee6c940c98mmentovai 45653261e8b6eac44a41341f112821482bee6c940c98mmentovai MDRawDirectory* directory_entry = &(*directory_)[info.stream_index]; 4566af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!SeekSet(directory_entry->location.rva)) { 4567af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "SeekToStreamType could not seek to stream type " << 4568af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai stream_type; 45693261e8b6eac44a41341f112821482bee6c940c98mmentovai return false; 4570af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 45713261e8b6eac44a41341f112821482bee6c940c98mmentovai 45723261e8b6eac44a41341f112821482bee6c940c98mmentovai *stream_length = directory_entry->location.data_size; 45733261e8b6eac44a41341f112821482bee6c940c98mmentovai 45743261e8b6eac44a41341f112821482bee6c940c98mmentovai return true; 45753261e8b6eac44a41341f112821482bee6c940c98mmentovai} 45763261e8b6eac44a41341f112821482bee6c940c98mmentovai 45773261e8b6eac44a41341f112821482bee6c940c98mmentovai 45783261e8b6eac44a41341f112821482bee6c940c98mmentovaitemplate<typename T> 45793261e8b6eac44a41341f112821482bee6c940c98mmentovaiT* Minidump::GetStream(T** stream) { 45803261e8b6eac44a41341f112821482bee6c940c98mmentovai // stream is a garbage parameter that's present only to account for C++'s 45813261e8b6eac44a41341f112821482bee6c940c98mmentovai // inability to overload a method based solely on its return type. 45823261e8b6eac44a41341f112821482bee6c940c98mmentovai 45836162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com const uint32_t stream_type = T::kStreamType; 4584af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai 4585af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG_IF(ERROR, !stream) << "Minidump::GetStream type " << stream_type << 4586af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai " requires |stream|"; 4587af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai assert(stream); 45883261e8b6eac44a41341f112821482bee6c940c98mmentovai *stream = NULL; 45893261e8b6eac44a41341f112821482bee6c940c98mmentovai 4590af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!valid_) { 4591af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "Invalid Minidump for GetStream type " << stream_type; 45923261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 4593af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 45943261e8b6eac44a41341f112821482bee6c940c98mmentovai 45953261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpStreamMap::iterator iterator = stream_map_->find(stream_type); 45963261e8b6eac44a41341f112821482bee6c940c98mmentovai if (iterator == stream_map_->end()) { 45973261e8b6eac44a41341f112821482bee6c940c98mmentovai // This stream type didn't exist in the directory. 4598af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(INFO) << "GetStream: type " << stream_type << " not present"; 45993261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 46003261e8b6eac44a41341f112821482bee6c940c98mmentovai } 46013261e8b6eac44a41341f112821482bee6c940c98mmentovai 46023261e8b6eac44a41341f112821482bee6c940c98mmentovai // Get a pointer so that the stored stream field can be altered. 46033261e8b6eac44a41341f112821482bee6c940c98mmentovai MinidumpStreamInfo* info = &iterator->second; 46043261e8b6eac44a41341f112821482bee6c940c98mmentovai 46053261e8b6eac44a41341f112821482bee6c940c98mmentovai if (info->stream) { 46063261e8b6eac44a41341f112821482bee6c940c98mmentovai // This cast is safe because info.stream is only populated by this 46073261e8b6eac44a41341f112821482bee6c940c98mmentovai // method, and there is a direct correlation between T and stream_type. 46083261e8b6eac44a41341f112821482bee6c940c98mmentovai *stream = static_cast<T*>(info->stream); 46093261e8b6eac44a41341f112821482bee6c940c98mmentovai return *stream; 46103261e8b6eac44a41341f112821482bee6c940c98mmentovai } 46113261e8b6eac44a41341f112821482bee6c940c98mmentovai 46126162aed3c3fcfc53373c963ac375d39a5dfa5a25ted.mielczarek@gmail.com uint32_t stream_length; 4613af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!SeekToStreamType(stream_type, &stream_length)) { 4614af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "GetStream could not seek to stream type " << stream_type; 46153261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 4616af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 46173261e8b6eac44a41341f112821482bee6c940c98mmentovai 46182466d8e993a800a17e00deda2f3a27e0505140e1mmentovai scoped_ptr<T> new_stream(new T(this)); 46193261e8b6eac44a41341f112821482bee6c940c98mmentovai 4620af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai if (!new_stream->Read(stream_length)) { 4621af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai BPLOG(ERROR) << "GetStream could not read stream type " << stream_type; 46223261e8b6eac44a41341f112821482bee6c940c98mmentovai return NULL; 4623af3c43f00e98047bc7f80dcf4c16b876e095769fmmentovai } 46243261e8b6eac44a41341f112821482bee6c940c98mmentovai 46253261e8b6eac44a41341f112821482bee6c940c98mmentovai *stream = new_stream.release(); 46263261e8b6eac44a41341f112821482bee6c940c98mmentovai info->stream = *stream; 46273261e8b6eac44a41341f112821482bee6c940c98mmentovai return *stream; 46283261e8b6eac44a41341f112821482bee6c940c98mmentovai} 46293261e8b6eac44a41341f112821482bee6c940c98mmentovai 46303261e8b6eac44a41341f112821482bee6c940c98mmentovai 4631e5dc60822e5938fea2ae892ccddb906641ba174emmentovai} // namespace google_breakpad 4632