Skip to content

Commit c1ee18d

Browse files
committed
Ability to look up names of signals
+ Incorporate to the debug output of Origin, which will look much nicer in the logs.
1 parent 1f271a0 commit c1ee18d

File tree

4 files changed

+121
-1
lines changed

4 files changed

+121
-1
lines changed

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
* `low_level::signal_name` to look up human readable name.
2+
* The `Origin`'s debug output now contains the human readable name of the
3+
signal.
4+
15
# 0.3.2
26

37
* Allow extracting Origin from the raw `siginfo_t` structure by hand, without

src/low_level/mod.rs

+3
Original file line numberDiff line numberDiff line change
@@ -12,9 +12,12 @@ pub mod channel;
1212
pub mod pipe;
1313
#[cfg(feature = "extended-siginfo-raw")]
1414
pub mod siginfo;
15+
mod signal_details;
1516

1617
pub use signal_hook_registry::{register, unregister};
1718

19+
pub use self::signal_details::signal_name;
20+
1821
/// The usual raise, just the safe wrapper around it.
1922
///
2023
/// This is async-signal-safe.

src/low_level/siginfo.rs

+21-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,12 @@
22
//!
33
//! See [`Origin`].
44
5+
use std::fmt::{Debug, Formatter, Result as FmtResult};
6+
57
use libc::{c_int, pid_t, siginfo_t, uid_t};
8+
9+
use crate::low_level;
10+
611
// Careful: make sure the signature and the constants match the C source
712
extern "C" {
813
fn sighook_signal_cause(info: &siginfo_t) -> ICause;
@@ -176,7 +181,7 @@ impl From<ICause> for Cause {
176181
///
177182
/// This is produced by the [`WithOrigin`] exfiltrator (or can be [extracted][Origin::extract] from
178183
/// `siginfo_t` by hand).
179-
#[derive(Clone, Debug, Eq, PartialEq)]
184+
#[derive(Clone, Eq, PartialEq)]
180185
#[non_exhaustive]
181186
pub struct Origin {
182187
/// The signal that happened.
@@ -204,6 +209,21 @@ pub struct Origin {
204209
pub cause: Cause,
205210
}
206211

212+
impl Debug for Origin {
213+
fn fmt(&self, fmt: &mut Formatter) -> FmtResult {
214+
fn named_signal(sig: c_int) -> String {
215+
low_level::signal_name(sig)
216+
.map(|n| format!("{} ({})", n, sig))
217+
.unwrap_or_else(|| sig.to_string())
218+
}
219+
fmt.debug_struct("Origin")
220+
.field("signal", &named_signal(self.signal))
221+
.field("process", &self.process)
222+
.field("cause", &self.cause)
223+
.finish()
224+
}
225+
}
226+
207227
impl Origin {
208228
/// Extracts the Origin from a raw `siginfo_t` structure.
209229
///

src/low_level/signal_details.rs

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
//! Providing auxiliary information for signals.
2+
3+
use libc::c_int;
4+
5+
use crate::consts::signal::*;
6+
7+
struct Details {
8+
signal: c_int,
9+
name: &'static str,
10+
}
11+
12+
macro_rules! s {
13+
($name: expr) => {
14+
Details {
15+
signal: $name,
16+
name: stringify!($name),
17+
}
18+
};
19+
}
20+
21+
#[cfg(not(windows))]
22+
const DETAILS: &[Details] = &[
23+
s!(SIGABRT),
24+
s!(SIGALRM),
25+
s!(SIGBUS),
26+
s!(SIGCHLD),
27+
s!(SIGCONT),
28+
s!(SIGFPE),
29+
s!(SIGHUP),
30+
s!(SIGILL),
31+
s!(SIGINT),
32+
s!(SIGIO),
33+
s!(SIGKILL),
34+
s!(SIGPIPE),
35+
s!(SIGPROF),
36+
s!(SIGQUIT),
37+
s!(SIGSEGV),
38+
s!(SIGSTOP),
39+
s!(SIGSYS),
40+
s!(SIGTERM),
41+
s!(SIGTRAP),
42+
s!(SIGTSTP),
43+
s!(SIGTTIN),
44+
s!(SIGTTOU),
45+
s!(SIGURG),
46+
s!(SIGUSR1),
47+
s!(SIGUSR2),
48+
s!(SIGVTALRM),
49+
s!(SIGWINCH),
50+
s!(SIGXCPU),
51+
s!(SIGXFSZ),
52+
];
53+
54+
#[cfg(windows)]
55+
const DETAILS: &[Details] = &[
56+
s!(SIGABRT),
57+
s!(SIGFPE),
58+
s!(SIGILL),
59+
s!(SIGINT),
60+
s!(SIGSEGV),
61+
s!(SIGTERM),
62+
];
63+
64+
/// Provides a human-readable name of a signal.
65+
///
66+
/// Note that the name does not have to be known (in case it is some less common, or non-standard
67+
/// signal).
68+
///
69+
/// # Examples
70+
///
71+
/// ```
72+
/// # use signal_hook::low_level::signal_name;
73+
/// assert_eq!("SIGKILL", signal_name(9).unwrap());
74+
/// assert!(signal_name(142).is_none());
75+
/// ```
76+
pub fn signal_name(signal: c_int) -> Option<&'static str> {
77+
DETAILS.iter().find(|d| d.signal == signal).map(|d| d.name)
78+
}
79+
80+
#[cfg(test)]
81+
mod test {
82+
use super::*;
83+
84+
#[test]
85+
fn existing() {
86+
assert_eq!("SIGTERM", signal_name(SIGTERM).unwrap());
87+
}
88+
89+
#[test]
90+
fn unknown() {
91+
assert!(signal_name(128).is_none());
92+
}
93+
}

0 commit comments

Comments
 (0)