libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
cosmos::SigInfo Class Reference

Signal information struct used when receiving signals. More...

#include <SigInfo.hxx>

Classes

struct  BusData
 Additional data delivered with SIGBUS signals. More...
 
struct  ChildData
 Additional data found in SigInfo with SIGCHILD. More...
 
struct  CustomData
 Additional custom SigInfo data. More...
 
struct  FaultData
 Additional data found in SigInfo for one of the memory fault / trap signals. More...
 
struct  FPEData
 Extra data delivered with SIGFPE signals. More...
 
struct  IllData
 Additional data delivered with SIGILL signals. More...
 
struct  MsgQueueData
 Additional data found in SigInfo with Source::MESGQ. More...
 
struct  PollData
 Additional data found in SigInfo with SIGPOLL. More...
 
struct  ProcessCtx
 Information about the process a signal is from or about. More...
 
struct  QueueSigData
 Additional data found in SigInfo with Source::QUEUE. More...
 
struct  SegfaultData
 Additional data delivered with SIGSEGV signals. More...
 
struct  SysData
 Additional data found in SigInfo delivered with SIGSYS. More...
 
struct  TimerData
 Additional data found in SigInfo with Source::TIMER. More...
 
struct  UserSigData
 Additional data found in SigInfo with Source::USER. More...
 

Public Types

enum class  Source : int {
  USER = SI_USER , KERNEL = SI_KERNEL , QUEUE = SI_QUEUE , TIMER = SI_TIMER ,
  MESGQ = SI_MESGQ , ASYNCIO = SI_ASYNCIO , QSIGIO = SI_SIGIO , TKILL = SI_TKILL
}
 The source of a signal. More...
 

Public Member Functions

 SigInfo ()
 Creates a zero-initialized SigInfo wrapper.
 
 SigInfo (const no_init_t)
 Leaves the underlying data structure uninitialized.
 
Signal sigNr () const
 Returns the signal number that occurred.
 
Source source () const
 Returns the source of the signal.
 
bool isTrustedSource () const
 Returns whether the signal was sent from a trusted source (i.e. the kernel).
 
bool isFaultSignal () const
 Returns whether the signal is one of the fault signals.
 
std::optional< const UserSigDatauserSigData () const
 Returns the Source::USER specific data.
 
std::optional< const QueueSigDataqueueSigData () const
 Returns the Source::QUEUE specific data.
 
std::optional< const MsgQueueDatamsgQueueData () const
 Returns the Source::MSGQ specific data.
 
std::optional< const TimerDatatimerData () const
 Returns the Source::TIMER specific data.
 
std::optional< const SysDatasysData () const
 Returns signal::BAD_SYS specific data.
 
std::optional< const ChildDatachildData () const
 Returns signal::CHILD specific data.
 
std::optional< const PollDatapollData () const
 Returns signal::POLL specific data.
 
std::optional< const IllDataillData () const
 Returns SIGILL specific data.
 
std::optional< const FPEDatafpeData () const
 Returns SIGFPE specific data.
 
std::optional< const SegfaultDatasegfaultData () const
 Returns SIGSEGV specific data.
 
std::optional< const BusDatabusData () const
 Returns SIGBUS specific data.
 
void clear ()
 Zeroes out the low level siginfo_t data structure.
 
const siginfo_t * raw () const
 
siginfo_t * raw ()
 

Protected Member Functions

Errno error () const
 Returns an error code that is generally unused on Linux (always 0).
 
ProcessCtx procCtx () const
 
ProcessID pid () const
 
UserID uid () const
 

Protected Attributes

siginfo_t m_raw
 

Detailed Description

Signal information struct used when receiving signals.

This kernel data structure is union-like and most of its fields only have meaning - and sometimes different meanings - depending on the SigInfo::Source value. Thus most information is returned as a std::optional containing a separate SigInfo::<Kind>Data type which contains the specialized data relevant for the context.

TODO: check whether it makes sense to add TrapData for SIGTRAP. It is a bit unclear what exactly to expect, since ptrace(2) is insanely complex.

Definition at line 79 of file SigInfo.hxx.

Member Enumeration Documentation

◆ Source

enum class cosmos::SigInfo::Source : int
strong

The source of a signal.

Enumerator
USER 

sent via kill().

KERNEL 

sent by the kernel.

QUEUE 

sent from user space via sigqueue().

TIMER 

POSIX timer expired.

MESGQ 

POSIX message queue state changed.

ASYNCIO 

AIO completed.

QSIGIO 

queued SIGIO (only up to Linux 2.2).

TKILL 

sent by tkill() or tgkill().

Definition at line 83 of file SigInfo.hxx.

83 : int {
84 USER = SI_USER,
85 KERNEL = SI_KERNEL,
86 QUEUE = SI_QUEUE,
87 TIMER = SI_TIMER,
88 MESGQ = SI_MESGQ,
89 ASYNCIO = SI_ASYNCIO,
90 QSIGIO = SI_SIGIO,
91 TKILL = SI_TKILL,
92 };
@ ASYNCIO
AIO completed.
@ USER
sent via kill().
@ KERNEL
sent by the kernel.
@ QSIGIO
queued SIGIO (only up to Linux 2.2).
@ MESGQ
POSIX message queue state changed.
@ TIMER
POSIX timer expired.
@ TKILL
sent by tkill() or tgkill().
@ QUEUE
sent from user space via sigqueue().

Constructor & Destructor Documentation

◆ SigInfo() [1/2]

cosmos::SigInfo::SigInfo ( )
inline

Creates a zero-initialized SigInfo wrapper.

Definition at line 471 of file SigInfo.hxx.

471 {
472 clear();
473 }
void clear()
Zeroes out the low level siginfo_t data structure.
Definition SigInfo.hxx:581

◆ SigInfo() [2/2]

cosmos::SigInfo::SigInfo ( const no_init_t )
inline

Leaves the underlying data structure uninitialized.

When SigInfo is used as an output parameter only (the typical case) then you can invoke this constructor to avoid unnecessary zero-initialization.

Definition at line 481 of file SigInfo.hxx.

481{}

Member Function Documentation

◆ busData()

std::optional< const SigInfo::BusData > cosmos::SigInfo::busData ( ) const

Returns SIGBUS specific data.

This data is only available if sigNr() == signal::BUS.

Definition at line 181 of file SigInfo.cxx.

181 {
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}
Signal sigNr() const
Returns the signal number that occurred.
Definition SigInfo.hxx:484
Reason
Different reasons for delivering a SIGBUS signal.
Definition SigInfo.hxx:304
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

◆ childData()

std::optional< const SigInfo::ChildData > cosmos::SigInfo::childData ( ) const

Returns signal::CHILD specific data.

This data is only available for sigNr() == signal::CHILD.

Definition at line 82 of file SigInfo.cxx.

82 {
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}
Errno error() const
Returns an error code that is generally unused on Linux (always 0).
Definition SigInfo.hxx:599
SignalNr
A primitive signal number specification.
Definition types.hxx:50
Event
Different events that can occur in a tracee leading to ptrace-event-stop.
Definition ptrace.hxx:58
Event
Types of SIGCHLD events that can occur.
Definition SigInfo.hxx:325
ClockTicks
Type used to express time in clock ticks unit in some APIs.
Definition types.hxx:25

◆ clear()

void cosmos::SigInfo::clear ( )
inline

Zeroes out the low level siginfo_t data structure.

Definition at line 581 of file SigInfo.hxx.

581 {
582 zero_object(m_raw);
583 }
void zero_object(T &obj)
Completely overwrites the given object with zeroes.
Definition memory.hxx:23

◆ error()

Errno cosmos::SigInfo::error ( ) const
inlineprotected

Returns an error code that is generally unused on Linux (always 0).

An exception is the case of SIGSYS generated by seccomp(2) filters.

Definition at line 599 of file SigInfo.hxx.

599 {
600 return Errno{m_raw.si_errno};
601 }
Errno
Strong enum type representing errno error constants.
Definition errno.hxx:29

◆ fpeData()

std::optional< const SigInfo::FPEData > cosmos::SigInfo::fpeData ( ) const

Returns SIGFPE specific data.

This data is only available if sigNr() == signal::FPE.

Definition at line 151 of file SigInfo.cxx.

151 {
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}
Reason
Different reasons for delivering floating-point exceptions.
Definition SigInfo.hxx:243

◆ illData()

std::optional< const SigInfo::IllData > cosmos::SigInfo::illData ( ) const

Returns SIGILL specific data.

This data is only available if sigNr() == signal::ILL.

Definition at line 140 of file SigInfo.cxx.

140 {
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}
Reason
Different reasons for delivering a SIGILL signal.
Definition SigInfo.hxx:220

◆ isFaultSignal()

bool cosmos::SigInfo::isFaultSignal ( ) const

Returns whether the signal is one of the fault signals.

Definition at line 28 of file SigInfo.cxx.

28 {
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}

◆ isTrustedSource()

bool cosmos::SigInfo::isTrustedSource ( ) const
inline

Returns whether the signal was sent from a trusted source (i.e. the kernel).

Only the kernel is allowed to set an si_code >= 0. This is an indicator whether we can fully trust the integrity of the data contained in the siginfo_t.

An exception is when a process sends itself a signal, but this can also be considered a trusted source in all but very special cases (like executing untrusted code in another thread?).

Definition at line 519 of file SigInfo.hxx.

519 {
520 return m_raw.si_code >= 0;
521 }

◆ msgQueueData()

std::optional< const SigInfo::MsgQueueData > cosmos::SigInfo::msgQueueData ( ) const

Returns the Source::MSGQ specific data.

Definition at line 63 of file SigInfo.cxx.

