2020-12-12 16:29:29 +00:00
|
|
|
#ifndef WINDOWS_CRASHCATCHER_HPP
|
|
|
|
#define WINDOWS_CRASHCATCHER_HPP
|
|
|
|
|
2022-06-19 11:28:33 +00:00
|
|
|
#include <filesystem>
|
2020-12-12 16:29:29 +00:00
|
|
|
|
|
|
|
#include <components/crashcatcher/crashcatcher.hpp>
|
2023-10-15 14:44:24 +00:00
|
|
|
#include <components/misc/windows.hpp>
|
2020-12-12 16:29:29 +00:00
|
|
|
|
|
|
|
namespace Crash
|
|
|
|
{
|
|
|
|
|
|
|
|
// The implementation spawns the current executable as a monitor process which waits
|
|
|
|
// for a global synchronization event which is sent when the parent process crashes.
|
|
|
|
// The monitor process then extracts crash information from the parent process while
|
|
|
|
// the parent process waits for the monitor process to finish. The crashed process
|
|
|
|
// quits and the monitor writes the crash information to a file.
|
|
|
|
//
|
|
|
|
// To detect unexpected shutdowns of the application which are not handled by the
|
|
|
|
// crash handler, the monitor periodically checks the exit code of the parent
|
|
|
|
// process and exits if it does not return STILL_ACTIVE. You can test this by closing
|
|
|
|
// the main openmw process in task manager.
|
|
|
|
|
|
|
|
static constexpr const int CrashCatcherTimeout = 2500;
|
|
|
|
|
|
|
|
struct CrashSHM;
|
|
|
|
|
|
|
|
class CrashCatcher final
|
|
|
|
{
|
|
|
|
public:
|
2023-07-05 19:38:46 +00:00
|
|
|
static CrashCatcher* instance() { return sInstance; }
|
|
|
|
|
2023-07-11 21:22:26 +00:00
|
|
|
CrashCatcher(int argc, char** argv, const std::filesystem::path& dumpPath,
|
|
|
|
const std::filesystem::path& crashDumpName, const std::filesystem::path& freezeDumpName);
|
2020-12-12 16:29:29 +00:00
|
|
|
~CrashCatcher();
|
|
|
|
|
2023-07-11 21:22:26 +00:00
|
|
|
void updateDumpPath(const std::filesystem::path& dumpPath);
|
|
|
|
|
|
|
|
void updateDumpNames(const std::filesystem::path& crashDumpName, const std::filesystem::path& freezeDumpName);
|
2023-07-05 19:38:46 +00:00
|
|
|
|
2020-12-12 16:29:29 +00:00
|
|
|
private:
|
|
|
|
static CrashCatcher* sInstance;
|
|
|
|
|
|
|
|
// mapped SHM area
|
|
|
|
CrashSHM* mShm = nullptr;
|
|
|
|
// the handle is allocated by the catcher and passed to the monitor
|
|
|
|
// process via the command line which maps the SHM and sends / receives
|
|
|
|
// events through it
|
|
|
|
HANDLE mShmHandle = nullptr;
|
|
|
|
// mutex which guards SHM area
|
|
|
|
HANDLE mShmMutex = nullptr;
|
|
|
|
|
|
|
|
// triggered when the monitor signals the application
|
|
|
|
HANDLE mSignalAppEvent = INVALID_HANDLE_VALUE;
|
|
|
|
|
|
|
|
// triggered when the application wants to wake the monitor process
|
|
|
|
HANDLE mSignalMonitorEvent = INVALID_HANDLE_VALUE;
|
|
|
|
|
|
|
|
void setupIpc();
|
|
|
|
|
|
|
|
void shmLock();
|
|
|
|
|
|
|
|
void shmUnlock();
|
|
|
|
|
2023-07-11 21:22:26 +00:00
|
|
|
void startMonitorProcess(const std::filesystem::path& dumpPath, const std::filesystem::path& crashDumpName,
|
|
|
|
const std::filesystem::path& freezeDumpName);
|
2020-12-12 16:29:29 +00:00
|
|
|
|
|
|
|
void waitMonitor();
|
|
|
|
|
|
|
|
void signalMonitor();
|
|
|
|
|
|
|
|
void installHandler();
|
|
|
|
|
|
|
|
void handleVectoredException(PEXCEPTION_POINTERS info);
|
|
|
|
|
|
|
|
public:
|
|
|
|
static LONG WINAPI vectoredExceptionHandler(PEXCEPTION_POINTERS info);
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Crash
|
|
|
|
|
|
|
|
#endif // WINDOWS_CRASHCATCHER_HPP
|