Change the way function and public symbols are obtained to use the findChildren DIA function.

This has a substantial performance improvement over using the getSymbolsByAddr iterator, especially on certain 64bit DLLS.  e.g. Time to process chrome_child.dll drops from 51 minutes to 21 secs.

Note: new test data looks different because the ordering of lines is no longer by memory address.  This does not affect processing.  The test data has been manually compared to old data and matches (except additional PUBLIC symbols).  Also, INFO lines are omitted because the source executable files are not checked in, so they are unavailable.

R=ivanpe@chromium.org, mark@chromium.org

Review URL: https://breakpad.appspot.com/1574002

git-svn-id: http://google-breakpad.googlecode.com/svn/trunk@1316 4c0a9323-5329-0410-9bdc-e9ce6186880e
This commit is contained in:
wfh@chromium.org 2014-04-16 22:50:28 +00:00
parent 7cc286a5a7
commit c83a81450c
7 changed files with 14694 additions and 13740 deletions

View file

@ -37,6 +37,7 @@
#include <stdio.h> #include <stdio.h>
#include <limits> #include <limits>
#include <set>
#include "common/windows/dia_util.h" #include "common/windows/dia_util.h"
#include "common/windows/guid_string.h" #include "common/windows/guid_string.h"
@ -54,9 +55,12 @@
* *
*/ */
typedef unsigned char UBYTE; typedef unsigned char UBYTE;
#if !defined(_WIN64)
#define UNW_FLAG_EHANDLER 0x01 #define UNW_FLAG_EHANDLER 0x01
#define UNW_FLAG_UHANDLER 0x02 #define UNW_FLAG_UHANDLER 0x02
#define UNW_FLAG_CHAININFO 0x04 #define UNW_FLAG_CHAININFO 0x04
#endif
union UnwindCode { union UnwindCode {
struct { struct {
@ -340,47 +344,59 @@ bool PDBSourceLineWriter::PrintSourceFiles() {
} }
bool PDBSourceLineWriter::PrintFunctions() { bool PDBSourceLineWriter::PrintFunctions() {
CComPtr<IDiaEnumSymbolsByAddr> symbols; ULONG count = 0;
if (FAILED(session_->getSymbolsByAddr(&symbols))) { DWORD rva = 0;
fprintf(stderr, "failed to get symbol enumerator\n"); CComPtr<IDiaSymbol> global;
std::set<DWORD> rvas;
HRESULT hr;
if (FAILED(session_->get_globalScope(&global))) {
fprintf(stderr, "get_globalScope failed\n");
return false; return false;
} }
CComPtr<IDiaSymbol> symbol; CComPtr<IDiaEnumSymbols> symbols = NULL;
if (FAILED(symbols->symbolByAddr(1, 0, &symbol))) {
fprintf(stderr, "failed to enumerate symbols\n");
return false;
}
DWORD rva_last = 0; // Find all public symbols.
if (FAILED(symbol->get_relativeVirtualAddress(&rva_last))) { hr = global->findChildren(SymTagPublicSymbol, NULL, nsNone, &symbols);
fprintf(stderr, "failed to get symbol rva\n");
return false;
}
ULONG count; if (SUCCEEDED(hr)) {
do { CComPtr<IDiaSymbol> symbol = NULL;
DWORD tag;
if (FAILED(symbol->get_symTag(&tag))) { while (SUCCEEDED(symbols->Next(1, &symbol, &count)) && count == 1) {
fprintf(stderr, "failed to get symbol tag\n"); if (!PrintCodePublicSymbol(symbol))
return false; return false;
symbol.Release();
} }
// For a given function, DIA seems to give either a symbol with symbols.Release();
// SymTagFunction or SymTagPublicSymbol, but not both. This means }
// that PDBSourceLineWriter will output either a FUNC or PUBLIC line,
// but not both. // Find all function symbols.
if (tag == SymTagFunction) { hr = global->findChildren(SymTagFunction, NULL, nsNone, &symbols);
if (!PrintFunction(symbol, symbol)) {
return false; if (SUCCEEDED(hr)) {
} CComPtr<IDiaSymbol> symbol = NULL;
} else if (tag == SymTagPublicSymbol) {
if (!PrintCodePublicSymbol(symbol)) { while (SUCCEEDED(symbols->Next(1, &symbol, &count)) && count == 1) {
if (SUCCEEDED(symbol->get_relativeVirtualAddress(&rva))) {
// To maintain existing behavior of one symbol per address, place the
// rva onto a set here to uniquify them.
if (rvas.count(rva) == 0) {
rvas.insert(rva);
if (!PrintFunction(symbol, symbol))
return false;
}
} else {
fprintf(stderr, "get_relativeVirtualAddress failed on the symbol\n");
return false; return false;
} }
symbol.Release();
} }
symbol.Release();
} while (SUCCEEDED(symbols->Next(1, &symbol, &count)) && count == 1); symbols.Release();
}
// When building with PGO, the compiler can split functions into // When building with PGO, the compiler can split functions into
// "hot" and "cold" blocks, and move the "cold" blocks out to separate // "hot" and "cold" blocks, and move the "cold" blocks out to separate
@ -389,12 +405,6 @@ bool PDBSourceLineWriter::PrintFunctions() {
// that are children of them. We can then find the lexical parents // that are children of them. We can then find the lexical parents
// of those blocks and print out an extra FUNC line for blocks // of those blocks and print out an extra FUNC line for blocks
// that are not contained in their parent functions. // that are not contained in their parent functions.
CComPtr<IDiaSymbol> global;
if (FAILED(session_->get_globalScope(&global))) {
fprintf(stderr, "get_globalScope failed\n");
return false;
}
CComPtr<IDiaEnumSymbols> compilands; CComPtr<IDiaEnumSymbols> compilands;
if (FAILED(global->findChildren(SymTagCompiland, NULL, if (FAILED(global->findChildren(SymTagCompiland, NULL,
nsNone, &compilands))) { nsNone, &compilands))) {
@ -440,6 +450,7 @@ bool PDBSourceLineWriter::PrintFunctions() {
compiland.Release(); compiland.Release();
} }
global.Release();
return true; return true;
} }

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff