libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
SigInfo.cxx
1// cosmos
2#include <cosmos/error/UsageError.hxx>
3#include <cosmos/proc/SigInfo.hxx>
4#include <cosmos/proc/signal.hxx>
5
6namespace cosmos {
7
8/* we want to be able to reinterpret_cast siginfo_t to SigInfo in async signal
9 * handlers */
10static_assert(sizeof(SigInfo) == sizeof(siginfo_t), "SigInfo size mismatch");
11
13 const std::initializer_list<Signal> SPECIAL_SIGS{
14 signal::FPE, signal::BUS, signal::ILL, signal::SEGV,
15 signal::BUS, signal::TRAP, signal::CHILD, signal::POLL,
16 signal::BAD_SYS};
17
18 // the lower level system call `rt_sigqueueinfo` allows userspace to
19 // send arbitrary codes < 0, thus the signal number alone doesn't mean
20 // the kernel was the source.
21 if (isTrustedSource() && in_list(this->sigNr(), SPECIAL_SIGS)) {
22 return Source::KERNEL;
23 }
24
25 return Source{m_raw.si_code};
26}
27
29 /*
30 * should we require isTrustedSource() as well here?
31 *
32 * I guess, formally yes, but there could be some situations
33 * like in emulation or test environments, where other user
34 * space processes send such signals.
35 *
36 * The kernel checks for privileges before it allows user
37 * space to send a signal to another process, so this is less
38 * of a security concern, more a concern of maintaining
39 * integrity. If user space sends these signals then we cannot
40 * know for sure that the data found in the siginfo_t is sane.
41 */
42 return in_list(sigNr(), {
43 signal::ILL, signal::FPE, signal::SEGV,
44 signal::BUS, signal::TRAP});
45}
46
47std::optional<const SigInfo::UserSigData> SigInfo::userSigData() const {
49 return UserSigData{this->procCtx()};
50 }
51
52 return std::nullopt;
53}
54
55std::optional<const SigInfo::QueueSigData> SigInfo::queueSigData() const {
57 return QueueSigData{this->procCtx(), CustomData{m_raw.si_value}};
58 }
59
60 return std::nullopt;
61}
62
63std::optional<const SigInfo::MsgQueueData> SigInfo::msgQueueData() const {
65 return MsgQueueData{this->procCtx(), CustomData{m_raw.si_value}};
66 }
67
68 return std::nullopt;
69}
70
71std::optional<const SigInfo::TimerData> SigInfo::timerData() const {
73 return TimerData{
74 TimerData::TimerID{m_raw.si_timerid},
75 m_raw.si_overrun
76 };
77 }
78
79 return std::nullopt;
80}
81
82std::optional<const SigInfo::ChildData> SigInfo::childData() const {
83 if (sigNr() == signal::CHILD) {
84 using Event = ChildData::Event;
85 const auto event = Event{m_raw.si_code};
86 /*
87 * since this siginfo_t is so hacky, let's hack it a bit on our own:
88 *
89 * the structure is reused in the waitid() system call, but
90 * not fully filled like when used with sigwaitinfo(). In this
91 * case there is no si_utime and no si_stime. To avoid
92 * duplicating this code just to leave out these two optionals
93 * the cosmos::proc::wait() sets si_errno to EINVAL (si_errno
94 * is also unused with waitid()).
95 */
96 const auto is_waitid_res = error() == Errno::INVALID_ARG;
97
98 return ChildData{
99 event,
100 this->procCtx(),
101 event == Event::EXITED ? std::make_optional(ExitStatus{m_raw.si_status}) : std::nullopt,
102 event == Event::EXITED ? std::nullopt : std::make_optional(Signal{SignalNr{m_raw.si_status}}),
103 is_waitid_res ? std::nullopt : std::make_optional(ClockTicks{m_raw.si_utime}),
104 is_waitid_res ? std::nullopt : std::make_optional(ClockTicks{m_raw.si_stime})
105 };
106 }
107
108 return std::nullopt;
109}
110
111std::optional<const SigInfo::SysData> SigInfo::sysData() const {
112 if (sigNr() == signal::BAD_SYS) {
113 return SysData{
114 SysData::Reason{m_raw.si_code},
115 m_raw.si_call_addr,
116 m_raw.si_syscall,
117 ptrace::Arch{m_raw.si_arch},
118 this->error()
119 };
120 }
121
122 return std::nullopt;
123}
124
125std::optional<const SigInfo::PollData> SigInfo::pollData() const {
126 // SIGIO and SIGPOLL are the same on Linux, but just to be sure ...
127 if (sigNr() == signal::IO_EVENT || sigNr() == signal::POLL) {
128 return PollData{
129 PollData::Reason{m_raw.si_code},
130 FileNum{m_raw.si_fd},
131 /* there are conflicting types, PollEvents is short
132 * but in sigaction it's a long */
133 PollEvents{static_cast<short>(m_raw.si_band)}
134 };
135 }
136
137 return std::nullopt;
138}
139
140std::optional<const SigInfo::IllData> SigInfo::illData() const {
141 if (sigNr() == signal::ILL) {
142 return IllData{
143 {m_raw.si_addr},
144 IllData::Reason{m_raw.si_code}
145 };
146 }
147
148 return std::nullopt;
149}
150
151std::optional<const SigInfo::FPEData> SigInfo::fpeData() const {
152 if (sigNr() == signal::FPE) {
153 return FPEData{
154 {m_raw.si_addr},
155 FPEData::Reason{m_raw.si_code}
156 };
157 }
158
159 return std::nullopt;
160}
161
162std::optional<const SigInfo::SegfaultData> SigInfo::segfaultData() const {
163 if (sigNr() == signal::SEGV) {
164 using Reason = SegfaultData::Reason;
165 const auto reason = Reason{m_raw.si_code};
166 return SegfaultData{
167 {m_raw.si_addr},
168 reason,
169 reason == Reason::BOUND_ERROR ?
170 std::make_optional(SegfaultData::Bound{m_raw.si_lower, m_raw.si_upper}) :
171 std::nullopt,
172 reason == Reason::PROT_KEY_ERROR ?
173 std::make_optional(SegfaultData::ProtectionKey{m_raw.si_pkey}) :
174 std::nullopt
175 };
176 }
177
178 return std::nullopt;
179}
180
181std::optional<const SigInfo::BusData> SigInfo::busData() const {
182 if (sigNr() == signal::BUS) {
183 using Reason = BusData::Reason;
184 const auto reason = Reason{m_raw.si_code};
185 return BusData{
186 {m_raw.si_addr},
187 reason,
188 in_list(reason, {Reason::MCE_ACTION_REQUIRED, Reason::MCE_ACTION_OPTIONAL}) ?
189 std::make_optional(m_raw.si_addr_lsb) :
190 std::nullopt
191 };
192 }
193
194 return std::nullopt;
195}
196
197} // end ns
std::optional< const ChildData > childData() const
Returns signal::CHILD specific data.
Definition SigInfo.cxx:82
bool isTrustedSource() const
Returns whether the signal was sent from a trusted source (i.e. the kernel).
Definition SigInfo.hxx:519
Source
The source of a signal.
Definition SigInfo.hxx:83
@ USER
sent via kill().
@ KERNEL
sent by the kernel.
@ MESGQ
POSIX message queue state changed.
@ TIMER
POSIX timer expired.
@ QUEUE
sent from user space via sigqueue().
Signal sigNr() const
Returns the signal number that occurred.
Definition SigInfo.hxx:484
std::optional< const QueueSigData > queueSigData() const
Returns the Source::QUEUE specific data.
Definition SigInfo.cxx:55
Errno error() const
Returns an error code that is generally unused on Linux (always 0).
Definition SigInfo.hxx:599
std::optional< const BusData > busData() const
Returns SIGBUS specific data.
Definition SigInfo.cxx:181
bool isFaultSignal() const
Returns whether the signal is one of the fault signals.
Definition SigInfo.cxx:28
std::optional< const SysData > sysData() const
Returns signal::BAD_SYS specific data.
Definition SigInfo.cxx:111
std::optional< const TimerData > timerData() const
Returns the Source::TIMER specific data.
Definition SigInfo.cxx:71
std::optional< const PollData > pollData() const
Returns signal::POLL specific data.
Definition SigInfo.cxx:125
std::optional< const IllData > illData() const
Returns SIGILL specific data.
Definition SigInfo.cxx:140
std::optional< const FPEData > fpeData() const
Returns SIGFPE specific data.
Definition SigInfo.cxx:151
std::optional< const MsgQueueData > msgQueueData() const
Returns the Source::MSGQ specific data.
Definition SigInfo.cxx:63
Source source() const
Returns the source of the signal.
Definition SigInfo.cxx:12
std::optional< const SegfaultData > segfaultData() const
Returns SIGSEGV specific data.
Definition SigInfo.cxx:162
std::optional< const UserSigData > userSigData() const
Returns the Source::USER specific data.
Definition SigInfo.cxx:47
Represents a POSIX signal number and offers a minimal API around it.
Definition types.hxx:96
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 delivered with SIGBUS signals.
Definition SigInfo.hxx:300
Reason
Different reasons for delivering a SIGBUS signal.
Definition SigInfo.hxx:304
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
Extra data delivered with SIGFPE signals.
Definition SigInfo.hxx:239
Reason
Different reasons for delivering floating-point exceptions.
Definition SigInfo.hxx:243
Additional data delivered with SIGILL signals.
Definition SigInfo.hxx:216
Reason
Different reasons for delivering a SIGILL signal.
Definition SigInfo.hxx:220
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 delivered with SIGSEGV signals.
Definition SigInfo.hxx:263
Reason
Different reasons for delivering a SIGSEGV signal.
Definition SigInfo.hxx:267
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