mirror of https://github.com/procxx/kepka.git
improved os x crash reports, possible linux (and windows) broken
This commit is contained in:
parent
d28483fad4
commit
74248b0284
|
@ -546,6 +546,21 @@ void _moveOldDataFiles(const QString &wasDir) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if defined Q_OS_MAC || defined Q_OS_LINUX32 || defined Q_OS_LINUX64
|
||||||
|
#include <execinfo.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <sys/syscall.h>
|
||||||
|
|
||||||
|
#ifdef Q_OS_MAC
|
||||||
|
|
||||||
|
#include <dlfcn.h>
|
||||||
|
#include <mach-o/dyld_images.h>
|
||||||
|
#include <mach-o/dyld.h>
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace SignalHandlers {
|
namespace SignalHandlers {
|
||||||
|
|
||||||
QByteArray CrashDumpPath;
|
QByteArray CrashDumpPath;
|
||||||
|
@ -629,7 +644,17 @@ namespace SignalHandlers {
|
||||||
bool LoggingCrashHeaderWritten = false;
|
bool LoggingCrashHeaderWritten = false;
|
||||||
QMutex LoggingCrashMutex;
|
QMutex LoggingCrashMutex;
|
||||||
|
|
||||||
|
// see https://github.com/benbjohnson/bandicoot
|
||||||
|
#if defined Q_OS_MAC || defined Q_OS_LINUX32 || defined Q_OS_LINUX64
|
||||||
|
struct sigaction SIG_def[32];
|
||||||
|
|
||||||
|
void Handler(int signum, siginfo_t *info, void *ucontext) {
|
||||||
|
sigaction(signum, &SIG_def[signum], 0);
|
||||||
|
|
||||||
|
#else
|
||||||
void Handler(int signum) {
|
void Handler(int signum) {
|
||||||
|
#endif
|
||||||
|
|
||||||
const char* name = 0;
|
const char* name = 0;
|
||||||
switch (signum) {
|
switch (signum) {
|
||||||
case SIGABRT: name = "SIGABRT"; break;
|
case SIGABRT: name = "SIGABRT"; break;
|
||||||
|
@ -681,15 +706,69 @@ namespace SignalHandlers {
|
||||||
dump() << "Caught signal " << signum << " in thread " << uint64(thread) << "\n";
|
dump() << "Caught signal " << signum << " in thread " << uint64(thread) << "\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if defined Q_OS_MAC || defined Q_OS_LINUX32 || defined Q_OS_LINUX64
|
||||||
|
ucontext_t *uc = (ucontext_t*)ucontext;
|
||||||
|
|
||||||
|
void *addresses[128] = { 0 };
|
||||||
|
void *caller = 0;
|
||||||
|
|
||||||
|
#if defined(__APPLE__) && !defined(MAC_OS_X_VERSION_10_6)
|
||||||
|
/* OSX < 10.6 */
|
||||||
|
#if defined(__x86_64__)
|
||||||
|
caller = (void*)uap->uc_mcontext->__ss.__rip;
|
||||||
|
#elif defined(__i386__)
|
||||||
|
caller = (void*)uap->uc_mcontext->__ss.__eip;
|
||||||
|
#else
|
||||||
|
caller = (void*)uap->uc_mcontext->__ss.__srr0;
|
||||||
|
#endif
|
||||||
|
#elif defined(__APPLE__) && defined(MAC_OS_X_VERSION_10_6)
|
||||||
|
/* OSX >= 10.6 */
|
||||||
|
#if defined(_STRUCT_X86_THREAD_STATE64) && !defined(__i386__)
|
||||||
|
caller = (void*)uc->uc_mcontext->__ss.__rip;
|
||||||
|
#else
|
||||||
|
caller = (void*)uap->uc_mcontext->__ss.__eip;
|
||||||
|
#endif
|
||||||
|
#elif defined(__linux__)
|
||||||
|
/* Linux */
|
||||||
|
#if defined(__i386__)
|
||||||
|
caller = (void*)uap->uc_mcontext.gregs[14]; /* Linux 32 */
|
||||||
|
#elif defined(__X86_64__) || defined(__x86_64__)
|
||||||
|
caller = (void*)uap->uc_mcontext.gregs[16]; /* Linux 64 */
|
||||||
|
#elif defined(__ia64__) /* Linux IA64 */
|
||||||
|
caller = (void*)uap->uc_mcontext.sc_ip;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
size_t size = backtrace(addresses, 128);
|
||||||
|
|
||||||
|
/* overwrite sigaction with caller's address */
|
||||||
|
if (caller) addresses[1] = caller;
|
||||||
|
|
||||||
|
#ifdef Q_OS_MAC
|
||||||
|
dump() << "\nBase image addresses:\n";
|
||||||
|
for (size_t i = 0; i < size; ++i) {
|
||||||
|
Dl_info info;
|
||||||
|
dump() << i << " ";
|
||||||
|
if (dladdr(addresses[i], &info)) {
|
||||||
|
dump() << uint64(info.dli_fbase) << " (" << info.dli_fname << ")\n";
|
||||||
|
} else {
|
||||||
|
dump() << "_unknown_module_\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
dump() << "\nBacktrace:\n";
|
dump() << "\nBacktrace:\n";
|
||||||
psWriteStackTrace(CrashDumpFileNo);
|
|
||||||
|
backtrace_symbols_fd(addresses, size, CrashDumpFileNo);
|
||||||
|
|
||||||
|
#else
|
||||||
|
psWriteStackTrace();
|
||||||
|
#endif
|
||||||
|
|
||||||
dump() << "\n";
|
dump() << "\n";
|
||||||
|
|
||||||
LoggingCrashThreadId = 0;
|
LoggingCrashThreadId = 0;
|
||||||
|
|
||||||
#ifndef Q_OS_WIN
|
|
||||||
exit(1);
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Status start() {
|
Status start() {
|
||||||
|
@ -729,13 +808,24 @@ namespace SignalHandlers {
|
||||||
t_assert(launchedBinaryName.size() < int(sizeof(LaunchedBinaryName)));
|
t_assert(launchedBinaryName.size() < int(sizeof(LaunchedBinaryName)));
|
||||||
memcpy(LaunchedBinaryName, launchedBinaryName.constData(), launchedBinaryName.size());
|
memcpy(LaunchedBinaryName, launchedBinaryName.constData(), launchedBinaryName.size());
|
||||||
|
|
||||||
|
#ifndef Q_OS_WIN
|
||||||
|
struct sigaction sigact;
|
||||||
|
|
||||||
|
sigact.sa_sigaction = SignalHandlers::Handler;
|
||||||
|
sigemptyset(&sigact.sa_mask);
|
||||||
|
sigact.sa_flags = SA_NODEFER | SA_RESETHAND | SA_SIGINFO;
|
||||||
|
|
||||||
|
sigaction(SIGABRT, &sigact, &SIG_def[SIGABRT]);
|
||||||
|
sigaction(SIGSEGV, &sigact, &SIG_def[SIGSEGV]);
|
||||||
|
sigaction(SIGILL, &sigact, &SIG_def[SIGILL]);
|
||||||
|
sigaction(SIGFPE, &sigact, &SIG_def[SIGFPE]);
|
||||||
|
sigaction(SIGBUS, &sigact, &SIG_def[SIGBUS]);
|
||||||
|
sigaction(SIGSYS, &sigact, &SIG_def[SIGSYS]);
|
||||||
|
#else
|
||||||
signal(SIGABRT, SignalHandlers::Handler);
|
signal(SIGABRT, SignalHandlers::Handler);
|
||||||
signal(SIGSEGV, SignalHandlers::Handler);
|
signal(SIGSEGV, SignalHandlers::Handler);
|
||||||
signal(SIGILL, SignalHandlers::Handler);
|
signal(SIGILL, SignalHandlers::Handler);
|
||||||
signal(SIGFPE, SignalHandlers::Handler);
|
signal(SIGFPE, SignalHandlers::Handler);
|
||||||
#ifndef Q_OS_WIN
|
|
||||||
signal(SIGBUS, SignalHandlers::Handler);
|
|
||||||
signal(SIGSYS, SignalHandlers::Handler);
|
|
||||||
#endif
|
#endif
|
||||||
return Started;
|
return Started;
|
||||||
}
|
}
|
||||||
|
|
|
@ -976,14 +976,6 @@ QAbstractNativeEventFilter *psNativeEventFilter() {
|
||||||
void psWriteDump() {
|
void psWriteDump() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void psWriteStackTrace(int file) {
|
|
||||||
void *addresses[1024] = { 0 };
|
|
||||||
|
|
||||||
size_t size = backtrace(addresses, 1024);
|
|
||||||
|
|
||||||
backtrace_symbols_fd(addresses, size, file);
|
|
||||||
}
|
|
||||||
|
|
||||||
QString demanglestr(const QString &mangled) {
|
QString demanglestr(const QString &mangled) {
|
||||||
QByteArray cmd = ("c++filt -n " + mangled).toUtf8();
|
QByteArray cmd = ("c++filt -n " + mangled).toUtf8();
|
||||||
FILE *f = popen(cmd.constData(), "r");
|
FILE *f = popen(cmd.constData(), "r");
|
||||||
|
|
|
@ -114,7 +114,6 @@ private:
|
||||||
};
|
};
|
||||||
|
|
||||||
void psWriteDump();
|
void psWriteDump();
|
||||||
void psWriteStackTrace(int file);
|
|
||||||
int psShowCrash(const QString &crashdump);
|
int psShowCrash(const QString &crashdump);
|
||||||
|
|
||||||
void psDeleteDir(const QString &dir);
|
void psDeleteDir(const QString &dir);
|
||||||
|
|
|
@ -529,14 +529,6 @@ void psWriteDump() {
|
||||||
SignalHandlers::dump() << "OS-Version: " << v;
|
SignalHandlers::dump() << "OS-Version: " << v;
|
||||||
}
|
}
|
||||||
|
|
||||||
void psWriteStackTrace(int file) {
|
|
||||||
void *addresses[1024] = { 0 };
|
|
||||||
|
|
||||||
size_t size = backtrace(addresses, 1024);
|
|
||||||
|
|
||||||
backtrace_symbols_fd(addresses, size, file);
|
|
||||||
}
|
|
||||||
|
|
||||||
QString demanglestr(const QString &mangled) {
|
QString demanglestr(const QString &mangled) {
|
||||||
QByteArray cmd = ("c++filt -n " + mangled).toUtf8();
|
QByteArray cmd = ("c++filt -n " + mangled).toUtf8();
|
||||||
FILE *f = popen(cmd.constData(), "r");
|
FILE *f = popen(cmd.constData(), "r");
|
||||||
|
@ -553,6 +545,69 @@ QString demanglestr(const QString &mangled) {
|
||||||
return result.trimmed();
|
return result.trimmed();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString escapeShell(const QString &str) {
|
||||||
|
QString result;
|
||||||
|
const QChar *b = str.constData(), *e = str.constEnd();
|
||||||
|
for (const QChar *ch = b; ch != e; ++ch) {
|
||||||
|
if (*ch == ' ' || *ch == '"' || *ch == '\'' || *ch == '\\') {
|
||||||
|
if (result.isEmpty()) {
|
||||||
|
result.reserve(str.size() * 2);
|
||||||
|
}
|
||||||
|
if (ch > b) {
|
||||||
|
result.append(b, ch - b);
|
||||||
|
}
|
||||||
|
result.append('\\');
|
||||||
|
b = ch;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (result.isEmpty()) return str;
|
||||||
|
|
||||||
|
if (e > b) {
|
||||||
|
result.append(b, e - b);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
QStringList atosstr(uint64 *addresses, int count, uint64 base) {
|
||||||
|
QStringList result;
|
||||||
|
if (!count) return result;
|
||||||
|
|
||||||
|
result.reserve(count);
|
||||||
|
QString cmdstr = "atos -o " + escapeShell(cExeDir() + cExeName()) + qsl("/Contents/MacOS/Telegram -l 0x%1").arg(base, 0, 16);
|
||||||
|
for (int i = 0; i < count; ++i) {
|
||||||
|
if (addresses[i]) {
|
||||||
|
cmdstr += qsl(" 0x%1").arg(addresses[i], 0, 16);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
QByteArray cmd = cmdstr.toUtf8();
|
||||||
|
FILE *f = popen(cmd.constData(), "r");
|
||||||
|
|
||||||
|
QStringList atosResult;
|
||||||
|
if (f) {
|
||||||
|
char buffer[4096] = {0};
|
||||||
|
while (!feof(f)) {
|
||||||
|
if (fgets(buffer, 4096, f) != NULL) {
|
||||||
|
atosResult.push_back(QString::fromUtf8(buffer));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pclose(f);
|
||||||
|
}
|
||||||
|
for (int i = 0, j = 0; i < count; ++i) {
|
||||||
|
if (addresses[i]) {
|
||||||
|
if (j < atosResult.size() && !atosResult.at(j).isEmpty() && !atosResult.at(j).startsWith(qstr("0x"))) {
|
||||||
|
result.push_back(atosResult.at(j).trimmed());
|
||||||
|
} else {
|
||||||
|
result.push_back(QString());
|
||||||
|
}
|
||||||
|
++j;
|
||||||
|
} else {
|
||||||
|
result.push_back(QString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
QString _showCrashDump(const QByteArray &crashdump, QString dumpfile) {
|
QString _showCrashDump(const QByteArray &crashdump, QString dumpfile) {
|
||||||
QString initial = QString::fromUtf8(crashdump), result;
|
QString initial = QString::fromUtf8(crashdump), result;
|
||||||
QStringList lines = initial.split('\n');
|
QStringList lines = initial.split('\n');
|
||||||
|
@ -560,6 +615,38 @@ QString _showCrashDump(const QByteArray &crashdump, QString dumpfile) {
|
||||||
int32 i = 0, l = lines.size();
|
int32 i = 0, l = lines.size();
|
||||||
|
|
||||||
while (i < l) {
|
while (i < l) {
|
||||||
|
uint64 addresses[1024] = { 0 };
|
||||||
|
for (; i < l; ++i) {
|
||||||
|
result.append(lines.at(i)).append('\n');
|
||||||
|
QString line = lines.at(i).trimmed();
|
||||||
|
if (line == qstr("Base image addresses:")) {
|
||||||
|
++i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64 base = 0;
|
||||||
|
for (int32 start = i; i < l; ++i) {
|
||||||
|
QString line = lines.at(i).trimmed();
|
||||||
|
if (line.isEmpty()) break;
|
||||||
|
|
||||||
|
if (!base) {
|
||||||
|
QRegularExpressionMatch m = QRegularExpression(qsl("^\\d+ (\\d+) \\((.+)\\)")).match(line);
|
||||||
|
if (m.hasMatch()) {
|
||||||
|
if (uint64 address = m.captured(1).toULongLong()) {
|
||||||
|
if (m.captured(2).endsWith(qstr("Contents/MacOS/Telegram"))) {
|
||||||
|
base = address;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (base) {
|
||||||
|
result.append(qsl("(base address read: 0x%1)\n").arg(base, 0, 16));
|
||||||
|
} else {
|
||||||
|
result.append(qsl("ERROR: base address not read!\n"));
|
||||||
|
}
|
||||||
|
|
||||||
for (; i < l; ++i) {
|
for (; i < l; ++i) {
|
||||||
result.append(lines.at(i)).append('\n');
|
result.append(lines.at(i)).append('\n');
|
||||||
QString line = lines.at(i).trimmed();
|
QString line = lines.at(i).trimmed();
|
||||||
|
@ -569,7 +656,22 @@ QString _showCrashDump(const QByteArray &crashdump, QString dumpfile) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int32 start = i; i < l; ++i) {
|
int32 start = i;
|
||||||
|
for (; i < l; ++i) {
|
||||||
|
QString line = lines.at(i).trimmed();
|
||||||
|
if (line.isEmpty()) break;
|
||||||
|
|
||||||
|
if (QRegularExpression(qsl("^\\d+")).match(line).hasMatch()) {
|
||||||
|
QStringList lst = line.split(' ', QString::SkipEmptyParts);
|
||||||
|
if (lst.size() > 2) {
|
||||||
|
uint64 addr = lst.at(2).startsWith(qstr("0x")) ? lst.at(2).mid(2).toULongLong(0, 16) : lst.at(2).toULongLong();
|
||||||
|
addresses[i - start] = addr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QStringList atos = atosstr(addresses, i - start, base);
|
||||||
|
for (i = start; i < l; ++i) {
|
||||||
QString line = lines.at(i).trimmed();
|
QString line = lines.at(i).trimmed();
|
||||||
if (line.isEmpty()) break;
|
if (line.isEmpty()) break;
|
||||||
|
|
||||||
|
@ -581,7 +683,18 @@ QString _showCrashDump(const QByteArray &crashdump, QString dumpfile) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
QStringList lst = line.split(' ', QString::SkipEmptyParts);
|
QStringList lst = line.split(' ', QString::SkipEmptyParts);
|
||||||
result.append(lst.at(0)).append(' ');
|
result.append('\n').append(lst.at(0)).append(qsl(". "));
|
||||||
|
if (lst.size() < 3) {
|
||||||
|
result.append(qstr("BAD LINE: ")).append(line).append('\n');
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (i - start < atos.size()) {
|
||||||
|
if (!atos.at(i - start).isEmpty()) {
|
||||||
|
result.append(atos.at(i - start)).append('\n');
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (int j = 1, s = lst.size();;) {
|
for (int j = 1, s = lst.size();;) {
|
||||||
if (lst.at(j).startsWith('_')) {
|
if (lst.at(j).startsWith('_')) {
|
||||||
result.append(demanglestr(lst.at(j)));
|
result.append(demanglestr(lst.at(j)));
|
||||||
|
@ -606,7 +719,7 @@ QString _showCrashDump(const QByteArray &crashdump, QString dumpfile) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
result.append('\n');
|
result.append(qsl(" [demangled]")).append('\n');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -141,7 +141,6 @@ private:
|
||||||
};
|
};
|
||||||
|
|
||||||
void psWriteDump();
|
void psWriteDump();
|
||||||
void psWriteStackTrace(int file);
|
|
||||||
int psShowCrash(const QString &crashdump);
|
int psShowCrash(const QString &crashdump);
|
||||||
|
|
||||||
void psDeleteDir(const QString &dir);
|
void psDeleteDir(const QString &dir);
|
||||||
|
|
|
@ -3008,7 +3008,7 @@ QString _showCrashDump(const QByteArray &crashdump, QString dumpfile) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
void psWriteStackTrace(int file) {
|
void psWriteStackTrace() {
|
||||||
if (!LoadDbgHelp()) {
|
if (!LoadDbgHelp()) {
|
||||||
SignalHandlers::dump() << "ERROR: Could not load dbghelp.dll!\n";
|
SignalHandlers::dump() << "ERROR: Could not load dbghelp.dll!\n";
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -118,7 +118,7 @@ LONG CALLBACK _exceptionFilter(EXCEPTION_POINTERS* pExceptionPointers);
|
||||||
LPTOP_LEVEL_EXCEPTION_FILTER WINAPI RedirectedSetUnhandledExceptionFilter(_In_opt_ LPTOP_LEVEL_EXCEPTION_FILTER lpTopLevelExceptionFilter);
|
LPTOP_LEVEL_EXCEPTION_FILTER WINAPI RedirectedSetUnhandledExceptionFilter(_In_opt_ LPTOP_LEVEL_EXCEPTION_FILTER lpTopLevelExceptionFilter);
|
||||||
|
|
||||||
void psWriteDump();
|
void psWriteDump();
|
||||||
void psWriteStackTrace(int file);
|
void psWriteStackTrace();
|
||||||
int psShowCrash(const QString &crashdump);
|
int psShowCrash(const QString &crashdump);
|
||||||
|
|
||||||
void psDeleteDir(const QString &dir);
|
void psDeleteDir(const QString &dir);
|
||||||
|
|
Loading…
Reference in New Issue