Make DumpSymbols methods take a vector of debug_dirs

r=thestig at https://breakpad.appspot.com/512002/

git-svn-id: http://google-breakpad.googlecode.com/svn/trunk@1102 4c0a9323-5329-0410-9bdc-e9ce6186880e
This commit is contained in:
ted.mielczarek@gmail.com 2013-01-22 13:36:39 +00:00
parent 834f7ee611
commit c6f6d9ef30
4 changed files with 60 additions and 44 deletions

View file

@ -387,7 +387,7 @@ template<typename ElfClass>
string ReadDebugLink(const char* debuglink, string ReadDebugLink(const char* debuglink,
size_t debuglink_size, size_t debuglink_size,
const string& obj_file, const string& obj_file,
const string& debug_dir) { const std::vector<string>& debug_dirs) {
size_t debuglink_len = strlen(debuglink) + 5; // '\0' + CRC32. size_t debuglink_len = strlen(debuglink) + 5; // '\0' + CRC32.
debuglink_len = 4 * ((debuglink_len + 3) / 4); // Round to nearest 4 bytes. debuglink_len = 4 * ((debuglink_len + 3) / 4); // Round to nearest 4 bytes.
@ -398,13 +398,26 @@ string ReadDebugLink(const char* debuglink,
return ""; return "";
} }
string debuglink_path = debug_dir + "/" + debuglink; bool found = false;
int debuglink_fd = open(debuglink_path.c_str(), O_RDONLY); int debuglink_fd = -1;
if (debuglink_fd < 0) { string debuglink_path;
std::vector<string>::const_iterator it;
for (it = debug_dirs.begin(); it < debug_dirs.end(); ++it) {
const string& debug_dir = *it;
debuglink_path = debug_dir + "/" + debuglink;
debuglink_fd = open(debuglink_path.c_str(), O_RDONLY);
if (debuglink_fd >= 0) {
found = true;
break;
}
}
if (!found) {
fprintf(stderr, "Failed to open debug ELF file '%s' for '%s': %s\n", fprintf(stderr, "Failed to open debug ELF file '%s' for '%s': %s\n",
debuglink_path.c_str(), obj_file.c_str(), strerror(errno)); debuglink_path.c_str(), obj_file.c_str(), strerror(errno));
return ""; return "";
} }
FDWrapper debuglink_fd_wrapper(debuglink_fd); FDWrapper debuglink_fd_wrapper(debuglink_fd);
// TODO(thestig) check the CRC-32 at the end of the .gnu_debuglink // TODO(thestig) check the CRC-32 at the end of the .gnu_debuglink
// section. // section.
@ -424,8 +437,8 @@ class LoadSymbolsInfo {
public: public:
typedef typename ElfClass::Addr Addr; typedef typename ElfClass::Addr Addr;
explicit LoadSymbolsInfo(const string &dbg_dir) : explicit LoadSymbolsInfo(const std::vector<string>& dbg_dirs) :
debug_dir_(dbg_dir), debug_dirs_(dbg_dirs),
has_loading_addr_(false) {} has_loading_addr_(false) {}
// Keeps track of which sections have been loaded so sections don't // Keeps track of which sections have been loaded so sections don't
@ -458,8 +471,8 @@ class LoadSymbolsInfo {
} }
// Setters and getters // Setters and getters
const string &debug_dir() const { const std::vector<string>& debug_dirs() const {
return debug_dir_; return debug_dirs_;
} }
string debuglink_file() const { string debuglink_file() const {
@ -470,7 +483,8 @@ class LoadSymbolsInfo {
} }
private: private:
const string &debug_dir_; // Directory with the debug ELF file. const std::vector<string>& debug_dirs_; // Directories in which to
// search for the debug ELF file.
string debuglink_file_; // Full path to the debug ELF file. string debuglink_file_; // Full path to the debug ELF file.
@ -601,14 +615,14 @@ bool LoadSymbols(const string& obj_file,
sections, names, sections, names,
names_end, elf_header->e_shnum); names_end, elf_header->e_shnum);
if (gnu_debuglink_section) { if (gnu_debuglink_section) {
if (!info->debug_dir().empty()) { if (!info->debug_dirs().empty()) {
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);
string debuglink_file string debuglink_file
= ReadDebugLink<ElfClass>(debuglink_contents, = ReadDebugLink<ElfClass>(debuglink_contents,
gnu_debuglink_section->sh_size, gnu_debuglink_section->sh_size,
obj_file, info->debug_dir()); obj_file, info->debug_dirs());
info->set_debuglink_file(debuglink_file); info->set_debuglink_file(debuglink_file);
} else { } else {
fprintf(stderr, ".gnu_debuglink section found in '%s', " fprintf(stderr, ".gnu_debuglink section found in '%s', "
@ -712,7 +726,7 @@ string BaseFileName(const string &filename) {
template<typename ElfClass> template<typename ElfClass>
bool ReadSymbolDataElfClass(const typename ElfClass::Ehdr* elf_header, bool ReadSymbolDataElfClass(const typename ElfClass::Ehdr* elf_header,
const string& obj_filename, const string& obj_filename,
const string& debug_dir, const std::vector<string>& debug_dirs,
bool cfi, bool cfi,
Module** out_module) { Module** out_module) {
typedef typename ElfClass::Ehdr Ehdr; typedef typename ElfClass::Ehdr Ehdr;
@ -744,10 +758,10 @@ bool ReadSymbolDataElfClass(const typename ElfClass::Ehdr* elf_header,
string os = "Linux"; string os = "Linux";
string id = FormatIdentifier(identifier); string id = FormatIdentifier(identifier);
LoadSymbolsInfo<ElfClass> info(debug_dir); LoadSymbolsInfo<ElfClass> info(debug_dirs);
scoped_ptr<Module> module(new Module(name, os, architecture, id)); scoped_ptr<Module> module(new Module(name, os, architecture, id));
if (!LoadSymbols<ElfClass>(obj_filename, big_endian, elf_header, if (!LoadSymbols<ElfClass>(obj_filename, big_endian, elf_header,
!debug_dir.empty(), &info, module.get())) { !debug_dirs.empty(), &info, module.get())) {
const string debuglink_file = info.debuglink_file(); const string debuglink_file = info.debuglink_file();
if (debuglink_file.empty()) if (debuglink_file.empty())
return false; return false;
@ -801,7 +815,7 @@ namespace google_breakpad {
// Not explicitly exported, but not static so it can be used in unit tests. // Not explicitly exported, but not static so it can be used in unit tests.
bool ReadSymbolDataInternal(const uint8_t* obj_file, bool ReadSymbolDataInternal(const uint8_t* obj_file,
const string& obj_filename, const string& obj_filename,
const string& debug_dir, const std::vector<string>& debug_dirs,
bool cfi, bool cfi,
Module** module) { Module** module) {
@ -813,12 +827,12 @@ bool ReadSymbolDataInternal(const uint8_t* obj_file,
int elfclass = ElfClass(obj_file); int elfclass = ElfClass(obj_file);
if (elfclass == ELFCLASS32) { if (elfclass == ELFCLASS32) {
return ReadSymbolDataElfClass<ElfClass32>( return ReadSymbolDataElfClass<ElfClass32>(
reinterpret_cast<const Elf32_Ehdr*>(obj_file), obj_filename, debug_dir, reinterpret_cast<const Elf32_Ehdr*>(obj_file), obj_filename, debug_dirs,
cfi, module); cfi, module);
} }
if (elfclass == ELFCLASS64) { if (elfclass == ELFCLASS64) {
return ReadSymbolDataElfClass<ElfClass64>( return ReadSymbolDataElfClass<ElfClass64>(
reinterpret_cast<const Elf64_Ehdr*>(obj_file), obj_filename, debug_dir, reinterpret_cast<const Elf64_Ehdr*>(obj_file), obj_filename, debug_dirs,
cfi, module); cfi, module);
} }
@ -826,11 +840,11 @@ bool ReadSymbolDataInternal(const uint8_t* obj_file,
} }
bool WriteSymbolFile(const string &obj_file, bool WriteSymbolFile(const string &obj_file,
const string &debug_dir, const std::vector<string>& debug_dirs,
bool cfi, bool cfi,
std::ostream &sym_stream) { std::ostream &sym_stream) {
Module* module; Module* module;
if (!ReadSymbolData(obj_file, debug_dir, cfi, &module)) if (!ReadSymbolData(obj_file, debug_dirs, cfi, &module))
return false; return false;
bool result = module->Write(sym_stream, cfi); bool result = module->Write(sym_stream, cfi);
@ -839,7 +853,7 @@ bool WriteSymbolFile(const string &obj_file,
} }
bool ReadSymbolData(const string& obj_file, bool ReadSymbolData(const string& obj_file,
const string& debug_dir, const std::vector<string>& debug_dirs,
bool cfi, bool cfi,
Module** module) { Module** module) {
MmapWrapper map_wrapper; MmapWrapper map_wrapper;
@ -848,7 +862,7 @@ bool ReadSymbolData(const string& obj_file,
return false; return false;
return ReadSymbolDataInternal(reinterpret_cast<uint8_t*>(elf_header), return ReadSymbolDataInternal(reinterpret_cast<uint8_t*>(elf_header),
obj_file, debug_dir, cfi, module); obj_file, debug_dirs, cfi, module);
} }
} // namespace google_breakpad } // namespace google_breakpad

View file

@ -37,6 +37,7 @@
#include <iostream> #include <iostream>
#include <string> #include <string>
#include <vector>
#include "common/using_std_string.h" #include "common/using_std_string.h"
@ -48,10 +49,10 @@ class Module;
// or shared library, and write it to SYM_STREAM in the Breakpad symbol // or shared library, and write it to SYM_STREAM in the Breakpad symbol
// file format. // file format.
// If OBJ_FILE has been stripped but contains a .gnu_debuglink section, // If OBJ_FILE has been stripped but contains a .gnu_debuglink section,
// then look for the debug file in DEBUG_DIR. // then look for the debug file in DEBUG_DIRS.
// If CFI is set to false, then omit the CFI section. // If CFI is set to false, then omit the CFI section.
bool WriteSymbolFile(const string &obj_file, bool WriteSymbolFile(const string &obj_file,
const string &debug_dir, const std::vector<string>& debug_dirs,
bool cfi, bool cfi,
std::ostream &sym_stream); std::ostream &sym_stream);
@ -59,7 +60,7 @@ bool WriteSymbolFile(const string &obj_file,
// instead of writing it to a stream. The caller owns the resulting // instead of writing it to a stream. The caller owns the resulting
// Module object and must delete it when finished. // Module object and must delete it when finished.
bool ReadSymbolData(const string& obj_file, bool ReadSymbolData(const string& obj_file,
const string& debug_dir, const std::vector<string>& debug_dirs,
bool cfi, bool cfi,
Module** module); Module** module);

View file

@ -46,8 +46,8 @@
namespace google_breakpad { namespace google_breakpad {
bool ReadSymbolDataInternal(const uint8_t* obj_file, bool ReadSymbolDataInternal(const uint8_t* obj_file,
const string &obj_filename, const string& obj_filename,
const string &debug_dir, const std::vector<string>& debug_dir,
bool cfi, bool cfi,
Module** module); Module** module);
} }
@ -84,10 +84,10 @@ TEST_F(DumpSymbols, Invalid) {
memset(&header, 0, sizeof(header)); memset(&header, 0, sizeof(header));
Module* module; Module* module;
EXPECT_FALSE(ReadSymbolDataInternal(reinterpret_cast<uint8_t*>(&header), EXPECT_FALSE(ReadSymbolDataInternal(reinterpret_cast<uint8_t*>(&header),
"foo", "foo",
"", vector<string>(),
true, true,
&module)); &module));
} }
TEST_F(DumpSymbols, SimplePublic32) { TEST_F(DumpSymbols, SimplePublic32) {
@ -117,7 +117,7 @@ TEST_F(DumpSymbols, SimplePublic32) {
Module* module; Module* module;
EXPECT_TRUE(ReadSymbolDataInternal(elfdata, EXPECT_TRUE(ReadSymbolDataInternal(elfdata,
"foo", "foo",
"", vector<string>(),
true, true,
&module)); &module));
@ -156,7 +156,7 @@ TEST_F(DumpSymbols, SimplePublic64) {
Module* module; Module* module;
EXPECT_TRUE(ReadSymbolDataInternal(elfdata, EXPECT_TRUE(ReadSymbolDataInternal(elfdata,
"foo", "foo",
"", vector<string>(),
true, true,
&module)); &module));

View file

@ -32,6 +32,7 @@
#include <cstring> #include <cstring>
#include <iostream> #include <iostream>
#include <string> #include <string>
#include <vector>
#include "common/linux/dump_symbols.h" #include "common/linux/dump_symbols.h"
@ -39,7 +40,7 @@ using google_breakpad::WriteSymbolFile;
int usage(const char* self) { int usage(const char* self) {
fprintf(stderr, "Usage: %s [OPTION] <binary-with-debugging-info> " fprintf(stderr, "Usage: %s [OPTION] <binary-with-debugging-info> "
"[directory-for-debug-file]\n\n", self); "[directories-for-debug-file]\n\n", self);
fprintf(stderr, "Options:\n"); fprintf(stderr, "Options:\n");
fprintf(stderr, " -c Do not generate CFI section\n"); fprintf(stderr, " -c Do not generate CFI section\n");
return 1; return 1;
@ -50,24 +51,24 @@ int main(int argc, char **argv) {
return usage(argv[0]); return usage(argv[0]);
bool cfi = true; bool cfi = true;
if (strcmp("-c", argv[1]) == 0) int binary_index = 1;
if (strcmp("-c", argv[1]) == 0) {
cfi = false; cfi = false;
++binary_index;
}
if (!cfi && argc == 2) if (!cfi && argc == 2)
return usage(argv[0]); return usage(argv[0]);
const char *binary; const char *binary;
std::string debug_dir; std::vector<string> debug_dirs;
if (cfi) { binary = argv[binary_index];
binary = argv[1]; for (int debug_dir_index = binary_index + 1;
if (argc == 3) debug_dir_index < argc;
debug_dir = argv[2]; ++debug_dir_index) {
} else { debug_dirs.push_back(argv[debug_dir_index]);
binary = argv[2];
if (argc == 4)
debug_dir = argv[3];
} }
if (!WriteSymbolFile(binary, debug_dir, cfi, std::cout)) { if (!WriteSymbolFile(binary, debug_dirs, cfi, std::cout)) {
fprintf(stderr, "Failed to write symbol file.\n"); fprintf(stderr, "Failed to write symbol file.\n");
return 1; return 1;
} }