aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/src/post_mortem.cpp
diff options
context:
space:
mode:
authorEgor Tensin <Egor.Tensin@gmail.com>2021-05-29 01:24:29 +0300
committerEgor Tensin <Egor.Tensin@gmail.com>2021-05-29 01:27:53 +0300
commite162f2a21632e3f7b1c7ab9dd7b530beb1f2d4a8 (patch)
tree65199bce3779ff1800821c6b1c6bc0bc4592e66c /src/post_mortem.cpp
parentnamespace pdb -> namespace winapi (diff)
downloadwinapi-debug-e162f2a21632e3f7b1c7ab9dd7b530beb1f2d4a8.tar.gz
winapi-debug-e162f2a21632e3f7b1c7ab9dd7b530beb1f2d4a8.zip
rename class Repo to PostMortem
Diffstat (limited to 'src/post_mortem.cpp')
-rw-r--r--src/post_mortem.cpp171
1 files changed, 171 insertions, 0 deletions
diff --git a/src/post_mortem.cpp b/src/post_mortem.cpp
new file mode 100644
index 0000000..d365b02
--- /dev/null
+++ b/src/post_mortem.cpp
@@ -0,0 +1,171 @@
+// 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 <winapi/debug.hpp>
+#include <winapi/file.hpp>
+
+#include <map>
+#include <sstream>
+#include <stdexcept>
+#include <string>
+#include <utility>
+
+namespace winapi {
+namespace {
+
+std::string pdb_already_loaded(Address online_base, const std::string& path) {
+ std::ostringstream oss;
+ oss << "module with online base address " << format_address(online_base)
+ << " has already been loaded: " << path;
+ return oss.str();
+}
+
+std::string pdb_already_loaded(const std::string& path) {
+ std::ostringstream oss;
+ oss << "module has already been loaded: " << path;
+ return oss.str();
+}
+
+std::string offline_base_already_used(Address base) {
+ std::ostringstream oss;
+ oss << "module with offline base address " << format_address(base)
+ << " has already been loaded (shouldn't happen)";
+ return oss.str();
+}
+
+std::string module_not_found(Address base) {
+ std::ostringstream oss;
+ oss << "module with base address " << format_address(base) << " wasn't found";
+ return oss.str();
+}
+
+std::string guess_module_no_modules(Address pivot) {
+ std::ostringstream oss;
+ oss << "couldn't select a module for address " << format_address(pivot)
+ << ": no modules have been loaded yet";
+ return oss.str();
+}
+
+std::string guess_module_address_too_low(Address pivot) {
+ std::ostringstream oss;
+ oss << "couldn't select a module for address " << format_address(pivot) << ": it's too low";
+ return oss.str();
+}
+
+template <typename Value>
+const Module& lookup_module(const std::map<Address, Value>& modules, Address base) {
+ const auto it = modules.find(base);
+ if (it == modules.cend())
+ throw std::runtime_error{module_not_found(base)};
+ return it->second;
+}
+
+template <typename Value>
+const Module& guess_module(const std::map<Address, Value>& modules, Address pivot) {
+ if (modules.empty())
+ throw std::range_error{guess_module_no_modules(pivot)};
+
+ auto it = modules.lower_bound(pivot);
+
+ if (it == modules.cend()) {
+ --it;
+ return it->second;
+ }
+
+ if (it->first > pivot) {
+ if (it == modules.cbegin())
+ throw std::range_error{guess_module_address_too_low(pivot)};
+ --it;
+ return it->second;
+ }
+
+ return it->second;
+}
+
+} // namespace
+
+Address PostMortem::add_pdb(Address online_base, const std::string& path) {
+ if (online_bases.find(online_base) != online_bases.cend())
+ throw std::runtime_error{pdb_already_loaded(online_base, path)};
+
+ auto file_id = winapi::File::open_read_attributes(path).query_id();
+ if (file_ids.find(file_id) != file_ids.cend())
+ throw std::runtime_error{pdb_already_loaded(path)};
+
+ Module module{online_base, dbghelp.load_pdb(path)};
+ const auto offline_base = module.get_offline_base();
+
+ if (offline_bases.find(offline_base) != offline_bases.cend())
+ throw std::runtime_error{offline_base_already_used(offline_base)};
+
+ file_ids.emplace(std::move(file_id));
+ const auto it = online_bases.emplace(online_base, std::move(module));
+ offline_bases.emplace(offline_base, it.first->second);
+
+ return offline_base;
+}
+
+void PostMortem::enum_symbols(const OnSymbol& callback) const {
+ for (const auto& it : offline_bases)
+ enum_symbols(it.second, callback);
+}
+
+void PostMortem::enum_symbols(Address offline_base, const OnSymbol& callback) const {
+ const auto it = offline_bases.find(offline_base);
+ if (it == offline_bases.cend())
+ throw std::runtime_error{"unknown module"};
+ enum_symbols(it->second, callback);
+}
+
+void PostMortem::enum_symbols(const Module& module, const OnSymbol& callback) const {
+ dbghelp.enum_symbols(
+ module, [&](const SymbolInfo& impl) { callback(symbol_from_buffer(module, impl)); });
+}
+
+Symbol PostMortem::resolve_symbol(Address online) const {
+ return symbol_from_buffer(dbghelp.resolve_symbol(address_online_to_offline(online)));
+}
+
+Symbol PostMortem::resolve_symbol(const std::string& name) const {
+ return symbol_from_buffer(dbghelp.resolve_symbol(name));
+}
+
+LineInfo PostMortem::resolve_line(Address online) const {
+ return dbghelp.resolve_line(address_online_to_offline(online));
+}
+
+const Module& PostMortem::module_with_online_base(Address base) const {
+ return lookup_module(online_bases, base);
+}
+
+const Module& PostMortem::module_with_offline_base(Address base) const {
+ return lookup_module(offline_bases, base);
+}
+
+Symbol PostMortem::symbol_from_buffer(const SymbolInfo& impl) const {
+ return symbol_from_buffer(module_with_offline_base(impl.get_offline_base()), impl);
+}
+
+Symbol PostMortem::symbol_from_buffer(const Module& module, const SymbolInfo& impl) {
+ return {module.translate_offline_address(impl.get_offline_address()), impl};
+}
+
+Address PostMortem::address_online_to_offline(Address online) const {
+ return module_from_online_address(online).translate_online_address(online);
+}
+
+Address PostMortem::address_offline_to_online(Address offline) const {
+ return module_from_offline_address(offline).translate_offline_address(offline);
+}
+
+const Module& PostMortem::module_from_online_address(Address online) const {
+ return guess_module(online_bases, online);
+}
+
+const Module& PostMortem::module_from_offline_address(Address offline) const {
+ return guess_module(offline_bases, offline);
+}
+
+} // namespace winapi