libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
SignalFD.cxx
1// Linux
2#include <unistd.h>
3
4// C++
5#include <iostream>
6
7// cosmos
8#include <cosmos/private/cosmos.hxx>
9#include <cosmos/error/ApiError.hxx>
10#include <cosmos/error/UsageError.hxx>
11#include <cosmos/error/RuntimeError.hxx>
12#include <cosmos/proc/SignalFD.hxx>
13#include <cosmos/utils.hxx>
14
15namespace cosmos {
16
17SignalFD::~SignalFD() {
18 try {
19 close();
20 } catch (const std::exception &ex) {
21 noncritical_error("Failed to close signal fd", ex);
22 }
23}
24
25void SignalFD::create(const SigSet &mask) {
26 close();
27 auto fd = ::signalfd(-1, mask.raw(), SFD_CLOEXEC);
28 if (fd == -1) {
29 cosmos_throw (ApiError("signalfd()"));
30 }
31
32 m_fd.setFD(FileNum{fd});
33}
34
35void SignalFD::adjustMask(const SigSet &mask) {
36 if (!valid()) {
37 cosmos_throw (UsageError("no signal fd currently open"));
38 }
39
40 // NOTE: it's unclear from the man page whether flags are used when
41 // modifying an existing signal fd. Let's pass on zero, hoping that no
42 // flags will be taken away again through this.
43 auto fd = ::signalfd(to_integral(m_fd.raw()), mask.raw(), 0);
44 if (fd == -1) {
45 cosmos_throw (ApiError("signalfd()"));
46 }
47}
48
50 auto raw = info.raw();
51 constexpr auto RAW_SIZE = sizeof(*raw);
52
53 auto res = ::read(to_integral(m_fd.raw()), raw, RAW_SIZE);
54
55 if (res < 0) {
56 cosmos_throw (ApiError("read(sigfd)"));
57 }
58 else if (static_cast<size_t>(res) < RAW_SIZE) {
59 cosmos_throw (RuntimeError("short read from signal fd"));
60 }
61}
62
63using Info = SignalFD::Info;
64
66 const std::initializer_list<Signal> SPECIAL_SIGS{
67 signal::FPE, signal::BUS, signal::ILL, signal::SEGV,
68 signal::BUS, signal::TRAP, signal::CHILD, signal::POLL,
69 signal::BAD_SYS};
70
71 // the lower level system call `rt_sigqueueinfo` allows userspace to
72 // send arbitrary codes < 0, thus the signal number alone doesn't mean
73 // the kernel was the source.
74 if (isTrustedSource() && in_list(this->sigNr(), SPECIAL_SIGS)) {
75 return Source::KERNEL;
76 }
77
78 return Source{m_raw.ssi_code};
79}
80
81std::optional<const Info::UserSigData> Info::userSigData() const {
83 return UserSigData{this->procCtx()};
84 }
85
86 return std::nullopt;
87}
88
89std::optional<const Info::QueueSigData> Info::queueSigData() const {
91 // the signalfd_siginfo does not involve a union, but contains
92 // always both the 32-bit int value and the 64-bit pointer
93 // value. The pointer always contains the lower 32-bit of the
94 // int, if an int was sent. Thus we place the pointer into the
95 // union and constructor SigInfo::CustomData from it.
96 union sigval val;
97 val.sival_ptr = reinterpret_cast<void*>(m_raw.ssi_ptr);
98 return QueueSigData{this->procCtx(), SigInfo::CustomData{val}};
99 }
100
101 return std::nullopt;
102}
103
104std::optional<const Info::MsgQueueData> Info::msgQueueData() const {
106 union sigval val;
107 val.sival_ptr = reinterpret_cast<void*>(m_raw.ssi_ptr);
108 return MsgQueueData{this->procCtx(), SigInfo::CustomData{val}};
109 }
110
111 return std::nullopt;
112}
113
114std::optional<const Info::TimerData> Info::timerData() const {
116 return TimerData{
117 // is unsigned here, so cast it to signed
118 TimerData::TimerID{static_cast<int>(m_raw.ssi_tid)},
119 // same here
120 static_cast<int>(m_raw.ssi_overrun)
121 };
122 }
123
124 return std::nullopt;
125}
126
127std::optional<const Info::SysData> Info::sysData() const {
128 if (sigNr() == signal::BAD_SYS) {
129 return SysData{
130 SysData::Reason{m_raw.ssi_code},
131 // is an uint64_t here, so cast it to void*
132 reinterpret_cast<void*>(m_raw.ssi_call_addr),
133 m_raw.ssi_syscall,
134 ptrace::Arch{m_raw.ssi_arch},
135 this->error()
136 };
137 }
138
139 return std::nullopt;
140}
141
142std::optional<const Info::ChildData> Info::childData() const {
143 if (sigNr() == signal::CHILD) {
144 using Event = ChildData::Event;
145 const auto event = Event{m_raw.ssi_code};
146
147 return ChildData{
148 event,
149 this->procCtx(),
150 event == Event::EXITED ? std::make_optional(ExitStatus{m_raw.ssi_status}) : std::nullopt,
151 event == Event::EXITED ? std::nullopt : std::make_optional(Signal{SignalNr{m_raw.ssi_status}}),
152 // is an uint64_t here, so cast it to clock_t for reuse
153 std::make_optional(ClockTicks{static_cast<clock_t>(m_raw.ssi_utime)}),
154 std::make_optional(ClockTicks{static_cast<clock_t>(m_raw.ssi_stime)})
155 };
156 }
157
158 return std::nullopt;
159}
160
161std::optional<const Info::PollData> Info::pollData() const {
162 // SIGIO and SIGPOLL are the same on Linux, but just to be sure ...
163 if (sigNr() == signal::IO_EVENT || sigNr() == signal::POLL) {
164 return PollData{
165 PollData::Reason{m_raw.ssi_code},
166 FileNum{m_raw.ssi_fd},
167 /* there are conflicting types, PollEvents is short
168 * but in sigaction it's a long */
169 PollEvents{static_cast<short>(m_raw.ssi_band)}
170 };
171 }
172
173 return std::nullopt;
174}
175
176} // end ns
Specialized exception type used when system APIs fail.
Definition ApiError.hxx:18
FileNum raw() const
Returns the primitive file descriptor contained in the object.
void setFD(const FileNum fd)
Assigns a new primitive file descriptor to the object.
Exception type for generic runtime errors.
Source
The source of a signal.
Definition SigInfo.hxx:83
@ USER
sent via kill().
@ MESGQ
POSIX message queue state changed.
@ TIMER
POSIX timer expired.
@ QUEUE
sent from user space via sigqueue().
A bit set of signal numbers for use in system calls.
Definition SigSet.hxx:25
sigset_t * raw()
Returns a pointer to the raw sigset_t data structure for use in API calls.
Definition SigSet.hxx:58
SigInfo style data structure returned by SignalFD::readEvent().
Definition SignalFD.hxx:116
std::optional< const UserSigData > userSigData() const
Returns the Source::USER specific data.
Definition SignalFD.cxx:81
bool isTrustedSource() const
Returns whether the signal was sent from a trusted source (i.e. the kernel).
Definition SignalFD.hxx:170
Errno error() const
Returns an error code that is generally unused on Linux (always 0).
Definition SignalFD.hxx:215
Source source() const
Returns the source of the signal.
Definition SignalFD.cxx:65
Signal sigNr() const
Returns the signal number that occurred.
Definition SignalFD.hxx:155
std::optional< const MsgQueueData > msgQueueData() const
Returns the Source::MSGQ specific data.
Definition SignalFD.cxx:104
std::optional< const TimerData > timerData() const
Returns the Source::TIMER specific data.
Definition SignalFD.cxx:114
std::optional< const PollData > pollData() const
Returns signal::POLL specific data.
Definition SignalFD.cxx:161
std::optional< const SysData > sysData() const
Returns signal::BAD_SYS specific data.
Definition SignalFD.cxx:127
std::optional< const QueueSigData > queueSigData() const
Returns the Source::QUEUE specific data.
Definition SignalFD.cxx:89
std::optional< const ChildData > childData() const
Returns signal::CHILD specific data.
Definition SignalFD.cxx:142
void readEvent(Info &info)
Reads the next event event from the SignalFD.
Definition SignalFD.cxx:49
auto fd()
Returns the FileDescriptor object associated with the SignalFD.
Definition SignalFD.hxx:98
void adjustMask(const SigSet &mask)
Change the signals the file descriptor is listening for.
Definition SignalFD.cxx:35
void create(const SigSet &mask)
Creates a new SignalFD.
Definition SignalFD.cxx:25
Represents a POSIX signal number and offers a minimal API around it.
Definition types.hxx:96
Exception type for logical usage errors within the application.
FileNum
Primitive file descriptor.
Definition types.hxx:32
ExitStatus
Represents an exit status code from a child process.
Definition types.hxx:43
SignalNr
A primitive signal number specification.
Definition types.hxx:50
Arch
System call ABI architecture.
Definition ptrace.hxx:174
Additional data found in SigInfo with SIGCHILD.
Definition SigInfo.hxx:321
Event
Types of SIGCHLD events that can occur.
Definition SigInfo.hxx:325
Additional custom SigInfo data.
Definition SigInfo.hxx:116
Additional data found in SigInfo with Source::MESGQ.
Definition SigInfo.hxx:162
Additional data found in SigInfo with SIGPOLL.
Definition SigInfo.hxx:446
Reason
Different reasons for delivering SIGPOLL.
Definition SigInfo.hxx:450
Additional data found in SigInfo with Source::QUEUE.
Definition SigInfo.hxx:153
Additional data found in SigInfo delivered with SIGSYS.
Definition SigInfo.hxx:422
Reason
Different reasons for delivering SIGYS.
Definition SigInfo.hxx:426
Additional data found in SigInfo with Source::TIMER.
Definition SigInfo.hxx:176
Additional data found in SigInfo with Source::USER.
Definition SigInfo.hxx:147
ClockTicks
Type used to express time in clock ticks unit in some APIs.
Definition types.hxx:25
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