aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/utils/addr2name.cpp
blob: 333ef2994488aa23cb3b02b0d5ab0df2cd3cb6f5 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
// Copyright (c) 2017 Egor Tensin <Egor.Tensin@gmail.com>
// This file is part of the "winapi-debug" project.
// For details, see https://github.com/egor-tensin/winapi-debug.
// Distributed under the MIT License.

#include "command_line.hpp"
#include "pdb_descr.hpp"

#include <winapi/debug.hpp>

#include <boost/program_options.hpp>

#include <exception>
#include <iostream>
#include <sstream>
#include <string>
#include <vector>

namespace {

class Addr2Name : public SettingsParser {
public:
    explicit Addr2Name(int argc, char** argv) : SettingsParser{argc, argv} {
        namespace po = boost::program_options;

        visible.add_options()(
            "pdb", po::value<std::vector<PDB>>(&pdbs)->value_name("ADDR,PATH"), "load a PDB file");
        visible.add_options()(
            "lines,l", po::bool_switch(&lines), "try to resolve source files & line numbers");
        hidden.add_options()(
            "address",
            po::value<std::vector<winapi::Address>>(&addresses)->value_name("ADDR"),
            "add an address to resolve");
        positional.add("address", -1);
    }

    const char* get_short_description() const override {
        return "[-h|--help] [--pdb ADDR,PATH]... [-l|--lines] [--] [ADDR]...";
    }

    std::vector<PDB> pdbs;
    std::vector<winapi::Address> addresses;
    bool lines = false;
};

std::string format_symbol(const winapi::Module& module, const winapi::Symbol& symbol) {
    std::ostringstream oss;
    oss << module.get_name() << '!' << symbol.get_name();
    const auto displacement = symbol.get_displacement();
    if (displacement)
        oss << '+' << winapi::address::format(displacement);
    return oss.str();
}

std::string format_line_info(const winapi::LineInfo& line_info) {
    std::ostringstream oss;
    oss << '[' << line_info.file_path << " @ " << line_info.line_number << ']';
    return oss.str();
}

void dump_error(const std::exception& e) {
    std::cerr << "error: " << e.what() << '\n';
}

void resolve_symbol(const winapi::PostMortem& analysis,
                    winapi::Address address,
                    bool lines = false) {
    try {
        const auto symbol = analysis.resolve_symbol(address);
        const auto& module = analysis.module_with_offline_base(symbol.get_offline_base());

        std::ostringstream msg;
        msg << format_symbol(module, symbol);

        if (lines) {
            try {
                const auto line_info = analysis.resolve_line(address);
                msg << ' ' << format_line_info(line_info);
            } catch (const std::exception& e) {
                dump_error(e);
            }
        }

        std::cout << msg.str() << '\n';
    } catch (const std::exception& e) {
        dump_error(e);
        std::cout << winapi::address::format(address) << '\n';
    }
}

} // namespace

int main(int argc, char* argv[]) {
    try {
        Addr2Name settings{argc, argv};

        try {
            settings.parse(argc, argv);
        } catch (const boost::program_options::error& e) {
            settings.usage_error(e);
            return 1;
        }

        if (settings.exit_with_usage) {
            settings.usage();
            return 0;
        }

        winapi::PostMortem analysis;

        for (const auto& pdb : settings.pdbs)
            analysis.add_pdb(pdb.online_base, pdb.path);

        for (const auto& address : settings.addresses)
            resolve_symbol(analysis, address, settings.lines);
    } catch (const std::exception& e) {
        dump_error(e);
        return 1;
    }
    return 0;
}