/* signal.cc Copyright 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011 Red Hat, Inc. Written by Steve Chamberlain of Cygnus Support, sac@cygnus.com Significant changes by Sergey Okhapkin This file is part of Cygwin. This software is a copyrighted work licensed under the terms of the Cygwin license. Please consult the file "CYGWIN_LICENSE" for details. */ #include "winsup.h" #include #include #include "pinfo.h" #include "sigproc.h" #include "cygtls.h" #include "path.h" #include "fhandler.h" #include "dtable.h" #include "cygheap.h" int sigcatchers; /* FIXME: Not thread safe. */ #define _SA_NORESTART 0x8000 static int sigaction_worker (int, const struct sigaction *, struct sigaction *, bool) __attribute__ ((regparm (3))); #define sigtrapped(func) ((func) != SIG_IGN && (func) != SIG_DFL) static inline void set_sigcatchers (void (*oldsig) (int), void (*cursig) (int)) { #ifdef DEBUGGING int last_sigcatchers = sigcatchers; #endif if (!sigtrapped (oldsig) && sigtrapped (cursig)) sigcatchers++; else if (sigtrapped (oldsig) && !sigtrapped (cursig)) sigcatchers--; #ifdef DEBUGGING if (last_sigcatchers != sigcatchers) sigproc_printf ("last %d, old %d, cur %p, cur %p", last_sigcatchers, sigcatchers, oldsig, cursig); #endif } extern "C" _sig_func_ptr signal (int sig, _sig_func_ptr func) { sig_dispatch_pending (); _sig_func_ptr prev; /* check that sig is in right range */ if (sig < 0 || sig >= NSIG || sig == SIGKILL || sig == SIGSTOP) { set_errno (EINVAL); syscall_printf ("SIG_ERR = signal (%d, %p)", sig, func); return (_sig_func_ptr) SIG_ERR; } prev = global_sigs[sig].sa_handler; struct sigaction& gs = global_sigs[sig]; if (gs.sa_flags & _SA_NORESTART) gs.sa_flags &= ~SA_RESTART; else gs.sa_flags |= SA_RESTART; gs.sa_mask = SIGTOMASK (sig); gs.sa_handler = func; gs.sa_flags &= ~SA_SIGINFO; set_sigcatchers (prev, func); syscall_printf ("%p = signal (%d, %p)", prev, sig, func); return prev; } extern "C" int clock_nanosleep (clockid_t clk_id, int flags, const struct timespec *rqtp, struct timespec *rmtp) { const bool abstime = (flags & TIMER_ABSTIME) ? true : false; int res = 0; sig_dispatch_pending (); pthread_testcancel (); if (rqtp->tv_sec < 0 || rqtp->tv_nsec < 0 || rqtp->tv_nsec > 999999999L) return EINVAL; /* Explicitly disallowed by POSIX. Needs to be checked first to avoid being caught by the following test. */ if (clk_id == CLOCK_THREAD_CPUTIME_ID) return EINVAL; /* support for CPU-time clocks is optional */ if (CLOCKID_IS_PROCESS (clk_id) || CLOCKID_IS_THREAD (clk_id)) return ENOTSUP; switch (clk_id) { case CLOCK_REALTIME: case CLOCK_MONOTONIC: break; default: /* unknown or illegal clock ID */ return EINVAL; } LARGE_INTEGER timeout; timeout.QuadPart = (LONGLONG) rqtp->tv_sec * NSPERSEC + ((LONGLONG) rqtp->tv_nsec + 99LL) / 100LL; if (abstime) { struct timespec tp; clock_gettime (clk_id, &tp); /* Check for immediate timeout */ if (tp.tv_sec > rqtp->tv_sec || (tp.tv_sec == rqtp->tv_sec && tp.tv_nsec > rqtp->tv_nsec)) return 0; if (clk_id == CLOCK_REALTIME) timeout.QuadPart += FACTOR; else { /* other clocks need to be handled with a relative timeout */ timeout.QuadPart -= tp.tv_sec * NSPERSEC + tp.tv_nsec / 100LL; timeout.QuadPart *= -1LL; } } else /* !abstime */ timeout.QuadPart *= -1LL; syscall_printf ("clock_nanosleep (%ld.%09ld)", rqtp->tv_sec, rqtp->tv_nsec); int rc = cancelable_wait (signal_arrived, &timeout); if (rc == WAIT_OBJECT_0) { _my_tls.call_signal_handler (); res = EINTR; } /* according to POSIX, rmtp is used only if !abstime */ if (rmtp && !abstime) { rmtp->tv_sec = (time_t) (timeout.QuadPart / NSPERSEC); rmtp->tv_nsec = (long) ((timeout.QuadPart % NSPERSEC) * 100LL); } syscall_printf ("%d = clock_nanosleep (%lu, %d, %ld.%09ld, %ld.%09.ld)", res, clk_id, flags, rqtp->tv_sec, rqtp->tv_nsec, rmtp ? rmtp->tv_sec : 0, rmtp ? rmtp->tv_nsec : 0); return res; } extern "C" int nanosleep (const struct timespec *rqtp, struct timespec *rmtp) { int res = clock_nanosleep (CLOCK_REALTIME, 0, rqtp, rmtp); if (res != 0) { set_errno (res); return -1; } return 0; } extern "C" unsigned int sleep (unsigned int seconds) { struct timespec req, rem; req.tv_sec = seconds; req.tv_nsec = 0; if (clock_nanosleep (CLOCK_REALTIME, 0, &req, &rem)) return rem.tv_sec + (rem.tv_nsec > 0); return 0; } extern "C" unsigned int usleep (useconds_t useconds) { struct timespec req; req.tv_sec = useconds / 1000000; req.tv_nsec = (useconds % 1000000) * 1000; int res = clock_nanosleep (CLOCK_REALTIME, 0, &req, NULL); if (res != 0) { set_errno (res); return -1; } return 0; } extern "C" int sigprocmask (int how, const sigset_t *set, sigset_t *oldset) { int res = handle_sigprocmask (how, set, oldset, _my_tls.sigmask); if (res) set_errno (res); return res ? -1 : 0; } int __stdcall handle_sigprocmask (int how, const sigset_t *set, sigset_t *oldset, sigset_t& opmask) { /* check that how is in right range */ if (how != SIG_BLOCK && how != SIG_UNBLOCK && how != SIG_SETMASK) { syscall_printf ("Invalid how value %d", how); return EINVAL; } myfault efault; if (efault.faulted (EFAULT)) return EFAULT; if (oldset) *oldset = opmask; if (set) { sigset_t newmask = opmask; switch (how) { case SIG_BLOCK: /* add set to current mask */ newmask |= *set; break; case SIG_UNBLOCK: /* remove set from current mask */ newmask &= ~*set; break; case SIG_SETMASK: /* just set it */ newmask = *set; break; } set_signal_mask (newmask, opmask); } return 0; } int __stdcall _pinfo::kill (siginfo_t& si) { int res; DWORD this_process_state; pid_t this_pid; sig_dispatch_pending (); if (exists ()) { bool sendSIGCONT; this_process_state = process_state; if ((sendSIGCONT = (si.si_signo < 0))) si.si_signo = -si.si_signo; if (si.si_signo == 0) res = 0; else if ((res = sig_send (this, si))) { sigproc_printf ("%d = sig_send, %E ", res); res = -1; } else if (sendSIGCONT) { siginfo_t si2 = {0}; si2.si_signo = SIGCONT; si2.si_code = SI_KERNEL; sig_send (this, si2); } this_pid = pid; } else if (si.si_signo == 0 && this && process_state == PID_EXITED) { this_process_state = process_state; this_pid = pid; res = 0; } else { set_errno (ESRCH); this_process_state = 0; this_pid = 0; res = -1; } syscall_printf ("%d = _pinfo::kill (%d), pid %d, process_state %p", res, si.si_signo, this_pid, this_process_state); return res; } int raise (int sig) { return kill (myself->pid, sig); } static int kill0 (pid_t pid, siginfo_t& si) { syscall_printf ("kill (%d, %d)", pid, si.si_signo); /* check that sig is in right range */ if (si.si_signo < 0 || si.si_signo >= NSIG) { set_errno (EINVAL); syscall_printf ("signal %d out of range", si.si_signo); return -1; } return (pid > 0) ? pinfo (pid)->kill (si) : kill_pgrp (-pid, si); } int killsys (pid_t pid, int sig) { siginfo_t si = {0}; si.si_signo = sig; si.si_code = SI_KERNEL; return kill0 (pid, si); } int kill (pid_t pid, int sig) { siginfo_t si = {0}; si.si_signo = sig; si.si_code = SI_USER; return kill0 (pid, si); } int kill_pgrp (pid_t pid, siginfo_t& si) { int res = 0; int found = 0; int killself = 0; sigproc_printf ("pid %d, signal %d", pid, si.si_signo); winpids pids ((DWORD) PID_MAP_RW); for (unsigned i = 0; i < pids.npids; i++) { _pinfo *p = pids[i]; if (!p->exists ()) continue; /* Is it a process we want to kill? */ if ((pid == 0 && (p->pgid != myself->pgid || p->ctty != myself->ctty)) || (pid > 1 && p->pgid != pid) || (si.si_signo < 0 && NOTSTATE (p, PID_STOPPED))) continue; sigproc_printf ("killing pid %d, pgrp %d, p->%s, %s", p->pid, p->pgid, p->__ctty (), myctty ()); if (p == myself) killself++; else if (p->kill (si)) res = -1; found++; } if (killself && !exit_state && myself->kill (si)) res = -1; if (!found) { set_errno (ESRCH); res = -1; } syscall_printf ("%d = kill (%d, %d)", res, pid, si.si_signo); return res; } extern "C" int killpg (pid_t pgrp, int sig) { return kill (-pgrp, sig); } extern "C" void abort (void) { _my_tls.incyg++; sig_dispatch_pending (); /* Ensure that SIGABRT can be caught regardless of blockage. */ sigset_t sig_mask; sigfillset (&sig_mask); sigdelset (&sig_mask, SIGABRT); set_signal_mask (sig_mask, _my_tls.sigmask); raise (SIGABRT); _my_tls.call_signal_handler (); /* Call any signal handler */ /* Flush all streams as per SUSv2. */ if (_GLOBAL_REENT->__cleanup) _GLOBAL_REENT->__cleanup (_GLOBAL_REENT); do_exit (SIGABRT); /* signal handler didn't exit. Goodbye. */ } static int sigaction_worker (int sig, const struct sigaction *newact, struct sigaction *oldact, bool isinternal) { sig_dispatch_pending (); /* check that sig is in right range */ if (sig < 0 || sig >= NSIG) { set_errno (EINVAL); sigproc_printf ("signal %d, newact %p, oldact %p", sig, newact, oldact); syscall_printf ("SIG_ERR = sigaction signal %d out of range", sig); return -1; } struct sigaction oa = global_sigs[sig]; if (!newact) sigproc_printf ("signal %d, newact %p, oa %p", sig, newact, oa, oa.sa_handler); else { sigproc_printf ("signal %d, newact %p (handler %p), oa %p", sig, newact, newact->sa_handler, oa, oa.sa_handler); if (sig == SIGKILL || sig == SIGSTOP) { set_errno (EINVAL); return -1; } struct sigaction na = *newact; struct sigaction& gs = global_sigs[sig]; if (!isinternal) na.sa_flags &= ~_SA_INTERNAL_MASK; gs = na; if (!(gs.sa_flags & SA_NODEFER)) gs.sa_mask |= SIGTOMASK(sig); if (gs.sa_handler == SIG_IGN) sig_clear (sig); if (gs.sa_handler == SIG_DFL && sig == SIGCHLD) sig_clear (sig); set_sigcatchers (oa.sa_handler, gs.sa_handler); if (sig == SIGCHLD) { myself->process_state &= ~PID_NOCLDSTOP; if (gs.sa_flags & SA_NOCLDSTOP) myself->process_state |= PID_NOCLDSTOP; } } if (oldact) { *oldact = oa; oa.sa_flags &= ~_SA_INTERNAL_MASK; } return 0; } extern "C" int sigaction (int sig, const struct sigaction *newact, struct sigaction *oldact) { return sigaction_worker (sig, newact, oldact, false); } extern "C" int sigaddset (sigset_t *set, const int sig) { /* check that sig is in right range */ if (sig <= 0 || sig >= NSIG) { set_errno (EINVAL); syscall_printf ("SIG_ERR = sigaddset signal %d out of range", sig); return -1; } *set |= SIGTOMASK (sig); return 0; } extern "C" int sigdelset (sigset_t *set, const int sig) { /* check that sig is in right range */ if (sig <= 0 || sig >= NSIG) { set_errno (EINVAL); syscall_printf ("SIG_ERR = sigdelset signal %d out of range", sig); return -1; } *set &= ~SIGTOMASK (sig); return 0; } extern "C" int sigismember (const sigset_t *set, int sig) { /* check that sig is in right range */ if (sig <= 0 || sig >= NSIG) { set_errno (EINVAL); syscall_printf ("SIG_ERR = sigdelset signal %d out of range", sig); return -1; } if (*set & SIGTOMASK (sig)) return 1; else return 0; } extern "C" int sigemptyset (sigset_t *set) { *set = (sigset_t) 0; return 0; } extern "C" int sigfillset (sigset_t *set) { *set = ~((sigset_t) 0); return 0; } extern "C" int sigsuspend (const sigset_t *set) { return handle_sigsuspend (*set); } extern "C" int sigpause (int signal_mask) { return handle_sigsuspend ((sigset_t) signal_mask); } extern "C" int pause (void) { return handle_sigsuspend (_my_tls.sigmask); } extern "C" int siginterrupt (int sig, int flag) { struct sigaction act; sigaction (sig, NULL, &act); if (flag) { act.sa_flags &= ~SA_RESTART; act.sa_flags |= _SA_NORESTART; } else { act.sa_flags &= ~_SA_NORESTART; act.sa_flags |= SA_RESTART; } return sigaction_worker (sig, &act, NULL, true); } extern "C" int sigwait (const sigset_t *set, int *sig_ptr) { int sig = sigwaitinfo (set, NULL); if (sig > 0) *sig_ptr = sig; return sig > 0 ? 0 : -1; } extern "C" int sigwaitinfo (const sigset_t *set, siginfo_t *info) { pthread_testcancel (); HANDLE h; h = _my_tls.event = CreateEvent (&sec_none_nih, FALSE, FALSE, NULL); if (!h) { __seterrno (); return -1; } _my_tls.sigwait_mask = *set; sig_dispatch_pending (true); int res; switch (WaitForSingleObject (h, INFINITE)) { case WAIT_OBJECT_0: if (!sigismember (set, _my_tls.infodata.si_signo)) { set_errno (EINTR); res = -1; } else { if (info) *info = _my_tls.infodata; res = _my_tls.infodata.si_signo; InterlockedExchange ((LONG *) &_my_tls.sig, (LONG) 0); } break; default: __seterrno (); res = -1; } CloseHandle (h); sigproc_printf ("returning signal %d", res); return res; } /* FIXME: SUSv3 says that this function should block until the signal has actually been delivered. Currently, this will only happen when sending signals to the current process. It will not happen when sending signals to other processes. */ extern "C" int sigqueue (pid_t pid, int sig, const union sigval value) { siginfo_t si = {0}; pinfo dest (pid); if (!dest) { set_errno (ESRCH); return -1; } si.si_signo = sig; si.si_code = SI_QUEUE; si.si_value = value; return sig_send (dest, si); }