mirror of
https://github.com/yuzu-emu/breakpad.git
synced 2025-01-11 13:35:32 +00:00
Follow debug link correctly
As thestig@chromium.org pointed out in https://breakpad.appspot.com/9684002, LoadSymbols() should return false if |read_gnu_debug_link| is false. BUG=chromium:453498 R=thestig@chromium.org Review URL: https://breakpad.appspot.com/2844002 git-svn-id: http://google-breakpad.googlecode.com/svn/trunk@1422 4c0a9323-5329-0410-9bdc-e9ce6186880e
This commit is contained in:
parent
9f00bce09a
commit
854b9f74a0
|
@ -628,6 +628,35 @@ bool LoadSymbols(const string& obj_file,
|
||||||
"DWARF debugging information\n", obj_file.c_str());
|
"DWARF debugging information\n", obj_file.c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// See if there are export symbols available.
|
||||||
|
const Shdr* dynsym_section =
|
||||||
|
FindElfSectionByName<ElfClass>(".dynsym", SHT_DYNSYM,
|
||||||
|
sections, names, names_end,
|
||||||
|
elf_header->e_shnum);
|
||||||
|
const Shdr* dynstr_section =
|
||||||
|
FindElfSectionByName<ElfClass>(".dynstr", SHT_STRTAB,
|
||||||
|
sections, names, names_end,
|
||||||
|
elf_header->e_shnum);
|
||||||
|
if (dynsym_section && dynstr_section) {
|
||||||
|
info->LoadedSection(".dynsym");
|
||||||
|
|
||||||
|
const uint8_t* dynsyms =
|
||||||
|
GetOffset<ElfClass, uint8_t>(elf_header,
|
||||||
|
dynsym_section->sh_offset);
|
||||||
|
const uint8_t* dynstrs =
|
||||||
|
GetOffset<ElfClass, uint8_t>(elf_header,
|
||||||
|
dynstr_section->sh_offset);
|
||||||
|
bool result =
|
||||||
|
ELFSymbolsToModule(dynsyms,
|
||||||
|
dynsym_section->sh_size,
|
||||||
|
dynstrs,
|
||||||
|
dynstr_section->sh_size,
|
||||||
|
big_endian,
|
||||||
|
ElfClass::kAddrSize,
|
||||||
|
module);
|
||||||
|
found_usable_info = found_usable_info || result;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (options.symbol_data != NO_CFI) {
|
if (options.symbol_data != NO_CFI) {
|
||||||
|
@ -689,8 +718,6 @@ bool LoadSymbols(const string& obj_file,
|
||||||
names_end, elf_header->e_shnum);
|
names_end, elf_header->e_shnum);
|
||||||
if (gnu_debuglink_section) {
|
if (gnu_debuglink_section) {
|
||||||
if (!info->debug_dirs().empty()) {
|
if (!info->debug_dirs().empty()) {
|
||||||
found_debug_info_section = true;
|
|
||||||
|
|
||||||
const char* debuglink_contents =
|
const char* debuglink_contents =
|
||||||
GetOffset<ElfClass, char>(elf_header,
|
GetOffset<ElfClass, char>(elf_header,
|
||||||
gnu_debuglink_section->sh_offset);
|
gnu_debuglink_section->sh_offset);
|
||||||
|
@ -709,47 +736,20 @@ bool LoadSymbols(const string& obj_file,
|
||||||
fprintf(stderr, "%s does not contain a .gnu_debuglink section.\n",
|
fprintf(stderr, "%s does not contain a .gnu_debuglink section.\n",
|
||||||
obj_file.c_str());
|
obj_file.c_str());
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
}
|
// Return true if some usable information was found, since the caller
|
||||||
|
// doesn't want to use .gnu_debuglink.
|
||||||
if (options.symbol_data != ONLY_CFI) {
|
|
||||||
const Shdr* dynsym_section =
|
|
||||||
FindElfSectionByName<ElfClass>(".dynsym", SHT_DYNSYM,
|
|
||||||
sections, names, names_end,
|
|
||||||
elf_header->e_shnum);
|
|
||||||
const Shdr* dynstr_section =
|
|
||||||
FindElfSectionByName<ElfClass>(".dynstr", SHT_STRTAB,
|
|
||||||
sections, names, names_end,
|
|
||||||
elf_header->e_shnum);
|
|
||||||
if (dynsym_section && dynstr_section) {
|
|
||||||
info->LoadedSection(".dynsym");
|
|
||||||
|
|
||||||
const uint8_t* dynsyms =
|
|
||||||
GetOffset<ElfClass, uint8_t>(elf_header,
|
|
||||||
dynsym_section->sh_offset);
|
|
||||||
const uint8_t* dynstrs =
|
|
||||||
GetOffset<ElfClass, uint8_t>(elf_header,
|
|
||||||
dynstr_section->sh_offset);
|
|
||||||
bool result =
|
|
||||||
ELFSymbolsToModule(dynsyms,
|
|
||||||
dynsym_section->sh_size,
|
|
||||||
dynstrs,
|
|
||||||
dynstr_section->sh_size,
|
|
||||||
big_endian,
|
|
||||||
ElfClass::kAddrSize,
|
|
||||||
module);
|
|
||||||
found_usable_info = found_usable_info || result;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (read_gnu_debug_link) {
|
|
||||||
return found_debug_info_section;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return true if some usable information was found
|
|
||||||
return found_usable_info;
|
return found_usable_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// No debug info was found, let the user try again with .gnu_debuglink
|
||||||
|
// if present.
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// Return the breakpad symbol file identifier for the architecture of
|
// Return the breakpad symbol file identifier for the architecture of
|
||||||
// ELF_HEADER.
|
// ELF_HEADER.
|
||||||
template<typename ElfClass>
|
template<typename ElfClass>
|
||||||
|
|
|
@ -79,6 +79,17 @@ void Module::AddFunction(Function *function) {
|
||||||
// FUNC lines must not hold an empty name, so catch the problem early if
|
// FUNC lines must not hold an empty name, so catch the problem early if
|
||||||
// callers try to add one.
|
// callers try to add one.
|
||||||
assert(!function->name.empty());
|
assert(!function->name.empty());
|
||||||
|
|
||||||
|
// FUNCs are better than PUBLICs as they come with sizes, so remove an extern
|
||||||
|
// with the same address if present.
|
||||||
|
Extern ext(function->address);
|
||||||
|
ExternSet::iterator it_ext = externs_.lower_bound(&ext);
|
||||||
|
if (it_ext != externs_.end() &&
|
||||||
|
(*it_ext)->address < function->address + function->size) {
|
||||||
|
delete *it_ext;
|
||||||
|
externs_.erase(it_ext);
|
||||||
|
}
|
||||||
|
|
||||||
std::pair<FunctionSet::iterator,bool> ret = functions_.insert(function);
|
std::pair<FunctionSet::iterator,bool> ret = functions_.insert(function);
|
||||||
if (!ret.second && (*ret.first != function)) {
|
if (!ret.second && (*ret.first != function)) {
|
||||||
// Free the duplicate that was not inserted because this Module
|
// Free the duplicate that was not inserted because this Module
|
||||||
|
@ -98,21 +109,12 @@ void Module::AddStackFrameEntry(StackFrameEntry *stack_frame_entry) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void Module::AddExtern(Extern *ext) {
|
void Module::AddExtern(Extern *ext) {
|
||||||
Function func(ext->name, ext->address);
|
|
||||||
|
|
||||||
// Since parsing debug section and public info are not necessarily
|
|
||||||
// mutually exclusive, check if the symbol has already been read
|
|
||||||
// as a function to avoid duplicates.
|
|
||||||
if (functions_.find(&func) == functions_.end()) {
|
|
||||||
std::pair<ExternSet::iterator,bool> ret = externs_.insert(ext);
|
std::pair<ExternSet::iterator,bool> ret = externs_.insert(ext);
|
||||||
if (!ret.second) {
|
if (!ret.second) {
|
||||||
// Free the duplicate that was not inserted because this Module
|
// Free the duplicate that was not inserted because this Module
|
||||||
// now owns it.
|
// now owns it.
|
||||||
delete ext;
|
delete ext;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
delete ext;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Module::GetFunctions(vector<Function *> *vec,
|
void Module::GetFunctions(vector<Function *> *vec,
|
||||||
|
|
Loading…
Reference in a new issue