63 {
65 return MsgQueueData{this->procCtx(), CustomData{m_raw.si_value}};
66 }
67
68 return std::nullopt;
69}
Source source() const
Returns the source of the signal.
Definition SigInfo.cxx:12

◆ pid()

ProcessID cosmos::SigInfo::pid ( ) const
inlineprotected

Definition at line 607 of file SigInfo.hxx.

607 {
608 return ProcessID{m_raw.si_pid};
609 }
ProcessID
Definition types.hxx:25

◆ pollData()

std::optional< const SigInfo::PollData > cosmos::SigInfo::pollData ( ) const

Returns signal::POLL specific data.

This data is only available for sigNr() == signal::POLL.

Definition at line 125 of file SigInfo.cxx.

125 {
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}
FileNum
Primitive file descriptor.
Definition types.hxx:32
BitMask< PollEvent > PollEvents
BitMask of PollEvent flags denoting the I/O status of a file.
Definition types.hxx:34
Reason
Different reasons for delivering SIGPOLL.
Definition SigInfo.hxx:450

◆ procCtx()

ProcessCtx cosmos::SigInfo::procCtx ( ) const
inlineprotected

Definition at line 603 of file SigInfo.hxx.

603 {
604 return ProcessCtx{pid(), uid()};
605 }

◆ queueSigData()

std::optional< const SigInfo::QueueSigData > cosmos::SigInfo::queueSigData ( ) const

Returns the Source::QUEUE specific data.

Definition at line 55 of file SigInfo.cxx.

55 {
57 return QueueSigData{this->procCtx(), CustomData{m_raw.si_value}};
58 }
59
60 return std::nullopt;
61}

◆ raw() [1/2]

siginfo_t * cosmos::SigInfo::raw ( )
inline

Definition at line 589 of file SigInfo.hxx.

589 {
590 return &m_raw;
591 }

◆ raw() [2/2]

const siginfo_t * cosmos::SigInfo::raw ( ) const
inline

Definition at line 585 of file SigInfo.hxx.

585 {
586 return &m_raw;
587 }

◆ segfaultData()

std::optional< const SigInfo::SegfaultData > cosmos::SigInfo::segfaultData ( ) const

Returns SIGSEGV specific data.

This data is only available if sigNr() == signal::SEGV.

Definition at line 162 of file SigInfo.cxx.

162 {
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}
Reason
Different reasons for delivering a SIGSEGV signal.
Definition SigInfo.hxx:267

◆ sigNr()

Signal cosmos::SigInfo::sigNr ( ) const
inline

Returns the signal number that occurred.

Definition at line 484 of file SigInfo.hxx.

484 {
485 return Signal{SignalNr{m_raw.si_signo}};
486 }

◆ source()

SigInfo::Source cosmos::SigInfo::source ( ) const

Returns the source of the signal.

For some special signals SigSource::KERNEL is implied if isTrustedSource() returns true. These special signals are SIGFPE, SIGBUS, SIGILL, SIGSEGV, SIGBUS, SIGTRAP, SIGCHILD, SIGPOLL/SIGIO and SIGSYS.

These signals use the si_code field for special data, but their source is the kernel, if isTrustedSource() returns true. Implying SigSource::KERNEL in these situations allows us to always return a value here instead of a std::optional, which could be empty in these cases.

Since other user space processes are allowed to set arbitrary Source values smaller than 0 it can happen that values outside the defined Source constants are returned here. The interpretation of source() is application specific in these cases (or should be ignored, if not expected).

Definition at line 12 of file SigInfo.cxx.

12 {
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}
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

◆ sysData()

std::optional< const SigInfo::SysData > cosmos::SigInfo::sysData ( ) const

Returns signal::BAD_SYS specific data.

This data is only available for sigNr() == signal::BAD_SYS.

Definition at line 111 of file SigInfo.cxx.

111 {
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}
Reason
Different reasons for delivering SIGYS.
Definition SigInfo.hxx:426

◆ timerData()

std::optional< const SigInfo::TimerData > cosmos::SigInfo::timerData ( ) const

Returns the Source::TIMER specific data.

Definition at line 71 of file SigInfo.cxx.

71 {
73 return TimerData{
74 TimerData::TimerID{m_raw.si_timerid},
75 m_raw.si_overrun
76 };
77 }
78
79 return std::nullopt;
80}

◆ uid()

UserID cosmos::SigInfo::uid ( ) const
inlineprotected

Definition at line 611 of file SigInfo.hxx.

611 {
612 return UserID{m_raw.si_uid};
613 }

◆ userSigData()

std::optional< const SigInfo::UserSigData > cosmos::SigInfo::userSigData ( ) const

Returns the Source::USER specific data.

Definition at line 47 of file SigInfo.cxx.

47 {
49 return UserSigData{this->procCtx()};
50 }
51
52 return std::nullopt;
53}

Member Data Documentation

◆ m_raw

siginfo_t cosmos::SigInfo::m_raw
protected

Definition at line 617 of file SigInfo.hxx.


The documentation for this class was generated from the following files: