mingw崩溃日志分析。

This commit is contained in:
2026-03-23 16:42:28 +08:00
commit 079c27652d
12 changed files with 824 additions and 0 deletions

20
.clang-format Normal file
View File

@@ -0,0 +1,20 @@
BasedOnStyle: LLVM
IndentWidth: 4
PointerAlignment: Left
AccessModifierOffset: -4
ReflowComments: true
SpacesBeforeTrailingComments: 3
AllowShortFunctionsOnASingleLine: None
AllowShortEnumsOnASingleLine: false
BreakBeforeBraces: Custom
BraceWrapping:
AfterFunction: true
AfterClass: true
TabWidth: 4
ColumnLimit: 130
IncludeBlocks: Regroup
IncludeCategories:
- Regex: '^<.*>'
Priority: 1
- Regex: '^".*"'
Priority: 2

12
.clangd Normal file
View File

@@ -0,0 +1,12 @@
Hover:
ShowAKA: Yes
Diagnostics:
UnusedIncludes: None # 禁用未使用头文件提示
Suppress: [
anon_type_definition, # 禁用匿名的typedef提示
unused-variable, # 禁用未使用变量提示
unused-function, # 禁用未使用函数提示
unused-includes,
]
ClangTidy:
Remove: misc-unused-alias-decls

5
.gitignore vendored Normal file
View File

@@ -0,0 +1,5 @@
build/
.cache/
.qtcreator/
.vs
out/

3
.gitmodules vendored Normal file
View File

@@ -0,0 +1,3 @@
[submodule "cxxLibrary"]
path = cxxLibrary
url = https://www.sinxmiao.cn/taynpg/cxxLibrary

150
.vscode/settings.json vendored Normal file
View File

@@ -0,0 +1,150 @@
{
"files.autoSave": "onFocusChange",
"editor.fontSize": 15,
"editor.fontFamily": "'Mononoki Nerd Font Mono Regular', 'Mononoki Nerd Font Mono Regular', 'Mononoki Nerd Font Mono Regular'",
"editor.wordWrap": "on",
"terminal.integrated.fontFamily": "Mononoki Nerd Font Mono Regular",
"cmake.configureOnOpen": true,
//"C_Cpp.intelliSenseEngine": "disabled",
"cmake.debugConfig": {
"console": "externalTerminal",
"setupCommands": [
{
"description": "-gdb-set charset utf-8",
"text": "-gdb-set charset UTF-8"
},
{
"description": "Enable gdb pretty-printing",
"text": "-enable-pretty-printing",
"ignoreFailures": true
}
],
//"visualizerFile": "${workspaceRoot}/.vscode/qt6.natvis",
"args": [
"-n",
"COT1",
"-show"
]
},
"cmake.configureArgs": [
"-Wno-dev"
],
// "cmake.configureSettings": {
// "CMAKE_TOOLCHAIN_FILE": "${env:VCPKG_ROOT}/scripts/buildsystems/vcpkg.cmake"
// },
"cmake.options.statusBarVisibility": "visible",
"cmake.generator": "Ninja",
"C_Cpp.default.compileCommands": "${workspaceRoot}/build/compile_commands.json",
"C_Cpp.default.cppStandard": "c++17",
"C_Cpp.default.configurationProvider": "ms-vscode.cmake-tools",
"editor.inlayHints.enabled": "off",
"editor.unicodeHighlight.allowedLocales": {
"ja": true,
"zh-hant": true,
"zh-hans": true
},
"files.associations": {
"*.cfg": "json",
"xstring": "cpp",
"algorithm": "cpp",
"array": "cpp",
"atomic": "cpp",
"cctype": "cpp",
"charconv": "cpp",
"chrono": "cpp",
"clocale": "cpp",
"cmath": "cpp",
"condition_variable": "cpp",
"cstddef": "cpp",
"cstdint": "cpp",
"cstdio": "cpp",
"cstdlib": "cpp",
"cstring": "cpp",
"ctime": "cpp",
"cwchar": "cpp",
"exception": "cpp",
"filesystem": "cpp",
"forward_list": "cpp",
"fstream": "cpp",
"functional": "cpp",
"initializer_list": "cpp",
"iomanip": "cpp",
"ios": "cpp",
"iosfwd": "cpp",
"iostream": "cpp",
"istream": "cpp",
"iterator": "cpp",
"limits": "cpp",
"list": "cpp",
"locale": "cpp",
"map": "cpp",
"memory": "cpp",
"mutex": "cpp",
"new": "cpp",
"optional": "cpp",
"ostream": "cpp",
"ratio": "cpp",
"sstream": "cpp",
"stdexcept": "cpp",
"streambuf": "cpp",
"string": "cpp",
"string_view": "cpp",
"system_error": "cpp",
"thread": "cpp",
"tuple": "cpp",
"type_traits": "cpp",
"typeinfo": "cpp",
"unordered_map": "cpp",
"utility": "cpp",
"vector": "cpp",
"xfacet": "cpp",
"xhash": "cpp",
"xiosbase": "cpp",
"xlocale": "cpp",
"xlocbuf": "cpp",
"xlocinfo": "cpp",
"xlocmes": "cpp",
"xlocmon": "cpp",
"xlocnum": "cpp",
"xloctime": "cpp",
"xmemory": "cpp",
"xmemory0": "cpp",
"xstddef": "cpp",
"xtr1common": "cpp",
"xtree": "cpp",
"xutility": "cpp",
"bit": "cpp",
"compare": "cpp",
"concepts": "cpp",
"coroutine": "cpp",
"format": "cpp",
"stop_token": "cpp",
"bitset": "cpp",
"complex": "cpp",
"cstdarg": "cpp",
"set": "cpp",
"variant": "cpp",
"expected": "cpp",
"source_location": "cpp",
"regex": "cpp",
"*.in": "cpp",
"deque": "cpp",
"future": "cpp",
"queue": "cpp",
"resumable": "cpp",
"any": "cpp",
"codecvt": "cpp",
"csignal": "cpp",
"cwctype": "cpp",
"memory_resource": "cpp",
"numeric": "cpp",
"random": "cpp",
"shared_mutex": "cpp",
"cinttypes": "cpp",
"cfenv": "cpp",
"unordered_set": "cpp",
"ranges": "cpp",
"typeindex": "cpp"
}
}

