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>
20namespace cosmos::signal {
24 void set_signal_mask(
int op,
const sigset_t *set, sigset_t *old) {
25 auto res = ::pthread_sigmask(op, set, old);
27 if (
const auto error = Errno{res}; error != Errno::NO_ERROR) {
28 cosmos_throw (ApiError(
"pthread_sigmask()", error));
34void raise(
const Signal s) {
35 if (::raise(to_integral(s.raw()))) {
36 cosmos_throw (ApiError(
"raise()"));
40void send(
const ProcessID proc,
const Signal s) {
41 if (::kill(to_integral(proc), to_integral(s.raw()))) {
42 cosmos_throw (ApiError(
"kill()"));
46void send(
const ProcessID proc,
const Signal s, std::variant<void*, int> data) {
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);
54 if (::sigqueue(to_integral(proc), to_integral(s.raw()), val)) {
55 cosmos_throw (ApiError(
"sigqueue()"));
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()"));
65void send(
const PidFD pidfd,
const Signal s) {
66 if (!running_on_valgrind) {
72 if (::pidfd_send_signal(to_integral(pidfd.raw()), to_integral(s.raw()),
nullptr, 0) != 0) {
73 cosmos_throw (ApiError(
"pidfd_send_signal()"));
81 std::string path{
"/proc/self/fdinfo/"};
82 path += std::to_string(to_integral(pidfd.raw()));
87 cosmos_throw (ApiError(
"open(\"/proc/self/fdinfo/<pidfd>\")"));
91 constexpr std::string_view PID_FIELD{
"Pid:"};
93 while (std::getline(fdinfo, line).good()) {
97 auto pid_str =
stripped(line.substr(PID_FIELD.size()));
99 auto pid = std::stoi(pid_str, &processed);
102 cosmos_throw (RuntimeError(
"failed to determine PID for PIDFD (valgrind fallback logic)"));
105 signal::send(ProcessID{pid}, s);
109 cosmos_throw (RuntimeError(
"couldn't parse PID for PIDFD (valgrind fallback logic)"));
118void suspend(
const SigSet &mask) {
120 ::sigsuspend(mask.raw());
123Signal wait(
const SigSet &set) {
125 const auto res = ::sigwait(set.raw(), &num);
128 cosmos_throw (ApiError(
"sigwait()", Errno{res}));
135void wait_info(
const SigSet &set, SigInfo &info) {
136 if (::sigwaitinfo(set.raw(), info.raw()) < 0) {
137 cosmos_throw (ApiError(
"sigwaitinfo()"));
141WaitRes timed_wait(
const SigSet &set, SigInfo &info,
const IntervalTime timeout) {
142 if (::sigtimedwait(set.raw(), info.raw(), &timeout) < 0) {
144 case Errno::AGAIN:
return WaitRes::NO_RESULT;
146 cosmos_throw (ApiError(
"sigtimedwait()"));
151 return WaitRes::SIGNALED;
154void block(
const SigSet &s, SigSet *old) {
155 set_signal_mask(SIG_BLOCK, s.raw(), old ? old->raw() :
nullptr);
158void unblock(
const SigSet &s, SigSet *old) {
159 set_signal_mask(SIG_UNBLOCK, s.raw(), old ? old->raw() :
nullptr);
162void set_sigmask(
const SigSet &s, SigSet *old) {
163 set_signal_mask(SIG_SETMASK, s.raw(), old ? old->raw() :
nullptr);
166void get_sigmask(SigSet &old) {
168 set_signal_mask(0,
nullptr, old.raw());
173void set_altstack(
const Stack &stack, Stack *old) {
174 if (::sigaltstack(stack.raw(), old ? old->raw() :
nullptr) != 0) {
175 cosmos_throw (ApiError(
"sigaltstack()"));
179void get_altstack(Stack &old) {
180 if (::sigaltstack(
nullptr, old.raw()) != 0) {
181 cosmos_throw (ApiError(
"sigalstack()"));
static size_t MIN_SIZE
Minimum size an alternate signal stack needs to have.
Errno get_errno()
Wrapper that returns the Errno strongly typed representation of the current errno
SignalNr
A primitive signal number specification.
WaitRes
Strong type to express timed_wait() and poll_info() results.
std::string stripped(const std::string_view s)
Returns a version of the given string with stripped off leading and trailing whitespace.
bool is_prefix(const std::string_view s, const std::string_view prefix)
Returns whether prefix is a prefix of s.