Signal (IPC) explained

Signals are standardized messages sent to a running program to trigger specific behavior, such as quitting or error handling. They are a limited form of inter-process communication (IPC), typically used in Unix, Unix-like, and other POSIX-compliant operating systems.

A signal is an asynchronous notification sent to a process or to a specific thread within the same process to notify it of an event. Common uses of signals are to interrupt, suspend, terminate or kill a process. Signals originated in 1970s Bell Labs Unix and were later specified in the POSIX standard.

When a signal is sent, the operating system interrupts the target process' normal flow of execution to deliver the signal. Execution can be interrupted during any non-atomic instruction. If the process has previously registered a signal handler, that routine is executed. Otherwise, the default signal handler is executed.

Embedded programs may find signals useful for inter-process communications, as signals are notable for their Algorithmic efficiency.

Signals are similar to interrupts, the difference being that interrupts are mediated by the CPU and handled by the kernel while signals are mediated by the kernel (possibly via system calls) and handled by individual processes. The kernel may pass an interrupt as a signal to the process that caused it (typical examples are SIGSEGV, SIGBUS, SIGILL and SIGFPE).

History

Sending signals

The system call sends a specified signal to a specified process, if permissions allow. Similarly, the command allows a user to send signals to processes. The library function sends the specified signal to the current process.

Exceptions such as division by zero, segmentation violation (SIGSEGV), and floating point exception (SIGFPE) will cause a core dump and terminate the program.

The kernel can generate signals to notify processes of events. For example, SIGPIPE will be generated when a process writes to a pipe which has been closed by the reader; by default, this causes the process to terminate, which is convenient when constructing shell pipelines.

Typing certain key combinations at the controlling terminal of a running process causes the system to send it certain signals:[3]

These default key combinations with modern operating systems can be changed with the command.

Handling signals

Signal handlers can be installed with the or system call. If a signal handler is not installed for a particular signal, the default handler is used. Otherwise the signal is intercepted and the signal handler is invoked. The process can also specify two default behaviors, without creating a handler: ignore the signal (SIG_IGN) and use the default signal handler (SIG_DFL). There are two signals which cannot be intercepted and handled: SIGKILL and SIGSTOP.

Risks

Signal handling is vulnerable to race conditions. As signals are asynchronous, another signal (even of the same type) can be delivered to the process during execution of the signal handling routine.

The call can be used to block and unblock delivery of signals. Blocked signals are not delivered to the process until unblocked. Signals that cannot be ignored (SIGKILL and SIGSTOP) cannot be blocked.

Signals can cause the interruption of a system call in progress, leaving it to the application to manage a non-transparent restart.

Signal handlers should be written in a way that does not result in any unwanted side-effects, e.g. alteration, signal mask alteration, signal disposition change, and other global process attribute changes. Use of non-reentrant functions, e.g., or, inside signal handlers is also unsafe. In particular, the POSIX specification and the Linux man page require that all system functions directly or indirectly called from a signal function are async-signal safe.[6] [7] The man page gives a list of such async-signal safe system functions (practically the system calls), otherwise it is an undefined behavior.[8] It is suggested to simply set some variable in a signal handler, and to test it elsewhere.[9]

Signal handlers can instead put the signal into a queue and immediately return. The main thread will then continue "uninterrupted" until signals are taken from the queue, such as in an event loop. "Uninterrupted" here means that operations that block may return prematurely and must be resumed, as mentioned above. Signals should be processed from the queue on the main thread and not by worker pools, as that reintroduces the problem of asynchronicity. However, managing a queue is not possible in an async-signal safe way with only, as only single reads and writes to such variables are guaranteed to be atomic, not increments or (fetch-and)-decrements, as would be required for a queue. Thus, effectively, only one signal per handler can be queued safely with until it has been processed.

Relationship with hardware exceptions

A process's execution may result in the generation of a hardware exception, for instance, if the process attempts to divide by zero or incurs a page fault.

In Unix-like operating systems, this event automatically changes the processor context to start executing a kernel exception handler. In case of some exceptions, such as a page fault, the kernel has sufficient information to fully handle the event itself and resume the process's execution.

Other exceptions, however, the kernel cannot process intelligently and it must instead defer the exception handling operation to the faulting process. This deferral is achieved via the signal mechanism, wherein the kernel sends to the process a signal corresponding to the current exception. For example, if a process attempted integer divide by zero on an x86 CPU, a divide error exception would be generated and cause the kernel to send the SIGFPE signal to the process.

Similarly, if the process attempted to access a memory address outside of its virtual address space, the kernel would notify the process of this violation via a SIGSEGV (segmentation violation signal). The exact mapping between signal names and exceptions is obviously dependent upon the CPU, since exception types differ between architectures.