11
CMakeLists.txt Normal file
View File

@@ -0,0 +1,11 @@
cmake_minimum_required(VERSION 3.16)
project(mingwSpecial LANGUAGES CXX)
set(CMAKE_CXX_STANDARD 17)
if (WIN32)
add_definitions(-D_WIN32_WINNT=0x0602)
endif()
add_subdirectory(cxxLibrary)
add_subdirectory(dumpDemo)

1
cxxLibrary Submodule

Submodule cxxLibrary added at 6daac723a9

8
dumpDemo/CMakeLists.txt Normal file
View File

@@ -0,0 +1,8 @@
cmake_minimum_required(VERSION 3.16)
project(dumpDemo LANGUAGES CXX)
set(CMAKE_CXX_STANDARD 17)
add_executable(dumpDemo crashdump.h crashdump.cpp main.cpp)
add_executable(dumpDemoParse dump_parse.cpp)
target_link_libraries(dumpDemoParse PRIVATE cxxLibrary)

176
dumpDemo/crashdump.cpp Normal file
View File

@@ -0,0 +1,176 @@
#include "crashdump.h"
#include <iomanip>
#include <sstream>
CrashDump::CrashDump(EXCEPTION_POINTERS* exp) : exp_(exp)
{
}
std::string CrashDump::GetModuleByReAddr(PBYTE retAddr, PBYTE& moduleAddr)
{
MODULEENTRY32 M = {sizeof(M)};
HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPMODULE, 0);
char moduleName[MAX_PATH]{};
if (hSnapshot != INVALID_HANDLE_VALUE && Module32First(hSnapshot, &M)) {
do {
if (DWORD(retAddr - M.modBaseAddr) < M.modBaseSize) {
std::snprintf(moduleName, MAX_PATH, "%s", M.szExePath);
moduleAddr = M.modBaseAddr;
break;
}
} while (Module32Next(hSnapshot, &M));
}
std::string name(moduleName);
return name;
}
std::string CrashDump::GetVersion()
{
OSVERSIONINFO V = {sizeof(OSVERSIONINFO)};
if (!GetVersionEx((POSVERSIONINFO)&V)) {
ZeroMemory(&V, sizeof(V));
V.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
GetVersionEx((POSVERSIONINFO)&V);
}
if (V.dwPlatformId != VER_PLATFORM_WIN32_NT) {
V.dwBuildNumber = LOWORD(V.dwBuildNumber);
}
char buffer[512]{};
std::snprintf(buffer, sizeof(buffer), "Windows: %ld.%ld.%ld, %s.", V.dwMajorVersion, V.dwMinorVersion, V.dwBuildNumber,
V.szCSDVersion);
std::string info(buffer);
return info;
}
std::string CrashDump::GetExceptionInfo()
{
char moduleName[MAX_PATH]{};
PBYTE moduleAddr{};
GetModuleFileName(NULL, moduleName, MAX_PATH);
if (exp_ == nullptr) {
return "Not Set Exception Pointer.";
}
std::ostringstream oss;
auto& E = exp_->ExceptionRecord;
auto& C = exp_->ContextRecord;
oss << "ExceptionAddr: " << E->ExceptionAddress << "\n";
oss << "ExceptionCode: " << E->ExceptionCode << "\n";
if (E->ExceptionCode == EXCEPTION_ACCESS_VIOLATION) {
oss << (E->ExceptionInformation[0] ? "Write " : "Read ");
oss << E->ExceptionInformation[1] << "\n";
}
oss << "Instruction:";
for (int i = 0; i < 16; i++) {
oss << " " << std::hex << std::setw(2) << std::setfill('0') << static_cast<int>(PBYTE(E->ExceptionAddress)[i]);
}
oss << std::hex << std::setfill('0');
// 1. 基本寄存器组
oss << "\nGeneral Purpose Registers:"
<< "\nRAX: " << std::setw(16) << C->Rax << " RBX: " << std::setw(16) << C->Rbx << " RCX: " << std::setw(16) << C->Rcx
<< " RDX: " << std::setw(16) << C->Rdx
<< "\nRSI: " << std::setw(16) << C->Rsi << " RDI: " << std::setw(16) << C->Rdi << " RBP: " << std::setw(16) << C->Rbp
<< " RSP: " << std::setw(16) << C->Rsp
<< "\nR8: " << std::setw(16) << C->R8 << " R9: " << std::setw(16) << C->R9 << " R10: " << std::setw(16) << C->R10
<< " R11: " << std::setw(16) << C->R11
<< "\nR12: " << std::setw(16) << C->R12 << " R13: " << std::setw(16) << C->R13 << " R14: " << std::setw(16) << C->R14
<< " R15: " << std::setw(16) << C->R15;
// 2. 关键控制寄存器
oss << "\n\nControl Registers:"
<< "\nRIP: " << std::setw(16) << C->Rip << " EFLAGS: " << std::setw(8) << C->EFlags << " CS: " << std::setw(4)
<< C->SegCs << " SS: " << std::setw(4) << C->SegSs;
// 3. 调试寄存器(如果有用)
if (C->ContextFlags & CONTEXT_DEBUG_REGISTERS) {
oss << "\n\nDebug Registers:"
<< "\nDR0: " << std::setw(16) << C->Dr0 << " DR1: " << std::setw(16) << C->Dr1 << " DR2: " << std::setw(16)
<< C->Dr2 << " DR3: " << std::setw(16) << C->Dr3 << "\nDR6: " << std::setw(16) << C->Dr6
<< " DR7: " << std::setw(16) << C->Dr7;
}
// 4. MXCSR状态(浮点/SSE)
if (C->ContextFlags & CONTEXT_FLOATING_POINT) {
oss << "\n\nFPU/SSE State:"
<< "\nMXCSR: " << std::setw(8) << C->MxCsr;
}
// 5. 最后分支记录(如果可用)
if (C->LastBranchToRip || C->LastBranchFromRip) {
oss << "\n\nLast Branch:"
<< "\nFrom: " << std::setw(16) << C->LastBranchFromRip << " To: " << std::setw(16) << C->LastBranchToRip;
}
oss << GetCallStack(exp_) << "\n";
return oss.str();
}
std::string CrashDump::GetCallStack(EXCEPTION_POINTERS* exp)
{
struct StackFrame {
StackFrame* prev_frame;
void* return_address;
};
std::ostringstream oss;
StackFrame* current_frame = nullptr;
// 1. 初始化栈帧指针
if (exp) {
// 异常情况:从异常上下文中获取栈帧
#if defined(_M_X64) || defined(__x86_64__)
current_frame = reinterpret_cast<StackFrame*>(exp->ContextRecord->Rbp);
#else
current_frame = reinterpret_cast<StackFrame*>(exp->ContextRecord->Ebp);
#endif
} else {
// 非异常情况:手动获取当前栈帧(编译器相关)
#if defined(__GNUC__) || defined(__clang__)
// GCC/Clang 内联汇编获取 EBP/RBP
#if defined(__x86_64__)
register void* frame_ptr asm("rbp");
#else
register void* frame_ptr asm("ebp");
#endif
current_frame = reinterpret_cast<StackFrame*>(frame_ptr);
#elif defined(_MSC_VER)
// MSVC 内置函数(如果可用)
current_frame = reinterpret_cast<StackFrame*>(_AddressOfReturnAddress() - sizeof(void*));
#else
return "Stack walking not supported on this compiler";
#endif
}
// 2. 遍历调用栈
constexpr size_t max_frames = 64;
PBYTE module_addr = nullptr;
for (size_t i = 0; i < max_frames && current_frame; ++i) {
// 安全检查
if (IsBadReadPtr(current_frame, sizeof(StackFrame)) ||
IsBadCodePtr(reinterpret_cast<FARPROC>(current_frame->return_address))) {
break;
}
// 格式化输出
oss << "\n0x" << std::hex << std::setw(8) << std::setfill('0')
<< reinterpret_cast<uintptr_t>(current_frame->return_address) << " ";
// 获取模块信息
std::string module_name = GetModuleByReAddr(reinterpret_cast<PBYTE>(current_frame->return_address), module_addr);
if (!module_name.empty()) {
DWORD rva = static_cast<DWORD>(reinterpret_cast<PBYTE>(current_frame->return_address) - module_addr);
// 输出模块信息和RVA
oss << " (RVA: 0x" << std::hex << std::setw(8) << std::setfill('0') << rva << ") " << module_name;
}
// 移动到上一栈帧
current_frame = current_frame->prev_frame;
}
return oss.str();
}

