2006-08-25 23:14:45 +02:00
|
|
|
// Copyright (C) 2006 Google Inc.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <map>
|
|
|
|
#include <string.h>
|
|
|
|
#include <vector>
|
|
|
|
#include <utility>
|
2006-09-05 21:42:57 +02:00
|
|
|
#include "processor/source_line_resolver.h"
|
2006-09-07 19:26:17 +02:00
|
|
|
#include "google/stack_frame.h"
|
2006-09-08 20:03:56 +02:00
|
|
|
#include "processor/linked_ptr.h"
|
|
|
|
#include "processor/range_map.h"
|
2006-08-25 23:14:45 +02:00
|
|
|
|
2006-08-30 22:05:05 +02:00
|
|
|
using std::map;
|
|
|
|
using std::vector;
|
|
|
|
using std::make_pair;
|
2006-08-25 23:14:45 +02:00
|
|
|
using __gnu_cxx::hash;
|
|
|
|
|
2006-08-30 22:05:05 +02:00
|
|
|
namespace google_airbag {
|
2006-08-25 23:14:45 +02:00
|
|
|
|
|
|
|
struct SourceLineResolver::Line {
|
2006-09-08 20:03:56 +02:00
|
|
|
Line(MemAddr addr, MemAddr code_size, int file_id, int source_line)
|
|
|
|
: address(addr)
|
|
|
|
, size(code_size)
|
|
|
|
, source_file_id(file_id)
|
|
|
|
, line(source_line) { }
|
2006-08-25 23:14:45 +02:00
|
|
|
|
|
|
|
MemAddr address;
|
2006-09-08 20:03:56 +02:00
|
|
|
MemAddr size;
|
2006-08-25 23:14:45 +02:00
|
|
|
int source_file_id;
|
|
|
|
int line;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct SourceLineResolver::Function {
|
2006-09-08 20:03:56 +02:00
|
|
|
Function(const string &function_name,
|
|
|
|
MemAddr function_address,
|
|
|
|
MemAddr code_size)
|
|
|
|
: name(function_name), address(function_address), size(code_size) { }
|
2006-08-25 23:14:45 +02:00
|
|
|
|
|
|
|
string name;
|
|
|
|
MemAddr address;
|
2006-09-08 20:03:56 +02:00
|
|
|
MemAddr size;
|
|
|
|
RangeMap<MemAddr, linked_ptr<Line> > lines;
|
2006-08-25 23:14:45 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class SourceLineResolver::Module {
|
|
|
|
public:
|
|
|
|
Module(const string &name) : name_(name) { }
|
|
|
|
|
|
|
|
// Loads the given map file, returning true on success.
|
|
|
|
bool LoadMap(const string &map_file);
|
|
|
|
|
2006-09-07 19:26:17 +02:00
|
|
|
// Looks up the given relative address, and fills the StackFrame struct
|
2006-08-25 23:14:45 +02:00
|
|
|
// with the result.
|
2006-09-07 19:26:17 +02:00
|
|
|
void LookupAddress(MemAddr address, StackFrame *frame) const;
|
2006-08-25 23:14:45 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
friend class SourceLineResolver;
|
|
|
|
typedef hash_map<int, string> FileMap;
|
|
|
|
|
|
|
|
// Parses a file declaration
|
|
|
|
void ParseFile(char *file_line);
|
|
|
|
|
|
|
|
// Parses a function declaration, returning a new Function object.
|
|
|
|
Function* ParseFunction(char *function_line);
|
|
|
|
|
|
|
|
// Parses a line declaration, returning a new Line object.
|
|
|
|
Line* ParseLine(char *line_line);
|
|
|
|
|
|
|
|
string name_;
|
|
|
|
FileMap files_;
|
2006-09-08 20:03:56 +02:00
|
|
|
RangeMap<MemAddr, linked_ptr<Function> > functions_;
|
2006-08-25 23:14:45 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
SourceLineResolver::SourceLineResolver() : modules_(new ModuleMap) {
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceLineResolver::~SourceLineResolver() {
|
|
|
|
ModuleMap::iterator it;
|
|
|
|
for (it = modules_->begin(); it != modules_->end(); ++it) {
|
|
|
|
delete it->second;
|
|
|
|
}
|
|
|
|
delete modules_;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SourceLineResolver::LoadModule(const string &module_name,
|
|
|
|
const string &map_file) {
|
|
|
|
// Make sure we don't already have a module with the given name.
|
|
|
|
if (modules_->find(module_name) != modules_->end()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Module *module = new Module(module_name);
|
|
|
|
if (!module->LoadMap(map_file)) {
|
|
|
|
delete module;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
modules_->insert(make_pair(module_name, module));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2006-09-07 19:26:17 +02:00
|
|
|
void SourceLineResolver::FillSourceLineInfo(StackFrame *frame) const {
|
|
|
|
ModuleMap::const_iterator it = modules_->find(frame->module_name);
|
2006-08-25 23:14:45 +02:00
|
|
|
if (it != modules_->end()) {
|
2006-09-08 04:35:53 +02:00
|
|
|
it->second->LookupAddress(frame->instruction - frame->module_base, frame);
|
2006-08-25 23:14:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SourceLineResolver::Module::LoadMap(const string &map_file) {
|
|
|
|
FILE *f = fopen(map_file.c_str(), "r");
|
|
|
|
if (!f) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
char buffer[1024];
|
|
|
|
Function *cur_func = NULL;
|
|
|
|
|
|
|
|
while (fgets(buffer, sizeof(buffer), f)) {
|
|
|
|
if (strncmp(buffer, "FILE ", 5) == 0) {
|
|
|
|
ParseFile(buffer);
|
|
|
|
} else if (strncmp(buffer, "FUNC ", 5) == 0) {
|
|
|
|
cur_func = ParseFunction(buffer);
|
|
|
|
if (!cur_func) {
|
|
|
|
return false;
|
|
|
|
}
|
2006-09-08 20:03:56 +02:00
|
|
|
functions_.StoreRange(cur_func->address, cur_func->size,
|
|
|
|
linked_ptr<Function>(cur_func));
|
2006-08-25 23:14:45 +02:00
|
|
|
} else {
|
|
|
|
if (!cur_func) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Line *line = ParseLine(buffer);
|
|
|
|
if (!line) {
|
|
|
|
return false;
|
|
|
|
}
|
2006-09-08 20:03:56 +02:00
|
|
|
cur_func->lines.StoreRange(line->address, line->size,
|
|
|
|
linked_ptr<Line>(line));
|
2006-08-25 23:14:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(f);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SourceLineResolver::Module::LookupAddress(MemAddr address,
|
2006-09-07 19:26:17 +02:00
|
|
|
StackFrame *frame) const {
|
2006-09-08 20:03:56 +02:00
|
|
|
linked_ptr<Function> func;
|
|
|
|
if (!functions_.RetrieveRange(address, &func)) {
|
2006-08-25 23:14:45 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-09-07 19:26:17 +02:00
|
|
|
frame->function_name = func->name;
|
2006-09-08 20:03:56 +02:00
|
|
|
linked_ptr<Line> line;
|
|
|
|
if (!func->lines.RetrieveRange(address, &line)) {
|
2006-08-25 23:14:45 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
FileMap::const_iterator it = files_.find(line->source_file_id);
|
|
|
|
if (it != files_.end()) {
|
2006-09-07 19:26:17 +02:00
|
|
|
frame->source_file_name = files_.find(line->source_file_id)->second;
|
2006-08-25 23:14:45 +02:00
|
|
|
}
|
2006-09-07 19:26:17 +02:00
|
|
|
frame->source_line = line->line;
|
2006-08-25 23:14:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SourceLineResolver::Module::ParseFile(char *file_line) {
|
|
|
|
// FILE <id> <filename>
|
|
|
|
file_line += 5; // skip prefix
|
|
|
|
char *id = strtok(file_line, " ");
|
|
|
|
if (!id) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int index = atoi(id);
|
|
|
|
if (index < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *filename = strtok(NULL, "\r\n");
|
|
|
|
if (filename) {
|
|
|
|
files_.insert(make_pair(index, string(filename)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceLineResolver::Function* SourceLineResolver::Module::ParseFunction(
|
|
|
|
char *function_line) {
|
|
|
|
// FUNC <address> <name>
|
|
|
|
function_line += 5; // skip prefix
|
|
|
|
char *addr = strtok(function_line, " ");
|
|
|
|
if (!addr) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2006-09-08 20:03:56 +02:00
|
|
|
char *size = strtok(NULL, " ");
|
|
|
|
if (!size) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2006-08-25 23:14:45 +02:00
|
|
|
char *name = strtok(NULL, "\r\n");
|
|
|
|
if (!name) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2006-09-08 20:03:56 +02:00
|
|
|
return new Function(name, strtoull(addr, NULL, 16), strtoull(size, NULL, 16));
|
2006-08-25 23:14:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
SourceLineResolver::Line* SourceLineResolver::Module::ParseLine(
|
|
|
|
char *line_line) {
|
|
|
|
// <address> <line number> <source file id>
|
|
|
|
char *addr = strtok(line_line, " ");
|
|
|
|
if (!addr) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2006-09-08 20:03:56 +02:00
|
|
|
char *size = strtok(NULL, " ");
|
|
|
|
if (!size) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2006-08-25 23:14:45 +02:00
|
|
|
char *line_num_str = strtok(NULL, "\r\n");
|
|
|
|
if (!line_num_str) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int line_number, source_file;
|
|
|
|
if (sscanf(line_num_str, "%d %d", &line_number, &source_file) != 2) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2006-09-08 20:03:56 +02:00
|
|
|
return new Line(strtoull(addr, NULL, 16),
|
|
|
|
strtoull(size, NULL, 16),
|
|
|
|
source_file,
|
|
|
|
line_number);
|
2006-08-25 23:14:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t SourceLineResolver::HashString::operator()(const string &s) const {
|
|
|
|
return hash<const char*>()(s.c_str());
|
|
|
|
}
|
|
|
|
|
2006-08-30 22:05:05 +02:00
|
|
|
} // namespace google_airbag
|