QuickCppLib 0.10
Eliminate all the tedious hassle when making state-of-the-art C++ 14 - 23 libraries!
|
The namespace for signal_guard. More...
Namespaces | |
namespace | detail |
Classes | |
class | signal_guard_global_decider |
Install a global signal continuation decider. More... | |
class | signal_guard_install |
On platforms where it is necessary (POSIX), installs, and potentially enables, the global signal handlers for the signals specified by guarded . Each signal installed is threadsafe reference counted, so this is safe to call from multiple threads or instantiate multiple times. It is also guaranteed safe to call from within static data init or deinit, so a very common use case is simply to place an instance into global static data. This ensures that dynamically loaded and unloaded shared objects compose signal guards appropriately. More... | |
class | signal_guard_watchdog |
Call an optional specified routine after a period of time, possibly on another thread. Async signal safe. More... | |
class | signal_raised |
Thrown by the default signal handler to abort the current operation. More... | |
Typedefs | |
using | raised_signal_info = ::raised_signal_info |
Enumerations | |
enum class | signalc { none = 0 , abort_process = SIGABRT , undefined_memory_access = SIGBUS , illegal_instruction = SIGILL , interrupt , broken_pipe = SIGPIPE , segmentation_fault = SIGSEGV , floating_point_error = SIGFPE , process_terminate , timer_expire = SIGALRM , child_exit = SIGCHLD , process_continue = SIGCONT , tty_hangup = SIGHUP , process_kill = SIGKILL , profile_event = SIGPROF , process_quit = SIGQUIT , process_stop = SIGSTOP , tty_stop = SIGTSTP , bad_system_call = SIGSYS , process_trap = SIGTRAP , tty_input = SIGTTIN , tty_output = SIGTTOU , urgent_condition = SIGURG , user_defined1 = SIGUSR1 , user_defined2 = SIGUSR2 , virtual_alarm_clock = SIGVTALRM , cpu_time_limit_exceeded = SIGXCPU , file_size_limit_exceeded = SIGXFSZ , cxx_out_of_memory = 32 , cxx_termination = 33 , _max_value } |
The signals which are supported. More... | |
Functions | |
QUICKCPPLIB_BITFIELD_BEGIN_T (signalc_set, uint64_t) | |
Bitfield for the signals which are supported. | |
template<class U , typename = decltype( std::declval<U>()((raised_signal_info *) 0) )> | |
signal_guard_global_decider< std::decay_t< U > > | make_signal_guard_global_decider (signalc_set guarded, U &&f, bool callfirst=false) |
Convenience instantiator of signal_guard_global_decider . | |
SIGNALGUARD_FUNC_DECL bool | thrd_raise_signal (signalc signo, void *raw_info=nullptr, void *raw_context=nullptr) |
Call the currently installed signal handler for a signal (POSIX), or raise a Win32 structured exception (Windows), returning false if no handler was called due to the currently installed handler being SIG_IGN (POSIX). | |
QUICKCPPLIB_NORETURN SIGNALGUARD_FUNC_DECL void | terminate_process_immediately (const char *msg=nullptr) noexcept |
This initiates a fast fail process termination which immediately exits the process without calling any handlers: on POSIX, this is SIGKILL , on Windows this is TerminateProcess() . | |
template<class U , typename = decltype( std::declval<U>()() )> | |
signal_guard_watchdog< std::decay_t< U > > | make_signal_guard_watchdog (U &&f, unsigned ms=3000) |
Convenience instantiator of signal_guard_watchdog . | |
signal_guard_watchdog< detail::invoke_terminate_process_immediately > | make_signal_guard_watchdog (unsigned ms=3000) |
Convenience instantiator of signal_guard_watchdog . | |
template<class F , class H , class C , class... Args, class R = decltype(std::declval<F>()(std::declval<Args>()...)), typename std::enable_if<(detail::is_constructible_or_void< R, decltype(std::declval< H >()(std::declval< const raised_signal_info * >()))>::value), bool >::type = true, typename std::enable_if<(detail::is_constructible_or_void< bool, decltype(std::declval< C >()(std::declval< raised_signal_info * >()))>::value), bool >::type = true> | |
R | signal_guard (signalc_set guarded, F &&f, H &&h, C &&c, Args &&...args) |
template<class F , class R = decltype(std::declval<F>()())> | |
R | signal_guard (signalc_set guarded, F &&f) |
This is an overloaded member function, provided for convenience. It differs from the above function only in what argument(s) it accepts. | |
template<class F , class H , class R = decltype(std::declval<F>()()), typename std::enable_if<(detail::is_constructible_or_void< R, decltype(std::declval< H >()(std::declval< const raised_signal_info * >()))>::value), bool >::type = true> | |
auto | signal_guard (signalc_set guarded, F &&f, H &&h) |
This is an overloaded member function, provided for convenience. It differs from the above function only in what argument(s) it accepts. | |
The namespace for signal_guard.
|
strong |
The signals which are supported.
quickcpplib::_xxx::signal_guard::QUICKCPPLIB_BITFIELD_BEGIN_T | ( | signalc_set | , |
uint64_t | |||
) |
Bitfield for the signals which are supported.
< The process is aborting (SIGABRT
)
< Attempt to access a memory location which can't exist (SIGBUS
)
< Execution of illegal instruction (SIGILL
)
< The process is interrupted (SIGINT
). Note on Windows the handler is ALWAYS called from a separate thread, and the process exits after you return (or take too long executing).
< Reader on a pipe vanished (SIGPIPE
). Note that Windows never raises this signal.
< Attempt to access a memory page whose permissions disallow (SIGSEGV
)
< Floating point error (SIGFPE
)
< Process termination requested (SIGTERM
). Note on Windows the handler is ALWAYS called from a separate thread, and the process exits after you return (or take too long executing).
< Timer has expired (SIGALRM
). POSIX only.
< Child has exited (SIGCHLD
). POSIX only.
< Process is being continued (SIGCONT
). POSIX only.
< Controlling terminal has hung up (SIGHUP
). POSIX only.
< Process has received kill signal (SIGKILL
). POSIX only.
< Profiling timer expired (SIGPROF
). POSIX only.
< Process is being quit (SIGQUIT
). POSIX only.
< Process is being stopped (SIGSTOP
). POSIX only.
< Terminal requests stop (SIGTSTP
). POSIX only.
< Bad system call (SIGSYS
). POSIX only.
< Process has reached a breakpoint (SIGTRAP
). POSIX only.
< Terminal has input (SIGTTIN
). POSIX only.
< Terminal ready for output (SIGTTOU
). POSIX only.
< Urgent condition (SIGURG
). POSIX only.
< User defined 1 (SIGUSR1
). POSIX only.
< User defined 2 (SIGUSR2
). POSIX only.
< Virtual alarm clock (SIGVTALRM
). POSIX only.
< CPU time limit exceeded (SIGXCPU
). POSIX only.
< File size limit exceeded (SIGXFSZ
). POSIX only.
< A call to operator new failed, and a throw is about to occur
< A call to std::terminate() was made
|
inline |
Convenience instantiator of signal_guard_global_decider
.
SIGNALGUARD_FUNC_DECL bool quickcpplib::_xxx::signal_guard::thrd_raise_signal | ( | signalc | signo, |
void * | raw_info = nullptr , |
||
void * | raw_context = nullptr |
||
) |
Call the currently installed signal handler for a signal (POSIX), or raise a Win32 structured exception (Windows), returning false if no handler was called due to the currently installed handler being SIG_IGN
(POSIX).
Note that on POSIX, we fetch the currently installed signal handler and try to call it directly. This allows us to supply custom raw_info
and raw_context
, and we do all the things which the signal handler flags tell us to do beforehand [1]. If the current handler has been defaulted, we enable the signal and execute pthread_kill(pthread_self(), signo)
in order to invoke the default handling.
Note that on Windows, raw_context
is ignored as there is no way to override the context thrown with a Win32 structured exception.
[1]: We currently do not implement alternative stack switching. If a handler requests that, we simply abort the process. Code donations implementing support are welcome.
|
noexcept |
This initiates a fast fail process termination which immediately exits the process without calling any handlers: on POSIX, this is SIGKILL
, on Windows this is TerminateProcess()
.
If msg
is specified, it async signal safely prints that message before termination. If you don't specify a message (nullptr
), a default message is printed. A message of ""
prints nothing.
|
inline |
Convenience instantiator of signal_guard_watchdog
.
|
inline |
Convenience instantiator of signal_guard_watchdog
.
|
inline |
Call a callable f
with signals guarded
protected for this thread only, returning whatever f
or h
returns.
Firstly, how to restore execution to this context is saved, and f(Args...)
is executed, returning whatever f(Args...)
returns if f
completes execution successfully. This is usually inlined code, so it will be quite fast. No memory allocation is performed if a signal_guard_install
for the guarded signal set is already instanced. Approximate best case overhead:
If during the execution of f
, any one of the signals guarded
is raised:
c
, which must have the prototype bool(raised_signal_info *)
, is called with the signal which was raised. You can fix the cause of the signal and return true
to continue execution, or else return false
to halt execution. Note that the variety of code you can call in c
is extremely limited, the same restrictions as for signal handlers apply. Note that on Windows only, signalc::interrupt
and signalc::process_terminate
call c
from some other kernel thread, and the return value is always ignored (the process is always exited).c
returned false
, the execution of f
is halted immediately without stack unwind, the thread is returned to the state just before the calling of f
, and the callable g
is called with the specific signal which occurred. g
must have the prototype R(const raised_signal_info *)
where R
is the return type of f
. g
is called with this signal guard removed, though a signal guard higher in the call chain may instead be active.Obviously all state which f
may have been in the process of doing will be thrown away, in particular any stack allocated variables not marked volatile
will have unspecified values. You should therefore make sure that f
never causes side effects, including the interruption in the middle of some operation, which cannot be fixed by the calling of h
. The default h
simply throws a signal_raised
C++ exception.
signal_guard_install
is not already instanced for the guarded set, one is temporarily installed, which is not quick. You are therefore very strongly recommended, when on POSIX, to call this function with a signal_guard_install
already installed for all the signals you will ever guard. signal_guard_install
is guaranteed to be composable and be safe to use within static data init, so a common use pattern is simply to place a guard install into your static data init.std::set_terminate()
is thread local, we ALWAYS install our termination handler for every thread on creation and we never uninstall it.__try
, so this will limit the complexity of types that can be returned directly by this function. You can usually work around this via an intermediate std::optional
.
|
inline |
This is an overloaded member function, provided for convenience. It differs from the above function only in what argument(s) it accepts.
|
inline |
This is an overloaded member function, provided for convenience. It differs from the above function only in what argument(s) it accepts.