POSIX signals

The list below documents the signals specified in the Single Unix Specification. All signals are defined as macro constants in the <signal.h> header file. The name of the macro constant consists of a "SIG" prefix followed by a mnemonic name for the signal.

and
  • "Signal abort", "signal input/output trap"
  • The SIGABRT signal is sent to a process to tell it to abort, i.e. to terminate. The signal is usually initiated by the process itself when it calls abort function of the C Standard Library, but it can be sent to the process from outside like any other signal.
  • SIGIOT indicates that the CPU has executed an explicit "trap" instruction (without a defined function), or an unimplemented instruction (when emulation is unavailable).
  • Note: "input/output trap" is a misnomer for any CPU "trap" instruction. The term reflects early usage of such instructions, predominantly to implement I/O functions, but they are not inherently tied to device I/O and may be used for other purposes such as communication between virtual & real hosts.
  • SIGIOT and SIGABRT are typically the same signal, and receipt of that signal may indicate any of the conditions above.
    , and
  • "Signal alarm", "signal virtual timer alarm", "signal profiling timer alarm"
  • The SIGALRM, SIGVTALRM and SIGPROF signals are sent to a process when the corresponding time limit is reached. The process sets these time limits by calling alarm or setitimer. The time limit for SIGALRM is based on real or clock time; SIGVTALRM is based on CPU time used by the process; and SIGPROF is based on CPU time used by the process and by the system on its behalf (known as a profiling timer). On some systems SIGALRM may be used internally by the implementation of the sleep function.
    "Signal bus"
  • The SIGBUS signal is sent to a process when it causes a bus error. The conditions that lead to the signal being sent are, for example, incorrect memory access alignment or non-existent physical address.
    "Signal child"
  • The SIGCHLD signal is sent to a process when a child process terminates, is stopped, or resumes after being stopped. One common usage of the signal is to instruct the operating system to clean up the resources used by a child process after its termination without an explicit call to the [[wait (system call)|wait]] system call.
    "Signal continue"
  • The SIGCONT signal instructs the operating system to continue (restart) a process previously paused by the SIGSTOP or SIGTSTP signal. One important use of this signal is in job control in the Unix shell.
    "Signal floating-point error"
  • The SIGFPE signal is sent to a process when an exceptional (but not necessarily erroneous) condition has been detected in the floating point or integer arithmetic hardware. This may include division by zero, floating point underflow or overflow, integer overflow, an invalid operation or an inexact computation. Behaviour may differ depending on hardware.
    "Signal hangup"
  • The SIGHUP signal is sent to a process when its controlling terminal is closed. It was originally designed to notify the process of a serial line drop (a hangup). In modern systems, this signal usually means that the controlling pseudo or virtual terminal has been closed.[10] Many daemons (who have no controlling terminal) interpret receipt of this signal as a request to reload their configuration files and flush/reopen their logfiles instead of exiting.[11] nohup is a command to make a command ignore the signal.
    "Signal illegal"
  • The SIGILL signal is sent to a process when it attempts to execute an illegal, malformed, unknown, or privileged instruction.
    "Signal interrupt"
  • The SIGINT signal is sent to a process by its controlling terminal when a user wishes to interrupt the process. This is typically initiated by pressing , but on some systems, the "delete" character or "break" key can be used.[12]
    "Signal kill"
  • The SIGKILL signal is sent to a process to cause it to terminate immediately (kill). In contrast to SIGTERM and SIGINT, this signal cannot be caught or ignored, and the receiving process cannot perform any clean-up upon receiving this signal. The following exceptions apply:

    SIGKILL is used as a last resort when terminating processes in most system shutdown procedures if it does not voluntarily exit in response to SIGTERM. To speed the computer shutdown procedure, Mac OS X 10.6, aka Snow Leopard, will send SIGKILL to applications that have marked themselves "clean" resulting in faster shutdown times with, presumably, no ill effects.[16] The command has a similar, while dangerous effect, when executed e.g. in Linux; it does not let programs save unsaved data. It has other options, and with none, uses the safer SIGTERM signal.

    "Signal pipe"
  • The SIGPIPE signal is sent to a process when it attempts to write to a pipe without a process connected to the other end.
    "Signal poll"
  • The SIGPOLL signal is sent when an event occurred on an explicitly watched file descriptor.[17] Using it effectively leads to making asynchronous I/O requests since the kernel will poll the descriptor in place of the caller. It provides an alternative to active polling.
    to
  • "Signal real-time minimum", "signal real-time maximum"
  • The SIGRTMIN to SIGRTMAX signals are intended to be used for user-defined purposes. They are real-time signals.
    "Signal quit"
  • The SIGQUIT signal is sent to a process by its controlling terminal when the user requests that the process quit and perform a core dump.
    "Signal segmentation violation"
  • The SIGSEGV signal is sent to a process when it makes an invalid virtual memory reference, or segmentation fault, i.e. when it performs a segmentation violation.[18]
    "Signal stop"
  • The SIGSTOP signal instructs the operating system to stop a process for later resumption.
    "Signal system call"
  • The SIGSYS signal is sent to a process when it passes a bad argument to a system call. In practice, this kind of signal is rarely encountered since applications rely on libraries (e.g. libc) to make the call for them. SIGSYS can be received by applications violating the Linux Seccomp security rules configured to restrict them. SIGSYS can also be used to emulate foreign system calls, e.g. emulate Windows system calls on Linux.[19]
    "Signal terminate"
  • The SIGTERM signal is sent to a process to request its termination. Unlike the SIGKILL signal, it can be caught and interpreted or ignored by the process. This allows the process to perform nice termination releasing resources and saving state if appropriate. SIGINT is nearly identical to SIGTERM.
    "Signal terminal stop"
  • The SIGTSTP signal is sent to a process by its controlling terminal to request it to stop (terminal stop). It is commonly initiated by the user pressing . Unlike SIGSTOP, the process can register a signal handler for, or ignore, the signal.
    and
  • The SIGTTIN and SIGTTOU signals are sent to a process when it attempts to read in or write out respectively from the tty while in the background. Typically, these signals are received only by processes under job control; daemons do not have controlling terminals and, therefore, should never receive these signals.
    "Signal trap"
  • The SIGTRAP signal is sent to a process when an exception (or trap) occurs: a condition that a debugger has requested to be informed of for example, when a particular function is executed, or when a particular variable changes value.
    "Signal urgent"
  • The SIGURG signal is sent to a process when a socket has urgent or out-of-band data available to read.
    and
  • "Signal user 1", "signal user 2""
  • The SIGUSR1 and SIGUSR2 signals are sent to a process to indicate user-defined conditions.
    "Signal exceeded CPU"
  • The SIGXCPU signal is sent to a process when it has used up the CPU for a duration that exceeds a certain predetermined user-settable value.[20] The arrival of a SIGXCPU signal provides the receiving process a chance to quickly save any intermediate results and to exit gracefully, before it is terminated by the operating system using the SIGKILL signal.
    "Signal excess file size"
  • The SIGXFSZ signal is sent to a process when it grows a file that exceeds the maximum allowed size.
    "Signal window change"
  • The SIGWINCH signal is sent to a process when its controlling terminal changes its size (a window change).[21]

    Default action

    A process can define how to handle incoming POSIX signals. If a process does not define a behaviour for a signal, then the default handler for that signal is being used. The table below lists some default actions for POSIX-compliant UNIX systems, such as FreeBSD, OpenBSD and Linux.

    Signal data-sort-type="number"Portable
    number !
    Default action Description
    SIGABRT 6 (core dump) Process abort signal
    SIGALRM 14 Alarm clock
    SIGBUS (core dump) Access to an undefined portion of a memory object
    SIGCHLD Ignore Child process terminated, stopped, or continued
    SIGCONT Continue Continue executing, if stopped
    SIGFPE 8 (core dump) Erroneous arithmetic operation
    SIGHUP 1 Hangup
    SIGILL 4 (core dump) Illegal instruction
    SIGINT 2 Terminal interrupt signal
    SIGKILL 9 Kill (cannot be caught or ignored)
    SIGPIPE 13 Write on a pipe with no one to read it
    SIGPOLL Pollable event
    SIGPROF Profiling timer expired
    SIGQUIT 3 (core dump) Terminal quit signal
    SIGSEGV 11 (core dump) Invalid memory reference
    SIGSTOP Stop executing (cannot be caught or ignored)
    SIGSYS (core dump) Bad system call
    SIGTERM 15 Termination signal
    SIGTRAP 5 (core dump) Trace/breakpoint trap
    SIGTSTP Terminal stop signal
    SIGTTIN Background process attempting read
    SIGTTOU Background process attempting write
    SIGUSR1 User-defined signal 1
    SIGUSR2 User-defined signal 2
    SIGURG Ignore Out-of-band data is available at a socket
    SIGVTALRM Virtual timer expired
    SIGXCPU (core dump) CPU time limit exceeded
    SIGXFSZ (core dump) File size limit exceeded
    SIGWINCH Ignore Terminal window size changed
    Portable number:
  • For most signals the corresponding signal number is implementation-defined. This column lists the numbers specified in the POSIX standard.[22]
    Actions explained:
  • Terminate Abnormal termination of the process. The process is terminated with all the consequences of _exit except that the status made available to wait and waitpid indicates abnormal termination by the specified signal.
  • Terminate (core dump) Abnormal termination of the process. Additionally, implementation-defined abnormal termination actions, such as creation of a core file, may occur.
  • Ignore Ignore the signal.
  • Stop Stop (or suspend) the process.
  • Continue Continue the process, if it is stopped; otherwise, ignore the signal.

    Miscellaneous signals

    The following signals are not specified in the POSIX specification. They are, however, sometimes used on various systems.

    The SIGEMT signal is sent to a process when an emulator trap occurs.
  • The SIGINFO signal is sent to a process when a status (info) request is received from the controlling terminal.
  • The SIGPWR signal is sent to a process when the system experiences a power failure.
  • The SIGLOST signal is sent to a process when a file lock is lost.
  • The SIGSTKFLT signal is sent to a process when the coprocessor experiences a stack fault (i.e. popping when the stack is empty or pushing when it is full).[23] It is defined by, but not used on Linux, where a x87 coprocessor stack fault will generate SIGFPE instead.[24]
  • The SIGUNUSED signal is sent to a process when a system call with an unused system call number is made. It is synonymous with SIGSYS on most architectures.[23]
  • The SIGCLD signal is synonymous with SIGCHLD.[23]
  • See also

    References

    External links

    Notes and References

    1. M. D. . McIlroy . Doug McIlroy . 1987 . A Research Unix reader: annotated excerpts from the Programmer's Manual, 1971–1986 . CSTR . 139 . Bell Labs.
    2. Web site: Gagliardi . Pietro . C Programming in Plan 9 from Bell Labs . doc.cat-v.org . 22 January 2022.
    3. Web site: Termination Signals . The GNU C Library).
    4. Web site: Job Control Signals . The GNU C Library.
    5. Web site: Miscellaneous Signals . The GNU C Library.
    6. Web site: The Open Group Base Specifications Issue 6, IEEE Std 1003.1, 2004 Edition: System Interfaces Chapter 2. 2020-12-20. pubs.opengroup.org.
    7. Web site: signal(7) - Linux manual page. 2020-12-20. man7.org.
    8. Web site: signal-safety(7) - Linux manual page. 2020-12-20. man7.org.
    9. Web site: The Open Group Base Specifications Issue 6, IEEE Std 1003.1, 2004 Edition: . 2020-12-20. pubs.opengroup.org.
    10. Web site: signal(7) . Linux Programmer's Manual (version 3.22). Michael Kerrisk . 25 July 2009. The Linux Kernel Archives. 23 September 2009.
    11. Web site: perlipc(1). Perl Programmers Reference Guide, version 5.18. perldoc.perl.org - Official documentation for the Perl programming language. 21 September 2013.
    12. Web site: Proper handling of SIGINT and SIGQUIT . 6 October 2012.
    13. https://manpages.ubuntu.com/manpages/zesty/man2/kill.2.html section NOTES
    14. Web site: SIGKILL init process (PID 1). Stack Overflow.
    15. Web site: Can root kill init process?. Unix & Linux Stack Exchange.
    16. Web site: Mac Dev Center: What's New in Mac OS X: Mac OS X v10.6 . 2009-08-28 . 18 November 2017.
    17. Web site: ioctl - controls a STREAM device . POSIX system call specification . . 19 June 2015.
    18. Web site: What is a "segmentation violation"?. support.microfocus.com. 2018-11-22.
    19. Web site: Syscall User Dispatch – The Linux Kernel documentation. 2021-02-11. kernel.org.
    20. Web site: getrlimit, setrlimit - control maximum resource consumption . POSIX system call specification . . 10 September 2009.
    21. Web site: 0001151: Introduce new signal SIGWINCH and functions tcsetsize, tcgetsize to get/set terminal window size . Clausecker . Robert . 2017-06-19 . Austin Group Defect Tracker . . 2017-10-12 . Accepted As Marked.
    22. Web site: IEEE Std 1003.1-2017 - kill. IEEE, Open Group. The correspondence between integer values and the sig value used is shown in the following list. The effects of specifying any signal_number other than those listed below are undefined..
    23. Web site: signal(7) — Linux manual pages. manpages.courier-mta.org. 2018-11-22.
    24. Web site: Linux 3.0 x86_64: When is SIGSTKFLT raised?. Stack Overflow.