2016-09-17 16:23:02 +00:00
|
|
|
// ossig.h - written and placed in the public domain by Jeffrey Walton
|
2016-09-17 18:39:32 +00:00
|
|
|
//
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \file ossig.h
|
|
|
|
/// \brief Utility class for trapping OS signals.
|
|
|
|
/// \since Crypto++ 5.6.5
|
2016-09-17 16:23:02 +00:00
|
|
|
|
|
|
|
#ifndef CRYPTOPP_OS_SIGNAL_H
|
|
|
|
#define CRYPTOPP_OS_SIGNAL_H
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2016-10-02 02:12:08 +00:00
|
|
|
#if defined(UNIX_SIGNALS_AVAILABLE)
|
2016-09-17 16:23:02 +00:00
|
|
|
# include <signal.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
NAMESPACE_BEGIN(CryptoPP)
|
|
|
|
|
|
|
|
// ************** Unix and Linux compatibles ***************
|
|
|
|
|
2016-10-02 02:12:08 +00:00
|
|
|
#if defined(UNIX_SIGNALS_AVAILABLE) || defined(CRYPTOPP_DOXYGEN_PROCESSING)
|
2016-09-18 21:29:32 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Signal handler function pointer
|
|
|
|
/// \details SignalHandlerFn is provided as a stand alone function pointer with external "C" linkage
|
|
|
|
/// \sa SignalHandler, NullSignalHandler
|
2016-09-17 16:23:02 +00:00
|
|
|
extern "C" {
|
|
|
|
typedef void (*SignalHandlerFn) (int);
|
2018-03-31 16:59:25 +00:00
|
|
|
}
|
2016-09-17 16:23:02 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Null signal handler function
|
|
|
|
/// \param unused the signal number
|
|
|
|
/// \details NullSignalHandler is provided as a stand alone function with external "C" linkage
|
|
|
|
/// and not a static member function due to the the member function's implicit
|
|
|
|
/// external "C++" linkage.
|
|
|
|
/// \sa SignalHandler, SignalHandlerFn
|
2016-09-20 02:35:53 +00:00
|
|
|
extern "C" {
|
2017-03-10 06:53:51 +00:00
|
|
|
inline void NullSignalHandler(int unused) {CRYPTOPP_UNUSED(unused);}
|
2018-03-31 16:59:25 +00:00
|
|
|
}
|
2016-09-20 02:35:53 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// Signal handler for Linux and Unix compatibles
|
|
|
|
/// \tparam S Signal number
|
|
|
|
/// \tparam O Flag indicating exsting handler should be overwriiten
|
|
|
|
/// \details SignalHandler() can be used to install a signal handler with the signature
|
|
|
|
/// <tt>void handler_fn(int)</tt>. If <tt>SignalHandlerFn</tt> is not <tt>NULL</tt>, then
|
|
|
|
/// the sigaction is set to the function and the sigaction flags is set to the flags.
|
|
|
|
/// If <tt>SignalHandlerFn</tt> is <tt>NULL</tt>, then a default handler is installed
|
|
|
|
/// using sigaction flags set to 0. The default handler only returns from the call.
|
|
|
|
/// \details Upon destruction the previous signal handler is restored if the former signal handler
|
|
|
|
/// was replaced.
|
|
|
|
/// \details On Cygwin systems using Newlib, you should define <tt>_XOPEN_SOURCE=700</tt> or
|
|
|
|
/// <tt>_GNU_SOURCE</tt>; or use <tt>-std=gnu++03</tt>, <tt>-std=gnu++11</tt>, or similar. If
|
|
|
|
/// you compile with <tt>-std=c++03</tt>, <tt>-std=c++11</tt> or similar, then define
|
|
|
|
/// <tt>_XOPEN_SOURCE=700</tt>.
|
|
|
|
/// \warning Do not use SignalHandler in a code block that uses <tt>setjmp</tt> or <tt>longjmp</tt>
|
|
|
|
/// because the destructor may not run.
|
|
|
|
/// \since Crypto++ 5.6.5
|
|
|
|
/// \sa NullSignalHandler, SignalHandlerFn, \ref CRYPTOPP_ASSERT "CRYPTOPP_ASSERT", DebugTrapHandler
|
2016-09-17 16:23:02 +00:00
|
|
|
template <int S, bool O=false>
|
|
|
|
struct SignalHandler
|
|
|
|
{
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Construct a signal handler
|
|
|
|
/// \param pfn Pointer to a signal handler function
|
|
|
|
/// \param flags Flags to use with the signal handler
|
|
|
|
/// \details SignalHandler() installs a signal handler with the signature
|
|
|
|
/// <tt>void handler_fn(int)</tt>. If <tt>SignalHandlerFn</tt> is not <tt>NULL</tt>, then
|
|
|
|
/// the sigaction is set to the function and the sigaction flags is set to the flags.
|
|
|
|
/// If <tt>SignalHandlerFn</tt> is <tt>NULL</tt>, then a default handler is installed
|
|
|
|
/// using sigaction flags set to 0. The default handler only returns from the call.
|
|
|
|
/// \details Upon destruction the previous signal handler is restored if the former signal handler
|
|
|
|
/// was overwritten.
|
|
|
|
/// \details On Cygwin systems using Newlib, you should define <tt>_XOPEN_SOURCE=700</tt> or
|
|
|
|
/// <tt>_GNU_SOURCE</tt>; or use <tt>-std=gnu++03</tt>, <tt>-std=gnu++11</tt>, or similar. If
|
|
|
|
/// you compile with <tt>-std=c++03</tt>, <tt>-std=c++11</tt> or similar, then define
|
|
|
|
/// <tt>_XOPEN_SOURCE=700</tt>.
|
|
|
|
/// \warning Do not use SignalHandler in a code block that uses <tt>setjmp</tt> or <tt>longjmp</tt>
|
|
|
|
/// because the destructor may not run. <tt>setjmp</tt> is why cpu.cpp does not use SignalHandler
|
|
|
|
/// during CPU feature testing.
|
|
|
|
/// \since Crypto++ 5.6.5
|
2017-03-01 11:10:06 +00:00
|
|
|
SignalHandler(SignalHandlerFn pfn = NULLPTR, int flags = 0) : m_installed(false)
|
2016-09-17 16:23:02 +00:00
|
|
|
{
|
|
|
|
// http://pubs.opengroup.org/onlinepubs/007908799/xsh/sigaction.html
|
|
|
|
struct sigaction new_handler;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
ret = sigaction (S, 0, &m_old);
|
|
|
|
if (ret != 0) break; // Failed
|
|
|
|
|
|
|
|
// Don't step on another's handler if Overwrite=false
|
|
|
|
if (m_old.sa_handler != 0 && !O) break;
|
|
|
|
|
2017-03-10 06:53:51 +00:00
|
|
|
// Cygwin/Newlib requires -D_XOPEN_SOURCE=700
|
2016-09-17 16:23:02 +00:00
|
|
|
ret = sigemptyset (&new_handler.sa_mask);
|
|
|
|
if (ret != 0) break; // Failed
|
2016-10-06 00:29:01 +00:00
|
|
|
|
|
|
|
new_handler.sa_handler = (pfn ? pfn : &NullSignalHandler);
|
|
|
|
new_handler.sa_flags = (pfn ? flags : 0);
|
2016-09-17 16:23:02 +00:00
|
|
|
|
|
|
|
// Install it
|
|
|
|
ret = sigaction (S, &new_handler, 0);
|
|
|
|
if (ret != 0) break; // Failed
|
|
|
|
|
|
|
|
m_installed = true;
|
|
|
|
|
|
|
|
} while(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
~SignalHandler()
|
|
|
|
{
|
|
|
|
if (m_installed)
|
|
|
|
sigaction (S, &m_old, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
struct sigaction m_old;
|
|
|
|
bool m_installed;
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Not copyable
|
|
|
|
SignalHandler(const SignalHandler &);
|
|
|
|
void operator=(const SignalHandler &);
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
NAMESPACE_END
|
|
|
|
|
|
|
|
#endif // CRYPTOPP_OS_SIGNAL_H
|