2015-03-13 12:52:59 -06:00
|
|
|
/* Copyright (c) 2008-2015, Avian Contributors
|
2014-02-21 17:06:17 -07:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
#ifndef AVIAN_SYSTEM_SIGNAL_H
|
|
|
|
#define AVIAN_SYSTEM_SIGNAL_H
|
|
|
|
|
2014-02-21 23:03:25 -07:00
|
|
|
#include <avian/common.h>
|
|
|
|
|
2014-02-21 17:06:17 -07:00
|
|
|
namespace avian {
|
|
|
|
namespace system {
|
|
|
|
|
2014-02-21 23:03:25 -07:00
|
|
|
// Crash the process.
|
|
|
|
// On posix, the just calls abort. On windows, we dereference a null pointer in
|
|
|
|
// order to trigger the crash dump logic.
|
|
|
|
NO_RETURN void crash();
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
// Registrar for unix-like "signals" (implemented with structured exceptions on
|
|
|
|
// windows).
|
2014-02-21 17:06:17 -07:00
|
|
|
// TODO: remove dependence on generated code having a well-known "thread"
|
|
|
|
// register. Use a thread-local variable instead.
|
|
|
|
class SignalRegistrar {
|
|
|
|
public:
|
|
|
|
class Handler {
|
|
|
|
public:
|
|
|
|
// This function receives state information about the paused thread.
|
|
|
|
// Returns whether to resume execution after the failure point.
|
|
|
|
virtual bool handleSignal(void** ip,
|
|
|
|
void** frame,
|
|
|
|
void** stack,
|
|
|
|
void** thread) = 0;
|
|
|
|
};
|
|
|
|
|
2014-02-21 23:23:01 -07:00
|
|
|
enum Signal {
|
|
|
|
// "Segmentation fault" exceptions (mostly null pointer dereference, but
|
|
|
|
// generally access to any non-mapped memory)
|
|
|
|
SegFault,
|
|
|
|
DivideByZero,
|
|
|
|
};
|
|
|
|
|
2014-02-21 17:06:17 -07:00
|
|
|
SignalRegistrar();
|
|
|
|
~SignalRegistrar();
|
|
|
|
|
2014-02-21 23:23:01 -07:00
|
|
|
// Register a handler for the given signal.
|
|
|
|
// After this method call, anytime the given signal is raised, it will be
|
|
|
|
// handled by the given handler.
|
2014-02-21 17:06:17 -07:00
|
|
|
// Returns true upon success, false upon failure
|
2014-02-21 23:23:01 -07:00
|
|
|
bool registerHandler(Signal signal, Handler* handler);
|
2014-02-21 17:06:17 -07:00
|
|
|
|
2014-02-21 23:23:01 -07:00
|
|
|
// Unregister a handler for the given signal.
|
|
|
|
// After this method call, the given signal will no longer be handled (or,
|
|
|
|
// rather, it go back to being handled by whatever was registered to handle it
|
|
|
|
// before us).
|
2014-02-21 17:06:17 -07:00
|
|
|
// Returns true upon success, false upon failure
|
2014-02-21 23:23:01 -07:00
|
|
|
bool unregisterHandler(Signal signal);
|
2014-02-21 17:06:17 -07:00
|
|
|
|
|
|
|
// Set the directory that a crash dump will be written to should an unhandled
|
|
|
|
// exception be thrown.
|
|
|
|
// Note: this only currently does anything on windows.
|
|
|
|
// TODO: move this out of this class, into a separate "CrashDumper" class or
|
|
|
|
// somesuch.
|
|
|
|
void setCrashDumpDirectory(const char* crashDumpDirectory);
|
|
|
|
|
|
|
|
// This is internal, implementation-specific data. It's declared in the
|
|
|
|
// specific implementation.
|
|
|
|
struct Data;
|
|
|
|
|
|
|
|
private:
|
|
|
|
Data* data;
|
|
|
|
};
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
} // namespace system
|
|
|
|
} // namespace avian
|
2014-02-21 17:06:17 -07:00
|
|
|
|
|
|
|
#endif
|