{-# LANGUAGE CApiFFI, CPP, DeriveDataTypeable, NondecreasingIndentation #-} {-# OPTIONS_GHC -fno-cse #-} -- global variables {-# LANGUAGE Trustworthy #-} ----------------------------------------------------------------------------- -- | -- Module : System.Posix.Signals -- Copyright : (c) The University of Glasgow 2002 -- License : BSD-style (see the file libraries/base/LICENSE) -- -- Maintainer : libraries@haskell.org -- Stability : provisional -- Portability : non-portable (requires POSIX) -- -- POSIX signal support -- ----------------------------------------------------------------------------- #include "HsUnixConfig.h" ##include "HsUnixConfig.h" #ifdef HAVE_SIGNAL_H #include #endif module System.Posix.Signals ( -- * The Signal type Signal, -- * Specific signals nullSignal, internalAbort, sigABRT, realTimeAlarm, sigALRM, busError, sigBUS, processStatusChanged, sigCHLD, continueProcess, sigCONT, floatingPointException, sigFPE, lostConnection, sigHUP, illegalInstruction, sigILL, keyboardSignal, sigINT, killProcess, sigKILL, openEndedPipe, sigPIPE, keyboardTermination, sigQUIT, segmentationViolation, sigSEGV, softwareStop, sigSTOP, softwareTermination, sigTERM, keyboardStop, sigTSTP, backgroundRead, sigTTIN, backgroundWrite, sigTTOU, userDefinedSignal1, sigUSR1, userDefinedSignal2, sigUSR2, #if CONST_SIGPOLL != -1 pollableEvent, sigPOLL, #endif profilingTimerExpired, sigPROF, badSystemCall, sigSYS, breakpointTrap, sigTRAP, urgentDataAvailable, sigURG, virtualTimerExpired, sigVTALRM, cpuTimeLimitExceeded, sigXCPU, fileSizeLimitExceeded, sigXFSZ, -- * Sending signals raiseSignal, signalProcess, signalProcessGroup, -- * Handling signals Handler(Default,Ignore,Catch,CatchOnce,CatchInfo,CatchInfoOnce), SignalInfo(..), SignalSpecificInfo(..), installHandler, -- * Signal sets SignalSet, emptySignalSet, fullSignalSet, reservedSignals, addSignal, deleteSignal, inSignalSet, -- * The process signal mask getSignalMask, setSignalMask, blockSignals, unblockSignals, -- * The alarm timer scheduleAlarm, -- * Waiting for signals getPendingSignals, awaitSignal, -- * The @NOCLDSTOP@ flag setStoppedChildFlag, queryStoppedChildFlag, -- MISSING FUNCTIONALITY: -- sigaction(), (inc. the sigaction structure + flags etc.) -- the siginfo structure -- sigaltstack() -- sighold, sigignore, sigpause, sigrelse, sigset -- siginterrupt ) where import Data.Word import Foreign.C import Foreign.ForeignPtr import Foreign.Marshal import Foreign.Ptr import Foreign.Storable import System.IO.Unsafe (unsafePerformIO) import System.Posix.Types import System.Posix.Internals import System.Posix.Process import System.Posix.Process.Internals import Data.Dynamic ##include "rts/Signals.h" import GHC.Conc hiding (Signal) -- ----------------------------------------------------------------------------- -- Specific signals nullSignal :: Signal nullSignal = 0 -- | Process abort signal. sigABRT :: CInt sigABRT = CONST_SIGABRT -- | Alarm clock. sigALRM :: CInt sigALRM = CONST_SIGALRM -- | Access to an undefined portion of a memory object. sigBUS :: CInt sigBUS = CONST_SIGBUS -- | Child process terminated, stopped, or continued. sigCHLD :: CInt sigCHLD = CONST_SIGCHLD -- | Continue executing, if stopped. sigCONT :: CInt sigCONT = CONST_SIGCONT -- | Erroneous arithmetic operation. sigFPE :: CInt sigFPE = CONST_SIGFPE -- | Hangup. sigHUP :: CInt sigHUP = CONST_SIGHUP -- | Illegal instruction. sigILL :: CInt sigILL = CONST_SIGILL -- | Terminal interrupt signal. sigINT :: CInt sigINT = CONST_SIGINT -- | Kill (cannot be caught or ignored). sigKILL :: CInt sigKILL = CONST_SIGKILL -- | Write on a pipe with no one to read it. sigPIPE :: CInt sigPIPE = CONST_SIGPIPE -- | Terminal quit signal. sigQUIT :: CInt sigQUIT = CONST_SIGQUIT -- | Invalid memory reference. sigSEGV :: CInt sigSEGV = CONST_SIGSEGV -- | Stop executing (cannot be caught or ignored). sigSTOP :: CInt sigSTOP = CONST_SIGSTOP -- | Termination signal. sigTERM :: CInt sigTERM = CONST_SIGTERM -- | Terminal stop signal. sigTSTP :: CInt sigTSTP = CONST_SIGTSTP -- | Background process attempting read. sigTTIN :: CInt sigTTIN = CONST_SIGTTIN -- | Background process attempting write. sigTTOU :: CInt sigTTOU = CONST_SIGTTOU -- | User-defined signal 1. sigUSR1 :: CInt sigUSR1 = CONST_SIGUSR1 -- | User-defined signal 2. sigUSR2 :: CInt sigUSR2 = CONST_SIGUSR2 #if CONST_SIGPOLL != -1 -- | Pollable event. sigPOLL :: CInt sigPOLL = CONST_SIGPOLL #endif -- | Profiling timer expired. sigPROF :: CInt sigPROF = CONST_SIGPROF -- | Bad system call. sigSYS :: CInt sigSYS = CONST_SIGSYS -- | Trace/breakpoint trap. sigTRAP :: CInt sigTRAP = CONST_SIGTRAP -- | High bandwidth data is available at a socket. sigURG :: CInt sigURG = CONST_SIGURG -- | Virtual timer expired. sigVTALRM :: CInt sigVTALRM = CONST_SIGVTALRM -- | CPU time limit exceeded. sigXCPU :: CInt sigXCPU = CONST_SIGXCPU -- | File size limit exceeded. sigXFSZ :: CInt sigXFSZ = CONST_SIGXFSZ -- | Alias for 'sigABRT'. internalAbort ::Signal internalAbort = sigABRT -- | Alias for 'sigALRM'. realTimeAlarm :: Signal realTimeAlarm = sigALRM -- | Alias for 'sigBUS'. busError :: Signal busError = sigBUS -- | Alias for 'sigCHLD'. processStatusChanged :: Signal processStatusChanged = sigCHLD -- | Alias for 'sigCONT'. continueProcess :: Signal continueProcess = sigCONT -- | Alias for 'sigFPE'. floatingPointException :: Signal floatingPointException = sigFPE -- | Alias for 'sigHUP'. lostConnection :: Signal lostConnection = sigHUP -- | Alias for 'sigILL'. illegalInstruction :: Signal illegalInstruction = sigILL -- | Alias for 'sigINT'. keyboardSignal :: Signal keyboardSignal = sigINT -- | Alias for 'sigKILL'. killProcess :: Signal killProcess = sigKILL -- | Alias for 'sigPIPE'. openEndedPipe :: Signal openEndedPipe = sigPIPE -- | Alias for 'sigQUIT'. keyboardTermination :: Signal keyboardTermination = sigQUIT -- | Alias for 'sigSEGV'. segmentationViolation :: Signal segmentationViolation = sigSEGV -- | Alias for 'sigSTOP'. softwareStop :: Signal softwareStop = sigSTOP -- | Alias for 'sigTERM'. softwareTermination :: Signal softwareTermination = sigTERM -- | Alias for 'sigTSTP'. keyboardStop :: Signal keyboardStop = sigTSTP -- | Alias for 'sigTTIN'. backgroundRead :: Signal backgroundRead = sigTTIN -- | Alias for 'sigTTOU'. backgroundWrite :: Signal backgroundWrite = sigTTOU -- | Alias for 'sigUSR1'. userDefinedSignal1 :: Signal userDefinedSignal1 = sigUSR1 -- | Alias for 'sigUSR2'. userDefinedSignal2 :: Signal userDefinedSignal2 = sigUSR2 #if CONST_SIGPOLL != -1 -- | Alias for 'sigPOLL'. pollableEvent :: Signal pollableEvent = sigPOLL #endif -- | Alias for 'sigPROF'. profilingTimerExpired :: Signal profilingTimerExpired = sigPROF -- | Alias for 'sigSYS'. badSystemCall :: Signal badSystemCall = sigSYS -- | Alias for 'sigTRAP'. breakpointTrap :: Signal breakpointTrap = sigTRAP -- | Alias for 'sigURG'. urgentDataAvailable :: Signal urgentDataAvailable = sigURG -- | Alias for 'sigVTALRM'. virtualTimerExpired :: Signal virtualTimerExpired = sigVTALRM -- | Alias for 'sigXCPU'. cpuTimeLimitExceeded :: Signal cpuTimeLimitExceeded = sigXCPU -- | Alias for 'sigXFSZ'. fileSizeLimitExceeded :: Signal fileSizeLimitExceeded = sigXFSZ -- ----------------------------------------------------------------------------- -- Signal-related functions -- | @signalProcess int pid@ calls @kill@ to signal process @pid@ -- with interrupt signal @int@. signalProcess :: Signal -> ProcessID -> IO () signalProcess sig pid = throwErrnoIfMinus1_ "signalProcess" (c_kill pid sig) foreign import ccall unsafe "kill" c_kill :: CPid -> CInt -> IO CInt -- | @signalProcessGroup int pgid@ calls @kill@ to signal -- all processes in group @pgid@ with interrupt signal @int@. signalProcessGroup :: Signal -> ProcessGroupID -> IO () signalProcessGroup sig pgid = throwErrnoIfMinus1_ "signalProcessGroup" (c_killpg pgid sig) foreign import ccall unsafe "killpg" c_killpg :: CPid -> CInt -> IO CInt -- | @raiseSignal int@ calls @kill@ to signal the current process -- with interrupt signal @int@. raiseSignal :: Signal -> IO () raiseSignal sig = throwErrnoIfMinus1_ "raiseSignal" (c_raise sig) -- See also note in GHC's rts/RtsUtils.c -- This is somewhat fragile because we need to keep the -- `#if`-conditional in sync with GHC's runtime. #if (defined(openbsd_HOST_OS) || defined(freebsd_HOST_OS) || defined(dragonfly_HOST_OS) || defined(netbsd_HOST_OS) || defined(darwin_HOST_OS)) foreign import ccall unsafe "genericRaise" c_raise :: CInt -> IO CInt #else foreign import ccall unsafe "raise" c_raise :: CInt -> IO CInt #endif type Signal = CInt -- | The actions to perform when a signal is received. data Handler = Default | Ignore -- not yet: | Hold | Catch (IO ()) | CatchOnce (IO ()) | CatchInfo (SignalInfo -> IO ()) -- ^ @since 2.7.0.0 | CatchInfoOnce (SignalInfo -> IO ()) -- ^ @since 2.7.0.0 deriving (Typeable) -- | Information about a received signal (derived from @siginfo_t@). -- -- @since 2.7.0.0 data SignalInfo = SignalInfo { siginfoSignal :: Signal, siginfoError :: Errno, siginfoSpecific :: SignalSpecificInfo } -- | Information specific to a particular type of signal -- (derived from @siginfo_t@). -- -- @since 2.7.0.0 data SignalSpecificInfo = NoSignalSpecificInfo | SigChldInfo { siginfoPid :: ProcessID, siginfoUid :: UserID, siginfoStatus :: ProcessStatus } -- | @installHandler int handler iset@ calls @sigaction@ to install an -- interrupt handler for signal @int@. If @handler@ is @Default@, -- @SIG_DFL@ is installed; if @handler@ is @Ignore@, @SIG_IGN@ is -- installed; if @handler@ is @Catch action@, a handler is installed -- which will invoke @action@ in a new thread when (or shortly after) the -- signal is received. -- If @iset@ is @Just s@, then the @sa_mask@ of the @sigaction@ structure -- is set to @s@; otherwise it is cleared. The previously installed -- signal handler for @int@ is returned installHandler :: Signal -> Handler -> Maybe SignalSet -- ^ other signals to block -> IO Handler -- ^ old handler #ifdef __PARALLEL_HASKELL__ installHandler = error "installHandler: not available for Parallel Haskell" #else installHandler sig handler _maybe_mask = do ensureIOManagerIsRunning -- for the threaded RTS -- if we're setting the action to DFL or IGN, we should do that *first* -- if we're setting a handler, -- if the previous action was handle, then setHandler is ok -- if the previous action was IGN/DFL, then setHandler followed by sig_install (old_action, old_handler) <- case handler of Ignore -> do old_action <- stg_sig_install sig STG_SIG_IGN nullPtr old_handler <- setHandler sig Nothing return (old_action, old_handler) Default -> do old_action <- stg_sig_install sig STG_SIG_DFL nullPtr old_handler <- setHandler sig Nothing return (old_action, old_handler) _some_kind_of_catch -> do -- I don't think it's possible to get CatchOnce right. If -- there's a signal in flight, then we might run the handler -- more than once. let dyn = toDyn handler old_handler <- case handler of Catch action -> setHandler sig (Just (const action,dyn)) CatchOnce action -> setHandler sig (Just (const action,dyn)) CatchInfo action -> setHandler sig (Just (getinfo action,dyn)) CatchInfoOnce action -> setHandler sig (Just (getinfo action,dyn)) #if __GLASGOW_HASKELL__ < 811 _ -> error "installHandler" #endif let action = case handler of Catch _ -> STG_SIG_HAN CatchOnce _ -> STG_SIG_RST CatchInfo _ -> STG_SIG_HAN CatchInfoOnce _ -> STG_SIG_RST #if __GLASGOW_HASKELL__ < 811 _ -> error "installHandler" #endif old_action <- stg_sig_install sig action nullPtr -- mask is pointless, so leave it NULL return (old_action, old_handler) case (old_handler,old_action) of (_, STG_SIG_DFL) -> return $ Default (_, STG_SIG_IGN) -> return $ Ignore (Nothing, _) -> return $ Ignore (Just (_,dyn), _) | Just h <- fromDynamic dyn -> return h | Just io <- fromDynamic dyn -> return (Catch io) -- handlers put there by the base package have type IO () | otherwise -> return Default foreign import ccall unsafe stg_sig_install :: CInt -- sig no. -> CInt -- action code (STG_SIG_HAN etc.) -> Ptr CSigset -- (in, out) blocked -> IO CInt -- (ret) old action code getinfo :: (SignalInfo -> IO ()) -> ForeignPtr Word8 -> IO () getinfo handler fp_info = do si <- unmarshalSigInfo fp_info handler si unmarshalSigInfo :: ForeignPtr Word8 -> IO SignalInfo unmarshalSigInfo fp = do withForeignPtr fp $ \p -> do sig <- (#peek siginfo_t, si_signo) p errno <- (#peek siginfo_t, si_errno) p extra <- case sig of _ | sig == sigCHLD -> do pid <- (#peek siginfo_t, si_pid) p uid <- (#peek siginfo_t, si_uid) p wstat <- (#peek siginfo_t, si_status) p pstat <- decipherWaitStatus wstat return SigChldInfo { siginfoPid = pid, siginfoUid = uid, siginfoStatus = pstat } _ | otherwise -> return NoSignalSpecificInfo return SignalInfo { siginfoSignal = sig, siginfoError = Errno errno, siginfoSpecific = extra } #endif /* !__PARALLEL_HASKELL__ */ -- ----------------------------------------------------------------------------- -- Alarms -- | @scheduleAlarm i@ calls @alarm@ to schedule a real time -- alarm at least @i@ seconds in the future. scheduleAlarm :: Int -> IO Int scheduleAlarm secs = do r <- c_alarm (fromIntegral secs) return (fromIntegral r) foreign import ccall unsafe "alarm" c_alarm :: CUInt -> IO CUInt -- ----------------------------------------------------------------------------- -- The NOCLDSTOP flag foreign import ccall "&nocldstop" nocldstop :: Ptr Int -- | Tells the system whether or not to set the @SA_NOCLDSTOP@ flag when -- installing new signal handlers. setStoppedChildFlag :: Bool -> IO Bool setStoppedChildFlag b = do rc <- peek nocldstop poke nocldstop $ fromEnum (not b) return (rc == (0::Int)) -- | Queries the current state of the stopped child flag. queryStoppedChildFlag :: IO Bool queryStoppedChildFlag = do rc <- peek nocldstop return (rc == (0::Int)) -- ----------------------------------------------------------------------------- -- Manipulating signal sets newtype SignalSet = SignalSet (ForeignPtr CSigset) emptySignalSet :: SignalSet emptySignalSet = unsafePerformIO $ do fp <- mallocForeignPtrBytes sizeof_sigset_t throwErrnoIfMinus1_ "emptySignalSet" (withForeignPtr fp $ c_sigemptyset) return (SignalSet fp) fullSignalSet :: SignalSet fullSignalSet = unsafePerformIO $ do fp <- mallocForeignPtrBytes sizeof_sigset_t throwErrnoIfMinus1_ "fullSignalSet" (withForeignPtr fp $ c_sigfillset) return (SignalSet fp) -- | A set of signals reserved for use by the implementation. In GHC, this will normally -- include either `sigVTALRM` or `sigALRM`. reservedSignals :: SignalSet reservedSignals = addSignal rtsTimerSignal emptySignalSet foreign import ccall rtsTimerSignal :: CInt infixr `addSignal`, `deleteSignal` addSignal :: Signal -> SignalSet -> SignalSet addSignal sig (SignalSet fp1) = unsafePerformIO $ do fp2 <- mallocForeignPtrBytes sizeof_sigset_t withForeignPtr fp1 $ \p1 -> withForeignPtr fp2 $ \p2 -> do copyBytes p2 p1 sizeof_sigset_t throwErrnoIfMinus1_ "addSignal" (c_sigaddset p2 sig) return (SignalSet fp2) deleteSignal :: Signal -> SignalSet -> SignalSet deleteSignal sig (SignalSet fp1) = unsafePerformIO $ do fp2 <- mallocForeignPtrBytes sizeof_sigset_t withForeignPtr fp1 $ \p1 -> withForeignPtr fp2 $ \p2 -> do copyBytes p2 p1 sizeof_sigset_t throwErrnoIfMinus1_ "deleteSignal" (c_sigdelset p2 sig) return (SignalSet fp2) inSignalSet :: Signal -> SignalSet -> Bool inSignalSet sig (SignalSet fp) = unsafePerformIO $ withForeignPtr fp $ \p -> do r <- throwErrnoIfMinus1 "inSignalSet" (c_sigismember p sig) return (r /= 0) -- | @getSignalMask@ calls @sigprocmask@ to determine the -- set of interrupts which are currently being blocked. getSignalMask :: IO SignalSet getSignalMask = do fp <- mallocForeignPtrBytes sizeof_sigset_t withForeignPtr fp $ \p -> throwErrnoIfMinus1_ "getSignalMask" (c_sigprocmask 0 nullPtr p) return (SignalSet fp) sigProcMask :: String -> CInt -> SignalSet -> IO () sigProcMask fn how (SignalSet set) = withForeignPtr set $ \p_set -> throwErrnoIfMinus1_ fn (c_sigprocmask how p_set nullPtr) -- | @setSignalMask mask@ calls @sigprocmask@ with -- @SIG_SETMASK@ to block all interrupts in @mask@. setSignalMask :: SignalSet -> IO () setSignalMask set = sigProcMask "setSignalMask" (CONST_SIG_SETMASK :: CInt) set -- | @blockSignals mask@ calls @sigprocmask@ with -- @SIG_BLOCK@ to add all interrupts in @mask@ to the -- set of blocked interrupts. blockSignals :: SignalSet -> IO () blockSignals set = sigProcMask "blockSignals" (CONST_SIG_BLOCK :: CInt) set -- | @unblockSignals mask@ calls @sigprocmask@ with -- @SIG_UNBLOCK@ to remove all interrupts in @mask@ from the -- set of blocked interrupts. unblockSignals :: SignalSet -> IO () unblockSignals set = sigProcMask "unblockSignals" (CONST_SIG_UNBLOCK :: CInt) set -- | @getPendingSignals@ calls @sigpending@ to obtain -- the set of interrupts which have been received but are currently blocked. getPendingSignals :: IO SignalSet getPendingSignals = do fp <- mallocForeignPtrBytes sizeof_sigset_t withForeignPtr fp $ \p -> throwErrnoIfMinus1_ "getPendingSignals" (c_sigpending p) return (SignalSet fp) -- | @awaitSignal iset@ suspends execution until an interrupt is received. -- If @iset@ is @Just s@, @awaitSignal@ calls @sigsuspend@, installing -- @s@ as the new signal mask before suspending execution; otherwise, it -- calls @sigsuspend@ with current signal mask. Note that RTS -- scheduler signal (either 'virtualTimerExpired' or 'realTimeAlarm') -- could cause premature termination of this call. It might be necessary to block that -- signal before invocation of @awaitSignal@ with 'blockSignals' 'reservedSignals'. -- -- @awaitSignal@ returns when signal was received and processed by a -- signal handler, or if the signal could not be caught. If you have -- installed any signal handlers with @installHandler@, it may be wise -- to call @yield@ directly after @awaitSignal@ to ensure that the -- signal handler runs as promptly as possible. awaitSignal :: Maybe SignalSet -> IO () awaitSignal maybe_sigset = do fp <- case maybe_sigset of Nothing -> do SignalSet fp <- getSignalMask; return fp Just (SignalSet fp) -> return fp withForeignPtr fp $ \p -> do _ <- c_sigsuspend p return () -- ignore the return value; according to the docs it can only ever be -- (-1) with errno set to EINTR. -- XXX My manpage says it can also return EFAULT. And why is ignoring -- EINTR the right thing to do? foreign import ccall unsafe "sigsuspend" c_sigsuspend :: Ptr CSigset -> IO CInt #if defined(darwin_HOST_OS) && __GLASGOW_HASKELL__ < 706 -- see http://ghc.haskell.org/trac/ghc/ticket/7359#comment:3 -- To be removed when support for GHC 7.4.x is dropped foreign import ccall unsafe "__hscore_sigdelset" c_sigdelset :: Ptr CSigset -> CInt -> IO CInt foreign import ccall unsafe "__hscore_sigfillset" c_sigfillset :: Ptr CSigset -> IO CInt foreign import ccall unsafe "__hscore_sigismember" c_sigismember :: Ptr CSigset -> CInt -> IO CInt #else foreign import capi unsafe "signal.h sigdelset" c_sigdelset :: Ptr CSigset -> CInt -> IO CInt foreign import capi unsafe "signal.h sigfillset" c_sigfillset :: Ptr CSigset -> IO CInt foreign import capi unsafe "signal.h sigismember" c_sigismember :: Ptr CSigset -> CInt -> IO CInt #endif foreign import ccall unsafe "sigpending" c_sigpending :: Ptr CSigset -> IO CInt