libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
SigAction.cxx
1// C++
2#include <atomic>
3#include <array>
4
5// cosmos
6#include <cosmos/error/ApiError.hxx>
7#include <cosmos/error/UsageError.hxx>
8#include <cosmos/proc/SigAction.hxx>
9#include <cosmos/proc/SigInfo.hxx>
10#include <cosmos/proc/signal.hxx>
11#include <cosmos/utils.hxx>
12
13namespace cosmos {
14
15namespace {
16
17/*
18 * lock-free tables for storing callback pointers for asynchronous signals
19 *
20 * this is a bit of a memory waste, but since we do not have efficient locking
21 * primitives available in signal handlers this seems the only way to achieve
22 * type safe callback pointers.
23 */
24std::array<std::atomic_intptr_t, to_integral(SignalNr::MAXIMUM)> simple_handlers;
25std::array<std::atomic_intptr_t, to_integral(SignalNr::MAXIMUM)> info_handlers;
26
27/*
28 * central entry point for old school simple signal handlers that only receive
29 * the signal number that occurred.
30 */
31void simple_handler(int sig) {
32 const auto &handler = simple_handlers[sig];
33
34 if (!handler)
35 // race?
36 return;
37
38 reinterpret_cast<SigAction::SimpleHandler>(handler.load())(Signal{SignalNr{sig}});
39}
40
41/*
42 * central entry point for extended siginfo signal handlers that additionally
43 * receive siginfo_t and signal handling context data.
44 */
45void info_handler(int sig, siginfo_t *info, void *ctx) {
46 // this is only used internally by libc, or could be used for
47 // profiling, but it is very low-level and currently unneeded.
48 (void)ctx;
49
50 const auto &handler = info_handlers[sig];
51
52 if (!handler)
53 // race?
54 return;
55
56 const auto info2 = *reinterpret_cast<SigInfo*>(info);
57
58 reinterpret_cast<SigAction::InfoHandler>(handler.load())(info2);
59}
60
61} // end anon ns
62
63namespace signal {
64
65
66void set_action(const Signal sig, const SigAction &action, SigAction *old) {
67
68 const auto raw_sig = to_integral(sig.raw());
69
70 if (::sigaction(raw_sig, action.raw(), old ? old->raw() : nullptr) != 0) {
71 cosmos_throw (ApiError("sigaction()"));
72 }
73
74 /*
75 * Here is a spot for a race condition, if previously a custom handler
76 * was registered, then the new signal configuration can arrive at the
77 * old handler until we rewrite the entries below.
78 *
79 * It is impossible to avoid that unless falling back to the plain C
80 * callback. Locking is not possible, because the asynchronous handler
81 * is not allowed to call locks (except for SYSV semaphores, which
82 * would be overkill).
83 */
84
85 using SimpleHandler = SigAction::SimpleHandler;
86 using InfoHandler = SigAction::InfoHandler;
87
88 const auto &variant = action.m_handler;
89 SimpleHandler old_simple = reinterpret_cast<SimpleHandler>(simple_handlers[raw_sig].load());
90 InfoHandler old_info = reinterpret_cast<InfoHandler>(info_handlers[raw_sig].load());;
91
92 if (std::holds_alternative<SimpleHandler>(variant)) {
93 const auto handler = std::get<SimpleHandler>(variant);
94
96 // reset callback pointers just to be sure
97 simple_handlers[raw_sig].store(0);
98 info_handlers[raw_sig].store(0);
99 } else {
100 const auto intptr = reinterpret_cast<intptr_t>(handler);
101 simple_handlers[raw_sig].store(intptr);
102 }
103 } else if (std::holds_alternative<InfoHandler>(variant)) {
104 const auto intptr = reinterpret_cast<intptr_t>(std::get<InfoHandler>(variant));
105 info_handlers[raw_sig].store(intptr);
106 }
107
108 if (old) {
109 old->updateFromOld(old_info, old_simple);
110 }
111}
112
113void get_action(const Signal sig, SigAction &action) {
114
115 const auto raw_sig = to_integral(sig.raw());
116
117 if (::sigaction(raw_sig, nullptr, action.raw())) {
118 cosmos_throw (ApiError("sigaction()"));
119 }
120
121 action.updateFromOld(
122 reinterpret_cast<SigAction::InfoHandler>(info_handlers[raw_sig].load()),
123 reinterpret_cast<SigAction::SimpleHandler>(simple_handlers[raw_sig].load())
124 );
125}
126
127} // end ns signal
128
129/*
130 * these cannot be constexpr (well, DEFAULT could be), because making a
131 * pointer out of a literal (1) needs a cast, and constexpr doesn't allow
132 * casts.
133 */
137
139 if (handler == UNKNOWN) {
140 cosmos_throw (UsageError("Cannot set UNKNOWN type handler"));
141 }
142
143 // NOTE: don't use setFlags() here, since it doesn't allow changing
144 // SA_SIGINFO!
145 m_raw.sa_flags &= ~SA_SIGINFO;
146
147 if (handler == IGNORE) {
148 m_raw.sa_handler = SIG_IGN;
149 } else if (handler == DEFAULT) {
150 m_raw.sa_handler = SIG_DFL;
151 } else {
152 m_raw.sa_handler = &simple_handler;
153 }
154
155 m_handler = handler;
156}
157
159 // NOTE: don't use setFlags() here, since it doesn't allow changing
160 // SA_SIGINFO!
161 m_raw.sa_flags |= SA_SIGINFO;
162 m_raw.sa_sigaction = &info_handler;
163 m_handler = handler;
164}
165
166void SigAction::updateFromOld(InfoHandler info, SimpleHandler simple) {
167 // we need to reflect the data found in m_raw into m_handler, if
168 // anybody should get the idea to compare it.
169
170 if (getFlags()[Flag::SIGINFO]) {
171 if (m_raw.sa_sigaction != info_handler) {
172 // it's a custom handler not set by libcosmos routines
174 } else {
175 // it's the old siginfo handler supplied to us
176 m_handler = info;
177 }
178 } else {
179 if (m_raw.sa_handler == SIG_DFL) {
181 } else if (m_raw.sa_handler == SIG_IGN) {
183 } else if (m_raw.sa_handler != simple_handler) {
184 // it's a custom handler not set by libcosmos routines
186 } else {
187 // it's the old simple handler supplied to us
188 m_handler = simple;
189 }
190 }
191}
192
193} // end ns
static const SimpleHandler DEFAULT
Special value of SimpleHandler to configure the default signal action as documented in man 7 signal.
Definition SigAction.hxx:76
void(*)(const Signal) SimpleHandler
Simple signal handler for receiving only the Signal number.
Definition SigAction.hxx:69
Flags getFlags() const
Retrieve the current flags.
std::variant< SimpleHandler, InfoHandler > m_handler
The currently configured callback.
struct sigaction m_raw
Low level sigaction struct.
static const SimpleHandler UNKNOWN
Special value of SimpleHandler in case a custom non-libcosmos handler is installed.
Definition SigAction.hxx:78
static const SimpleHandler IGNORE
Special value of SimpleHandler to ignore signals.
Definition SigAction.hxx:74
@ SIGINFO
The signal handler callback takes three arguments providing additional information (SigInfo).
void setHandler(SimpleHandler handler)
Sets a new SimpleHandler style signal handler function.
void(*)(const SigInfo &) InfoHandler
Extended signal handler for receiving additional SigInfo data.
Definition SigAction.hxx:71
Exception type for logical usage errors within the application.
SignalNr
A primitive signal number specification.
Definition types.hxx:50
bool in_list(const T &v, const std::initializer_list< T > &l)
Checks whether the value v is found in the given list of values l.
Definition utils.hxx:117