Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
57 changes: 57 additions & 0 deletions src/viam/sdk/module/signal_manager.cpp
Original file line number Diff line number Diff line change
@@ -1,22 +1,79 @@
#include <viam/sdk/module/signal_manager.hpp>

#ifdef _WIN32
#include <windows.h>
#else
#include <csignal>
#include <pthread.h>
#endif

#include <stdexcept>

namespace viam {
namespace sdk {

#ifdef _WIN32
SignalManager* SignalManager::instance_ = nullptr;
#endif

SignalManager::SignalManager() {
#ifdef _WIN32
SignalManager::instance_ = this;
SetConsoleCtrlHandler(SignalManager::console_handler_routine_, TRUE);
#else
sigemptyset(&sigset_);
sigaddset(&sigset_, SIGINT);
sigaddset(&sigset_, SIGTERM);
pthread_sigmask(SIG_BLOCK, &sigset_, NULL);
#endif
}

SignalManager::~SignalManager() {
#ifdef _WIN32
SetConsoleCtrlHandler(NULL, FALSE);
if (signal_event_ != INVALID_HANDLE_VALUE) {
CloseHandle(signal_event_);
}
SignalManager::instance_ = nullptr;
#endif
}

int SignalManager::wait() {
#ifdef _WIN32
signal_event_ = CreateEvent(NULL, TRUE, FALSE, NULL);
if (signal_event_ == INVALID_HANDLE_VALUE) {
throw std::runtime_error("Failed to create signal event");
}
WaitForSingleObject(signal_event_, INFINITE);
return static_cast<DWORD>(signal_code_);
#else
int sig = 0;
return sigwait(&sigset_, &sig);
#endif
}

#ifdef _WIN32

BOOL WINAPI SignalManager::console_handler_routine_(DWORD dwCtrlType) {
if (dwCtrlType == CTRL_C_EVENT || dwCtrlType == CTRL_BREAK_EVENT) {
SignalManager::instance_->handle_signal_(SIGINT);
return TRUE;
} else if (dwCtrlType == CTRL_CLOSE_EVENT || dwCtrlType == CTRL_LOGOFF_EVENT ||
dwCtrlType == CTRL_SHUTDOWN_EVENT) {
SignalManager::instance_->handle_signal_(SIGTERM);
return TRUE;
}
return FALSE;
}

void SignalManager::handle_signal_(int signal) {
if (signal_event_ != INVALID_HANDLE_VALUE) {
signal_code_ = signal;
SetEvent(signal_event_);
}
}

#endif

} // namespace sdk
} // namespace viam
18 changes: 17 additions & 1 deletion src/viam/sdk/module/signal_manager.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,10 @@

#include <signal.h>

#ifdef _WIN32
#include <windows.h>
#endif

namespace viam {
namespace sdk {

Expand All @@ -14,13 +18,25 @@ class SignalManager {
/// @brief Creates a new SignalManager.
explicit SignalManager();

/// @brief Destructor.
~SignalManager();

/// @brief Wait for SignalManager to receive SIGINT or SIGTERM.
/// @return The signal number if successful.
/// @throws `std::runtime_error` if the underlying sigwait call was unsuccessful.
/// @throws `std::runtime_error` if the underlying wait operation was unsuccessful.
int wait();

private:
#ifdef _WIN32
static BOOL WINAPI console_handler_routine_(DWORD dwCtrlType);
void handle_signal_(int signal);

static SignalManager* instance_;
HANDLE signal_event_ = INVALID_HANDLE_VALUE;
int signal_code_ = 0;
#else
sigset_t sigset_;
#endif
};

} // namespace sdk
Expand Down