use libc::{
c_int, pthread_kill, pthread_sigmask, pthread_t, sigaction, sigaddset, sigemptyset, siginfo_t,
sigismember, sigpending, sigset_t, sigtimedwait, timespec, EAGAIN, EINTR, EINVAL, SA_RESTART,
SIG_BLOCK, SIG_UNBLOCK,
};
use std::mem;
use std::os::unix::thread::JoinHandleExt;
use std::ptr::{null, null_mut};
use std::result;
use std::thread::JoinHandle;
use {errno, errno_result};
#[derive(Debug)]
pub enum Error {
CreateSigset(errno::Error),
SignalAlreadyBlocked(c_int),
CompareBlockedSignals(errno::Error),
BlockSignal(errno::Error),
RetrieveSignalMask(i32),
UnblockSignal(errno::Error),
ClearWaitPending(errno::Error),
ClearGetPending(errno::Error),
ClearCheckPending(errno::Error),
}
pub type SignalResult<T> = result::Result<T, Error>;
#[link(name = "c")]
extern "C" {
fn __libc_current_sigrtmin() -> c_int;
fn __libc_current_sigrtmax() -> c_int;
}
#[allow(non_snake_case)]
pub fn SIGRTMIN() -> c_int {
unsafe { __libc_current_sigrtmin() }
}
#[allow(non_snake_case)]
pub fn SIGRTMAX() -> c_int {
unsafe { __libc_current_sigrtmax() }
}
fn valid_signal_num(num: c_int) -> bool {
num >= SIGRTMIN() && num <= SIGRTMAX()
}
pub unsafe fn register_signal_handler(
num: c_int,
handler: extern "C" fn() -> (),
) -> errno::Result<()> {
if !valid_signal_num(num) {
return Err(errno::Error::new(EINVAL));
}
let mut sigact: sigaction = mem::zeroed();
sigact.sa_flags = SA_RESTART;
sigact.sa_sigaction = handler as *const () as usize;
let ret = sigaction(num, &sigact, null_mut());
if ret < 0 {
return errno_result();
}
Ok(())
}
pub fn create_sigset(signals: &[c_int]) -> errno::Result<sigset_t> {
let mut sigset: sigset_t = unsafe { mem::zeroed() };
let ret = unsafe { sigemptyset(&mut sigset) };
if ret < 0 {
return errno_result();
}
for signal in signals {
let ret = unsafe { sigaddset(&mut sigset, *signal) };
if ret < 0 {
return errno_result();
}
}
Ok(sigset)
}
pub fn get_blocked_signals() -> SignalResult<Vec<c_int>> {
let mut mask = Vec::new();
unsafe {
let mut old_sigset: sigset_t = mem::zeroed();
let ret = pthread_sigmask(SIG_BLOCK, null(), &mut old_sigset as *mut sigset_t);
if ret < 0 {
return Err(Error::RetrieveSignalMask(ret));
}
for num in 0..=SIGRTMAX() {
if sigismember(&old_sigset, num) > 0 {
mask.push(num);
}
}
}
Ok(mask)
}
pub fn block_signal(num: c_int) -> SignalResult<()> {
let sigset = create_sigset(&[num]).map_err(Error::CreateSigset)?;
unsafe {
let mut old_sigset: sigset_t = mem::zeroed();
let ret = pthread_sigmask(SIG_BLOCK, &sigset, &mut old_sigset as *mut sigset_t);
if ret < 0 {
return Err(Error::BlockSignal(errno::Error::last()));
}
let ret = sigismember(&old_sigset, num);
if ret < 0 {
return Err(Error::CompareBlockedSignals(errno::Error::last()));
} else if ret > 0 {
return Err(Error::SignalAlreadyBlocked(num));
}
}
Ok(())
}
pub fn unblock_signal(num: c_int) -> SignalResult<()> {
let sigset = create_sigset(&[num]).map_err(Error::CreateSigset)?;
let ret = unsafe { pthread_sigmask(SIG_UNBLOCK, &sigset, null_mut()) };
if ret < 0 {
return Err(Error::UnblockSignal(errno::Error::last()));
}
Ok(())
}
pub fn clear_signal(num: c_int) -> SignalResult<()> {
let sigset = create_sigset(&[num]).map_err(Error::CreateSigset)?;
while {
unsafe {
let mut siginfo: siginfo_t = mem::zeroed();
let ts = timespec {
tv_sec: 0,
tv_nsec: 0,
};
let ret = sigtimedwait(&sigset, &mut siginfo, &ts);
if ret < 0 {
let e = errno::Error::last();
match e.errno() {
EAGAIN | EINTR => {}
_ => {
return Err(Error::ClearWaitPending(errno::Error::last()));
}
}
}
let mut chkset: sigset_t = mem::zeroed();
let ret = sigpending(&mut chkset);
if ret < 0 {
return Err(Error::ClearGetPending(errno::Error::last()));
}
let ret = sigismember(&chkset, num);
if ret < 0 {
return Err(Error::ClearCheckPending(errno::Error::last()));
}
ret != 0
}
} {}
Ok(())
}
pub unsafe trait Killable {
fn pthread_handle(&self) -> pthread_t;
fn kill(&self, num: c_int) -> errno::Result<()> {
if !valid_signal_num(num) {
return Err(errno::Error::new(EINVAL));
}
let ret = unsafe { pthread_kill(self.pthread_handle(), num) };
if ret < 0 {
return errno_result();
}
Ok(())
}
}
unsafe impl<T> Killable for JoinHandle<T> {
fn pthread_handle(&self) -> pthread_t {
self.as_pthread_t()
}
}