libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
signal.cxx
1// C++
2#include <fstream>
3#include <string>
4
5// cosmos
6#include <cosmos/error/ApiError.hxx>
7#include <cosmos/error/RuntimeError.hxx>
8#include <cosmos/formatting.hxx>
9#include <cosmos/fs/FileDescriptor.hxx>
10#include <cosmos/private/cosmos.hxx>
11#include <cosmos/proc/pidfd.h>
12#include <cosmos/proc/SigInfo.hxx>
13#include <cosmos/proc/signal.hxx>
14#include <cosmos/proc/SigSet.hxx>
15#include <cosmos/string.hxx>
16
17// Linux
18#include <signal.h>
19
20namespace cosmos::signal {
21
22namespace {
23
24 void set_signal_mask(int op, const sigset_t *set, sigset_t *old) {
25 auto res = ::pthread_sigmask(op, set, old);
26
27 if (const auto error = Errno{res}; error != Errno::NO_ERROR) {
28 cosmos_throw (ApiError("pthread_sigmask()", error));
29 }
30 }
31
32} // end anon ns
33
34void raise(const Signal s) {
35 if (::raise(to_integral(s.raw()))) {
36 cosmos_throw (ApiError("raise()"));
37 }
38}
39
40void send(const ProcessID proc, const Signal s) {
41 if (::kill(to_integral(proc), to_integral(s.raw()))) {
42 cosmos_throw (ApiError("kill()"));
43 }
44}
45
46void send(const ProcessID proc, const Signal s, std::variant<void*, int> data) {
47 union sigval val{};
48 if (std::holds_alternative<void*>(data)) {
49 val.sival_ptr = std::get<void*>(data);
50 } else if (std::holds_alternative<int>(data)) {
51 val.sival_int = std::get<int>(data);
52 }
53
54 if (::sigqueue(to_integral(proc), to_integral(s.raw()), val)) {
55 cosmos_throw (ApiError("sigqueue()"));
56 }
57}
58
59void send(const ProcessID proc, const ThreadID thread, const Signal s) {
60 if (::tgkill(to_integral(proc), to_integral(thread), to_integral(s.raw()))) {
61 cosmos_throw (ApiError("tgkill()"));
62 }
63}
64
65void send(const PidFD pidfd, const Signal s) {
66 if (!running_on_valgrind) {
67 // there's no glibc wrapper for this yet
68 //
69 // the third siginfo_t argument allows more precise control of the
70 // signal auxiliary data, but the defaults are just like kill(), so
71 // let's use them for now.
72 if (::pidfd_send_signal(to_integral(pidfd.raw()), to_integral(s.raw()), nullptr, 0) != 0) {
73 cosmos_throw (ApiError("pidfd_send_signal()"));
74 }
75 } else {
76 // when running on Valgrind then this system isn't covered yet
77 // by Valgrind's virtual machine. Implement a fallback to
78 // emulate the system call behaviour.
79 //
80 // we can find out the PID the FD is for by inspect proc
81 std::string path{"/proc/self/fdinfo/"};
82 path += std::to_string(to_integral(pidfd.raw()));
83 std::ifstream fdinfo;
84 fdinfo.open(path);
85
86 if (!fdinfo) {
87 cosmos_throw (ApiError("open(\"/proc/self/fdinfo/<pidfd>\")"));
88 }
89
90 std::string line;
91 constexpr std::string_view PID_FIELD{"Pid:"};
92
93 while (std::getline(fdinfo, line).good()) {
94 if (!is_prefix(line, PID_FIELD))
95 continue;
96
97 auto pid_str = stripped(line.substr(PID_FIELD.size()));
98 size_t processed = 0;
99 auto pid = std::stoi(pid_str, &processed);
100
101 if (processed < 1) {
102 cosmos_throw (RuntimeError("failed to determine PID for PIDFD (valgrind fallback logic)"));
103 }
104
105 signal::send(ProcessID{pid}, s);
106 return;
107 }
108
109 cosmos_throw (RuntimeError("couldn't parse PID for PIDFD (valgrind fallback logic)"));
110 }
111}
112
113void pause() {
114 // this call always returns -1 with errno set to EINT
115 ::pause();
116}
117
118void suspend(const SigSet &mask) {
119 // same as with pause() this call always returns -1 with errno set to EINT
120 ::sigsuspend(mask.raw());
121}
122
123Signal wait(const SigSet &set) {
124 int num{};
125 const auto res = ::sigwait(set.raw(), &num);
126
127 if (res != 0) {
128 cosmos_throw (ApiError("sigwait()", Errno{res}));
129 }
130
131 return Signal{SignalNr{num}};
132
133}
134
135void wait_info(const SigSet &set, SigInfo &info) {
136 if (::sigwaitinfo(set.raw(), info.raw()) < 0) {
137 cosmos_throw (ApiError("sigwaitinfo()"));
138 }
139}
140
141WaitRes timed_wait(const SigSet &set, SigInfo &info, const IntervalTime timeout) {
142 if (::sigtimedwait(set.raw(), info.raw(), &timeout) < 0) {
143 switch(get_errno()) {
144 case Errno::AGAIN: return WaitRes::NO_RESULT;
145 default: {
146 cosmos_throw (ApiError("sigtimedwait()"));
147 }
148 }
149 }
150
151 return WaitRes::SIGNALED;
152}
153
154void block(const SigSet &s, SigSet *old) {
155 set_signal_mask(SIG_BLOCK, s.raw(), old ? old->raw() : nullptr);
156}
157
158void unblock(const SigSet &s, SigSet *old) {
159 set_signal_mask(SIG_UNBLOCK, s.raw(), old ? old->raw() : nullptr);
160}
161
162void set_sigmask(const SigSet &s, SigSet *old) {
163 set_signal_mask(SIG_SETMASK, s.raw(), old ? old->raw() : nullptr);
164}
165
166void get_sigmask(SigSet &old) {
167 // `how` is ignored when no `set` is provided. Just return the current mask in `old`.
168 set_signal_mask(0, nullptr, old.raw());
169}
170
171size_t Stack::MIN_SIZE = MINSIGSTKSZ;
172
173void set_altstack(const Stack &stack, Stack *old) {
174 if (::sigaltstack(stack.raw(), old ? old->raw() : nullptr) != 0) {
175 cosmos_throw (ApiError("sigaltstack()"));
176 }
177}
178
179void get_altstack(Stack &old) {
180 if (::sigaltstack(nullptr, old.raw()) != 0) {
181 cosmos_throw (ApiError("sigalstack()"));
182 }
183}
184
185} // end ns
static size_t MIN_SIZE
Minimum size an alternate signal stack needs to have.
Definition signal.hxx:328
Errno get_errno()
Wrapper that returns the Errno strongly typed representation of the current errno
Definition errno.hxx:111
SignalNr
A primitive signal number specification.
Definition types.hxx:50
WaitRes
Strong type to express timed_wait() and poll_info() results.
Definition signal.hxx:179
std::string stripped(const std::string_view s)
Returns a version of the given string with stripped off leading and trailing whitespace.
Definition string.hxx:54
bool is_prefix(const std::string_view s, const std::string_view prefix)
Returns whether prefix is a prefix of s.
Definition string.hxx:71