25
dumpDemo/crashdump.h Normal file
View File

@@ -0,0 +1,25 @@
#ifndef CRASHDUMP_H
#define CRASHDUMP_H
#include <comdef.h>
#include <string>
#include <tlhelp32.h>
#include <wbemidl.h>
#include <windows.h>
class CrashDump
{
public:
CrashDump(EXCEPTION_POINTERS* exp);
public:
std::string GetModuleByReAddr(PBYTE retAddr, PBYTE& moduleAddr);
std::string GetVersion();
std::string GetExceptionInfo();
std::string GetCallStack(EXCEPTION_POINTERS* exp);
private:
EXCEPTION_POINTERS* exp_{};
};
#endif // CRASHDUMP_H

338
dumpDemo/dump_parse.cpp Normal file
View File

@@ -0,0 +1,338 @@
#define WIN32_LEAN_AND_MEAN
#include <boost/asio.hpp>
#include <boost/filesystem.hpp>
#include <boost/nowide/args.hpp>
#include <boost/nowide/filesystem.hpp>
#include <boost/nowide/fstream.hpp>
#include <boost/nowide/iostream.hpp>
#include <boost/process.hpp>
#include <cstdint>
#include <iostream>
#include <regex>
#include <string>
#include <vector>
#define COLOR_RESET "\033[0m"
#define COLOR_RED "\033[31m"
#define COLOR_GREEN "\033[32m"
#define COLOR_YELLOW "\033[33m"
#define COLOR_BLUE "\033[34m"
#define COLOR_MAGENTA "\033[35m"
#define COLOR_CYAN "\033[36m"
bool is_utf8(const std::string& data)
{
for (size_t i = 0; i < data.size();) {
uint8_t c = static_cast<uint8_t>(data[i]);
if (c <= 0x7F) { // ASCII
i++;
} else if ((c & 0xE0) == 0xC0) { // 2字节
if (i + 1 >= data.size() || (data[i + 1] & 0xC0) != 0x80)
return false;
i += 2;
} else if ((c & 0xF0) == 0xE0) { // 3字节
if (i + 2 >= data.size() || (data[i + 1] & 0xC0) != 0x80 || (data[i + 2] & 0xC0) != 0x80)
return false;
i += 3;
} else if ((c & 0xF8) == 0xF0) { // 4字节
if (i + 3 >= data.size() || (data[i + 1] & 0xC0) != 0x80 || (data[i + 2] & 0xC0) != 0x80 ||
(data[i + 3] & 0xC0) != 0x80)
return false;
i += 4;
} else {
return false; // 非法UTF-8序列
}
}
return true;
}
std::string to_u8(const std::string& str)
{
if (str.empty()) {
return "";
}
// 1. ANSI (当前代码页) → UTF-16 (WideChar)
int wide_len = MultiByteToWideChar(CP_ACP, 0, str.c_str(), -1, nullptr, 0);
if (wide_len == 0) {
return "";
}
std::wstring wide_str(wide_len, L'\0');
MultiByteToWideChar(CP_ACP, 0, str.c_str(), -1, &wide_str[0], wide_len);
// 2. UTF-16 → UTF-8
int utf8_len = WideCharToMultiByte(CP_UTF8, 0, wide_str.c_str(), -1, nullptr, 0, nullptr, nullptr);
if (utf8_len == 0) {
return "";
}
std::string utf8_str(utf8_len, '\0');
WideCharToMultiByte(CP_UTF8, 0, wide_str.c_str(), -1, &utf8_str[0], utf8_len, nullptr, nullptr);
// 移除末尾的\0
if (!utf8_str.empty() && utf8_str.back() == '\0') {
utf8_str.pop_back();
}
return utf8_str;
}
void enable_ansi_color()
{
#if defined(_WIN32)
HANDLE hConsole = GetStdHandle(STD_OUTPUT_HANDLE);
DWORD mode = 0;
GetConsoleMode(hConsole, &mode);
mode |= ENABLE_VIRTUAL_TERMINAL_PROCESSING;
SetConsoleMode(hConsole, mode);
#endif
}
void PrintSource(const std::string& str)
{
std::string location = str;
size_t colon_pos = location.find_last_of(':');
if (colon_pos == std::string::npos) {
return; // 格式不符
}
std::string key("at ");
auto sp = location.find(key);
std::string file_path = location.substr(sp + key.size(), colon_pos - sp - key.size());
int line_number = std::stoi(location.substr(colon_pos + 1));
// 读取源文件并提取对应行
int upper = line_number + 5;
int lower = line_number - 5;
lower = lower < 1 ? 1 : lower;
boost::nowide::ifstream source_file(file_path);
if (!source_file) {
boost::nowide::cerr << "无法打开源文件: " << file_path << std::endl;
return;
}
std::string line_content;
int current_line = 0;
while (std::getline(source_file, line_content)) {
current_line++;
if (current_line >= lower && current_line <= upper) {
if (current_line == line_number) {
boost::nowide::cout << COLOR_YELLOW << current_line << ":" << line_content << COLOR_RESET << std::endl;
} else {
boost::nowide::cout << current_line << ":" << line_content << std::endl;
}
}
if (current_line > upper) {
break;
}
}
}
std::pair<int, std::string> exe_cmd(const std::string& exe, const std::vector<std::string>& args)
{
boost::asio::io_context ctx;
auto bin_path = boost::process::environment::find_executable(exe);
if (bin_path.empty()) {
return {-1, "Error: Command not found: " + exe};
}
boost::asio::readable_pipe rp_out(ctx), rp_err(ctx);
std::string out, err;
try {
// 启动进程
boost::process::process proc =
boost::process::process(ctx, bin_path, args, boost::process::process_stdio{{}, rp_out, rp_err});
// 同步读取(可能阻塞)
boost::system::error_code ec_out, ec_err;
boost::asio::read(rp_out, boost::asio::dynamic_buffer(out), ec_out);
boost::asio::read(rp_err, boost::asio::dynamic_buffer(err), ec_err);
proc.wait();
return {proc.exit_code(), out + (err.empty() ? "" : "\n" + err)};
} catch (const std::exception& e) {
return {-1, std::string("Exception: ") + e.what()};
} catch (...) {
return {-1, "Unknown exception"};
}
}
class CrashAnalyzer
{
public:
CrashAnalyzer(const std::string& logPath, const std::string& exePath) : m_logPath(logPath), m_exePath(exePath)
{
}
bool Analyze()
{
if (!CheckTools())
return false;
if (!ParseLog())
return false;
return ResolveSymbols();
}
private:
struct StackFrame {
uint64_t address;
uint64_t rva;
std::string module;
};
bool CheckTools()
{
auto check = [](const std::string& cmd) {
auto r = exe_cmd(cmd, {"--version"});
return r.first == 0;
};
if (!check("addr2line")) {
boost::nowide::cerr << "错误:未找到 addr2line 工具\n";
return false;
}
if (!check("objdump")) {
boost::nowide::cerr << "错误:未找到 objdump 工具\n";
return false;
}
return true;
}
bool ParseLog()
{
boost::nowide::ifstream file(m_logPath);
if (!file) {
boost::nowide::cerr << "无法打开日志文件: " << m_logPath << "\n";
return false;
}
std::regex addrRegex(R"(0x([0-9a-fA-F]+)\s+\(RVA:\s+(0x[0-9a-fA-F]+)\)\s+(.+))");
std::string line;
while (getline(file, line)) {
if (line.find("ExceptionAddr:") != std::string::npos) {
sscanf(line.c_str(), "ExceptionAddr: %llx", &m_exceptionAddr);
}
std::smatch match;
if (regex_match(line, match, addrRegex)) {
StackFrame frame;
frame.address = std::stoull(match[1].str(), nullptr, 16);
frame.rva = std::stoull(match[2].str(), nullptr, 16);
frame.module = match[3].str();
if (!is_utf8(frame.module)) {
frame.module = to_u8(frame.module);
}
m_stackFrames.push_back(frame);
}
}
if (m_stackFrames.empty()) {
boost::nowide::cerr << "日志中未找到有效的调用栈信息\n";
return false;
}
// else {
// int i = 0;
// for (const auto& f : m_stackFrames) {
// ++i;
// boost::nowide::cout << i << "PARSEADDR:" << f.address << "\n";
// }
// }
// 计算运行时基址(取第一个栈帧作为基准)
m_runtimeBase = m_stackFrames[0].address - m_stackFrames[0].rva;
return true;
}
bool ResolveSymbols()
{
// 获取编译时基址
auto result = exe_cmd("objdump", {"-p", m_exePath});
if (result.first != 0) {
boost::nowide::cerr << "objdump 执行失败: " << result.second << "\n";
return false;
}
uint64_t imageBase = 0;
std::istringstream iss(result.second);
std::string line;
while (getline(iss, line)) {
if (line.find("ImageBase") != std::string::npos) {
sscanf(line.c_str(), " ImageBase %llx", &imageBase);
break;
}
}
if (imageBase == 0) {
boost::nowide::cerr << "无法获取可执行文件的ImageBase\n";
return false;
}
// 解析异常地址
uint64_t exceptionRva = m_exceptionAddr - m_runtimeBase;
uint64_t compileAddr = imageBase + exceptionRva;
PrintSourceInfo("\n============================= 异常位置 ===============================\n", m_exePath, compileAddr);
// 解析调用栈
int l = 0;
for (const auto& frame : m_stackFrames) {
++l;
uint64_t compileAddr = imageBase + frame.rva;
PrintSourceInfo("调用栈" + std::to_string(l) + " ", frame.module, compileAddr);
}
return true;
}
void PrintSourceInfo(const std::string& type, const std::string& module, uint64_t addr)
{
// 将 addr 转为全小写 16 进制字符串
std::ostringstream oss;
oss << std::hex << std::nouppercase << addr; // 无前缀、全小写
std::string addr_hex = oss.str();
// 调用 addr2line
// boost::nowide::cout << "READY:" << module << "|" << addr_hex << std::endl;
auto result = exe_cmd("addr2line", {"-e", module, "-f", "-C", "-p", addr_hex});
if (result.first == 0) {
boost::nowide::cout << COLOR_CYAN << type << COLOR_RESET << "[" << addr_hex << "]: " << result.second;
boost::nowide::cout << COLOR_YELLOW << "--> " << module << COLOR_RESET << std::endl;
}
if (type.find("=") != std::string::npos) {
PrintSource(result.second);
boost::nowide::cout << COLOR_CYAN << "======================================================================\n"
<< COLOR_RESET << std::endl;
}
}
std::string m_logPath;
std::string m_exePath;
uint64_t m_exceptionAddr = 0;
uint64_t m_runtimeBase = 0;
std::vector<StackFrame> m_stackFrames;
};
int main(int argc, char* argv[])
{
boost::nowide::args args(argc, argv);
boost::nowide::nowide_filesystem();
enable_ansi_color();
if (argc != 3) {
boost::nowide::cerr << "用法: " << argv[0] << " <崩溃日志文件> <可执行文件路径>\n";
return 1;
}
CrashAnalyzer analyzer(argv[1], argv[2]);
if (!analyzer.Analyze()) {
return 1;
}
return 0;
}

