Report errors on installing crash handler

ini_importer_tests
elsid 12 months ago
parent 2ef286b27a
commit 140cc53b55
No known key found for this signature in database
GPG Key ID: 4DE04C198CBA7625

@ -74,14 +74,15 @@ namespace
{ {
int mCode; int mCode;
const char* mDescription; const char* mDescription;
const char* mName = "";
}; };
constexpr SignalInfo signals[] = { constexpr SignalInfo signals[] = {
{ SIGSEGV, "Segmentation fault" }, { SIGSEGV, "Segmentation fault", "SIGSEGV" },
{ SIGILL, "Illegal instruction" }, { SIGILL, "Illegal instruction", "SIGILL" },
{ SIGFPE, "FPU exception" }, { SIGFPE, "FPU exception", "SIGFPE" },
{ SIGBUS, "System BUS error" }, { SIGBUS, "System BUS error", "SIGBUS" },
{ SIGABRT, "Abnormal termination condition" }, { SIGABRT, "Abnormal termination condition", "SIGABRT" },
}; };
constexpr SignalInfo sigIllCodes[] = { constexpr SignalInfo sigIllCodes[] = {
@ -388,11 +389,15 @@ static void getExecPath(char** argv)
if (sysctl(mib, 4, argv0, &size, nullptr, 0) == 0) if (sysctl(mib, 4, argv0, &size, nullptr, 0) == 0)
return; return;
Log(Debug::Warning) << "Failed to call sysctl: " << std::generic_category().message(errno);
#endif #endif
#if defined(__APPLE__) #if defined(__APPLE__)
if (proc_pidpath(getpid(), argv0, sizeof(argv0)) > 0) if (proc_pidpath(getpid(), argv0, sizeof(argv0)) > 0)
return; return;
Log(Debug::Warning) << "Failed to call proc_pidpath: " << std::generic_category().message(errno);
#endif #endif
const char* statusPaths[] = { "/proc/self/exe", "/proc/self/file", "/proc/curproc/exe", "/proc/curproc/file" }; const char* statusPaths[] = { "/proc/self/exe", "/proc/self/file", "/proc/curproc/exe", "/proc/curproc/file" };
memset(argv0, 0, sizeof(argv0)); memset(argv0, 0, sizeof(argv0));
@ -401,18 +406,28 @@ static void getExecPath(char** argv)
{ {
if (readlink(path, argv0, sizeof(argv0)) != -1) if (readlink(path, argv0, sizeof(argv0)) != -1)
return; return;
Log(Debug::Warning) << "Failed to call readlink for \"" << path
<< "\": " << std::generic_category().message(errno);
} }
if (argv[0][0] == '/') if (argv[0][0] == '/')
{
snprintf(argv0, sizeof(argv0), "%s", argv[0]); snprintf(argv0, sizeof(argv0), "%s", argv[0]);
else if (getcwd(argv0, sizeof(argv0)) != nullptr) return;
}
if (getcwd(argv0, sizeof(argv0)) == nullptr)
{ {
Log(Debug::Error) << "Failed to call getcwd: " << std::generic_category().message(errno);
return;
}
const int cwdlen = strlen(argv0); const int cwdlen = strlen(argv0);
snprintf(argv0 + cwdlen, sizeof(argv0) - cwdlen, "/%s", argv[0]); snprintf(argv0 + cwdlen, sizeof(argv0) - cwdlen, "/%s", argv[0]);
} }
}
static int crashCatcherInstallHandlers(char** argv) static bool crashCatcherInstallHandlers(char** argv)
{ {
getExecPath(argv); getExecPath(argv);
@ -423,25 +438,33 @@ static int crashCatcherInstallHandlers(char** argv)
altss.ss_sp = altstack; altss.ss_sp = altstack;
altss.ss_flags = 0; altss.ss_flags = 0;
altss.ss_size = SIGSTKSZ; altss.ss_size = SIGSTKSZ;
sigaltstack(&altss, nullptr); if (sigaltstack(&altss, nullptr) == -1)
{
Log(Debug::Error) << "Failed to call sigaltstack: " << std::generic_category().message(errno);
return false;
}
struct sigaction sa; struct sigaction sa;
memset(&sa, 0, sizeof(sa)); memset(&sa, 0, sizeof(sa));
sa.sa_sigaction = crash_catcher; sa.sa_sigaction = crash_catcher;
sa.sa_flags = SA_RESETHAND | SA_NODEFER | SA_SIGINFO | SA_ONSTACK; sa.sa_flags = SA_RESETHAND | SA_NODEFER | SA_SIGINFO | SA_ONSTACK;
sigemptyset(&sa.sa_mask); if (sigemptyset(&sa.sa_mask) == -1)
{
constexpr int signals[] = { SIGSEGV, SIGILL, SIGFPE, SIGBUS, SIGABRT }; Log(Debug::Error) << "Failed to call sigemptyset: " << std::generic_category().message(errno);
return false;
}
int retval = 0; for (const SignalInfo& signal : signals)
for (const int signal : signals)
{ {
if (sigaction(signal, &sa, nullptr) == -1) if (sigaction(signal.mCode, &sa, nullptr) == -1)
{ {
retval = -1; Log(Debug::Error) << "Failed to call sigaction for signal " << signal.mName << " (" << signal.mCode
<< "): " << std::generic_category().message(errno);
return false;
} }
} }
return retval;
return true;
} }
static bool is_debugger_present() static bool is_debugger_present()
@ -509,9 +532,9 @@ void crashCatcherInstall(int argc, char** argv, const std::filesystem::path& cra
if (is_debugger_present()) if (is_debugger_present())
return; return;
if (crashCatcherInstallHandlers(argv) == -1) if (crashCatcherInstallHandlers(argv))
Log(Debug::Warning) << "Installing crash handler failed";
else
Log(Debug::Info) << "Crash handler installed"; Log(Debug::Info) << "Crash handler installed";
else
Log(Debug::Warning) << "Installing crash handler failed";
#endif #endif
} }

Loading…
Cancel
Save