75
dumpDemo/main.cpp Normal file
View File

@@ -0,0 +1,75 @@
#include <chrono>
#include <ctime>
#include <fstream>
#include <iomanip>
#include <iostream>
#include <sstream>
#include <thread>
#include "crashdump.h"
long __stdcall DumpCall(EXCEPTION_POINTERS* excp)
{
// 1. 生成带时间戳的文件名
auto now = std::chrono::system_clock::now();
auto now_time_t = std::chrono::system_clock::to_time_t(now);
auto now_ms = std::chrono::duration_cast<std::chrono::milliseconds>(now.time_since_epoch()) % 1000;
std::ostringstream oss;
oss << std::put_time(std::localtime(&now_time_t), "%Y%m%d_%H%M%S") << "_" << std::setfill('0') << std::setw(3)
<< now_ms.count() << "_crash.log";
std::string filename = oss.str();
// 2. 获取崩溃信息
CrashDump dump(excp);
std::string crashInfo = dump.GetExceptionInfo();
// 3. 写入文件 (C++11 风格)
try {
std::ofstream outfile(filename, std::ios::out | std::ios::trunc);
if (outfile) {
outfile << "=== Crash Dump ===\n\n";
outfile << "Timestamp: " << std::put_time(std::localtime(&now_time_t), "%Y-%m-%d %H:%M:%S") << "."
<< std::setfill('0') << std::setw(3) << now_ms.count() << "\n";
outfile << crashInfo;
outfile << "\n=== End of Dump ===";
// 确保数据写入磁盘
outfile.flush();
if (outfile.good()) {
return EXCEPTION_EXECUTE_HANDLER;
}
}
} catch (...) {
// 文件写入失败的备用方案
OutputDebugStringA(("Failed to write crash dump to " + filename).c_str());
}
return EXCEPTION_EXECUTE_HANDLER;
}
void tasks()
{
int c = 5;
while (--c > 0) {
std::this_thread::sleep_for(std::chrono::seconds(1));
}
int* p = nullptr;
*p = 11;
std::cout << "over..." << std::endl;
}
int main()
{
SetUnhandledExceptionFilter(DumpCall);
std::cout << "Hello World!" << std::endl;
std::thread thread(tasks);
thread.join();
std::cout << "End." << std::endl;
return 0;
}