Chapter 8 Signals Source: Robbins and Robbins, UNIX Systems Programming, Prentice Hall, 2003.

Slides:



Advertisements
Similar presentations
1 CS345 Operating Systems Φροντιστήριο Άσκησης 1.
Advertisements

CSCC69: Operating Systems
15-213, Fall 06 Outline Shell Lab Processes Signals.
15-213/ Intro to Computer Systems by btan with reference to Spring 10’s slides.
UNIX Process Control Bach 7 Operating Systems Course Hebrew University Spring 2007.
CPSC 451 Editors and Systems Calls1 Minix editors Mined - (mined) is a simple screen editor. Elle - (elle) is a clone of Emacs. Elvis - (elvis, ex, vi)
Process Process: the UNIX abstraction of a stand-along computer that manages resources (memory, CPU, I/O resources) comprising a running program. Processes.
CS Lecture 17 Outline Named pipes Signals Lecture 17
Signals Hua LiSystems ProgrammingCS2690Signals. Topics: Sending Signals -- kill(), raise() Signal Handling -- signal() sig_talk.c -- complete example.
Operating Systems Course Hebrew University Spring 2007 Signals & User Thread.
Slide 6-1 Copyright © 2004 Pearson Education, Inc. Operating Systems: A Modern Perspective, Chapter 6 Threads and Scheduling 6.
1 Signals COS Goals of Today’s Lecture Overview of signals  Notifications sent to a process  UNIX signal names and numbers  Ways to generate.
CS Lecture 16 Outline Inter-process Communication (IPC) – Pipes – Signals Lecture 161CS Operating Systems 1.
Signal Signal : - is a notification sent to a process to notify it of some event - interrupts whatever the process is doing and force it to handle a signal.
CSc 352 Signal Handling in Unix Saumya Debray Dept. of Computer Science The University of Arizona, Tucson
UNIX Signals Bach 7.2 Operating Systems Course The Hebrew University Spring 2010.
Signals & Timers CS241 Discussion Section Spring 2009 Week 6.
1Reference “Introduction To Unix Signals Programming” in the reference material section Man page – sigprocmask, alarm “Understanding the Linux Kernel”
Process. Processes A process is an abstraction for sequence of operations that implement a computation/program. A process may be manipulated, suspended,
Operating Systems Chapter 2
Lecture 3 Process Concepts. What is a Process? A process is the dynamic execution context of an executing program. Several processes may run concurrently,
Chapter 6 UNIX Special Files Source: Robbins and Robbins, UNIX Systems Programming, Prentice Hall, 2003.
* POSIX-Defined Signals * Signaling Processes * Signal Mask * sigaction * kill and sigaction * alarm Topics.
Agenda  Working with Processes: Purpose Running Programs within same process (execl, execlp, execle, execv, execvp, execve) “Spawning” other process (fork,
System calls for Process management
Scis.regis.edu ● CS 468: Advanced UNIX Class 5 Dr. Jesús Borrego Regis University 1.
Linux Processes Travis Willey Jeff Mihalik. What is a process? A process is a program in execution A process includes: –program counter –stack –data section.
Copyright ©: Nahrstedt, Angrave, Abdelzaher1 Processes and Threads.
UNIX Socket Programming CS 6378 Project Reference Book: Unix Network programming: Networking APIs: Sockets and XTI (2nd edition), Prentice Hall >> Threads.
Signals and Signal Processing CIS 370 Lab 7 Umass Dartmouth.
Outline for Today Objectives –Finish discussion of Birrell –UNIX Signals –Eraser Administrative –Spider talk after class.
Signals and Signal Handling. Signals A predefined message sent between two processes or from the kernel to a process, or by a user to a process A software.
Signals (Chap 10 in the book “Advanced Programming in the UNIX Environment”) Acknowledgement : Prof. Y. Moon at Kangwon Nat’l Univ.
NCHU System & Network Lab Lab #8 Signals Operating System Lab.
1 Signals (continued) CS 241 April 9, 2012 University of Illinois.
1 Computer Systems II Introduction to Processes. 2 First Two Major Computer System Evolution Steps Led to the idea of multiprogramming (multiple concurrent.
SignalsSignals. What is a Signal? A signal is a notification that some event has occurred. Usually a signal is sent to a process asynchronously and whatever.
UNIX Signals * POSIX-Defined Signals * Signaling Processes * Signal Mask * sigaction * kill and sigaction * alarm * Interval Timers * POSIX.1b Timers *
Operating Systems Recitation 4, April th, 2002 Signals.
CSC Advanced Unix Programming, Fall, 2008 Welcome back to UNIX System Programming! Monday, September 22, class 5.
1 UNIX System Programming Signals. 2 Overview 1. Definition 2. Signal Types 3. Generating a Signal 4. Responding to a Signal 5. Common Uses of Signals.
KUKUM Real Time System Module #3 POSIX programming Lecture 2.
© 숙대 창병모 1 제 10 장 신호 (Signal). © 숙대 창병모 2 Contents 1. Signal Concepts 2. signal() 3. Interrupted System Calls 4. kill() /raise() 5. alarm() pause() 6.
Today’s topic Environment variables Signal. The list of environment variables –try ‘env’ –Environment variables can be defined in shell setenv DISPLAY.
Today’s topics Signals and how to control the program behavior in handling signals. Terminal I/O.
S -1 Processes. S -2 wait and waitpid (11.2) Recall from a previous slide: pid_t wait( int *status ) wait() can: (a) block; (b) return with status; (c)
System calls for Process management Process creation, termination, waiting.
CS 241 Section Week #10 (04/01/10) ‏. Topics This Section  MP5 Overview  Signals.
1 Lecture 19: Unix signals and Terminal management n what is a signal n signal handling u kernel u user n signal generation n signal example usage n terminal.
Copyright ©: Nahrstedt, Angrave, Abdelzaher1 Signals.
Copyright ©: Nahrstedt, Angrave, Abdelzaher, Caccamo1 Signals.
CS241 Systems Programming Discussion Section Week 2 Original slides by: Stephen Kloder.
Slide 1 COMP 3438 System Programming UNIX Processes UNIX Processes (Chapters 2 & 3)
CS241 Systems Programming Discussion Section Week 2 Original slides by: Stephen Kloder.
Process Related System Calls By Neha Hulkoti & Kavya Bhat.
Mutual Exclusion -- Addendum. Mutual Exclusion in Critical Sections.
Operating Systems Practical Session 3, Signals 1.
Operating Systems Inter-Process Communication Signals Moti Geva
Recitation 7 – 3/18/02 Outline fork and scheduling Signals
G.Jyostna.
Signals What is a Signal?
Task Control: Signals and Alarms Chapter 7 and 8
Why exception handling in C++?
Unix Process Management
Example questions… Can a shell kill itself? Can a shell within a shell kill the parent shell? What happens to background processes when you exit from.
Tarek Abdelzaher Vikram Adve Marco Caccamo
Chapter 14 - Advanced C Topics
CSC Advanced Unix Programming, Fall 2015
Interrupt handling Explain how interrupts are used to obtain processor time and how processing of interrupted jobs may later be resumed, (typical.
Signals.
Presentation transcript:

Chapter 8 Signals Source: Robbins and Robbins, UNIX Systems Programming, Prentice Hall, 2003.

8.1 Basic Signal Concepts

3 Basic Signal Concepts A signal is a software notification to a process of an event A signal is generated when the event that causes the signal occurs A signal is delivered when the process takes action based on the signal The lifetime of a signal is the interval between its generation and its delivery A signal that has been generated but not yet delivered is said to be pending –There may be considerable time between signal generation and signal delivery –The process must be running on a computer at the time of signal delivery A program installs a signal handler by calling the sigaction() function with the name of a user-written function A process handles a signal if it executes a signal handler when the signal is delivered If the process is set to ignore a signal, that signal is thrown away when delivered and has no effect on the process

4 Basic Signal Concepts (continued)‏ The action taken when a signal is generated depends on the current signal handler for that signal and on the process signal mask The signal mask contains a list of currently blocked signals Blocked signals are not thrown away as ignored signals are If a signal is blocked, it is delivered when the process unblocks that signal A program blocks a signal by changing its process signal mask using sigprocmask()‏ A program ignores a signal by setting the signal handler to SIG_IGN using sigaction()

8.2 Generating Signals

6 Signal Names Every signal has a symbolic name starting with SIG The signal names are defined in the signal.h file, which should be included by any C program that uses signals The table on the following slide lists some of the POSIX signals and their default actions Two signals, SIGUSR1 and SIGUSR2, are available for users and do not have a preassigned use Some signals such as SIGFPE and SIGSEGV are generated when certain errors occur; other signals are generated by specific function calls such as SIGALRM by the alarm() function

7 CertainPOSIX Signals Abnormal terminationSegmentation fault11SIGSEGV Implementation dependentInteractive termination; core dump3SIGQUIT ContinueExecution continued if stopped25SIGCONT NBR Abnormal terminationUser-defined signal 2SIGUSR2 Abnormal terminationUser-defined signal 1SIGUSR1 Abnormal terminationTerminationSIGTERM StopExecution stop (cannot be caught or ignored)‏ SIGSTOP Abnormal terminationTermination (cannot be caught or ignored)‏ SIGKILL Abnormal terminationInteractive attention signal (Ctrl-C)‏SIGINT Implementation dependentArithmetic error (i.e., divide by zero)‏ SIGFPE Abnormal terminationalarm clockSIGALRM Implementation dependentProcess abort (sent by assert)‏SIGABRT DEFAULT ACTIONDESCRIPTIONSIGNAL

8 kill Command A user can send signals to a process from a command shell using the kill command The name of the command is a slight misnomer in that it can be used to send any of the POSIX signals, not just the ones that cause abnormal process termination The kill command sends a signal to the process or processes that are specified by each pid operand kill [-signal_name] pid... kill [-signal_number] pid... The first form of the kill command take a signal name as defined in signal.h without the SIG prefix (i.e., INT instead of SIGINT)‏ The second form takes a signal number If no signal name or number is specified, SIGTERM is sent by default Example usage kill 3425 kill –9 896 kill –KILL 3425 kill –INT 9328 kill –USR1 567

9 kill() Function In a program, a user can send a signal to a process using the kill() function #include int kill(pid_t pid, int signalNbr); The function takes a process ID and a signal number (i.e., a symbolic name) as parameters If successful, the function returns zero; otherwise, it returns –1 and sets errno A user may only send a signal to a process that he or she owns –For most signals, the kill() function determines permissions by comparing the user IDs of caller process and target process Example use of the kill() function int status; pid_t childPid;... status = kill(childPid, SIGUSR1); if (status == -1) perror("Failed to send the SIGUSR1 signal to child");

10 raise() Function A process can send a signal to itself with the raise() function #include int raise(int sig); The function takes just one parameter, a signal number If successful, the function returns zero; otherwise, it returns a nonzero error value and sets errno –The function sets errno to EINVAL if the signal number is invalid Example use of raise() function int status; status = raise(SIGUSR1); if (status != 0) perror("Failed to raise SIGUSR1");

11 alarm() Function The alarm() function causes a SIGALRM signal to be sent to the calling process after a specified number of real seconds has elapsed #include unsigned int alarm(unsigned int seconds); The function returns the number of seconds remaining on the alarm before the call resets the value, or zero if no previous alarm was set The function never reports an error Requests to alarm() are not stacked, so a call to alarm() before the previous timer expires causes the alarm to be reset to the new value If alarm() is called with a zero value argument, it cancels a previous alarm request The default action for SIGALRM is to terminate the process The following program runs for five seconds of wall-clock time and then terminates #include int main(void) { alarm(5); for ( ; ; ); return 0; } // End main

12 assert() Function The assert() function is used to put diagnostic checks into a program by means of an assertion expression that evaluates to true or false (i.e., a nonzero or a zero result) #include void assert(int expression); When the value of the expression passed to the assert() function is false, the function first prints a diagnostic message to stderr containing the assertion expression, the file name and the line number in the file where the assertion is located. It than calls the abort() function. An example output is shown below assertion “nbrOfValues <= 7" failed: file “sample-program.c", line [sig] a 3640 open_stackdumpfile: Dumping stack trace to a.exe.stackdump The abort() function sends the SIGABRT signal to the process (using raise() ) and never returns. At that point, the process either executes the default signal handler for SIGABRT or calls the signal handler registered by the process. In either case, the process terminates The default action for SIGABRT is to request a stack trace dump and then terminate If NDEBUG is defined in a program before the #include for assert.h, then any assert statements in the program are ignored by the compiler

13 Use of assert() #include // #define NDEBUG #include #define MIN_SCORES 2 #define MAX_SCORES 5 // ****************************************** int main(void)‏ { float average; int i; int score; float sum = 0; int nbrOfScores; int scoreTable[MAX_SCORES]; printf("Enter the number of scores (%d - %d: “, MIN_SCORES, MAX_SCORES); scanf("%d", &nbrOfScores); printf(“\n"); assert( (nbrOfScores >= MIN_SCORES) && (nbrOfScores <= MAX_SCORES) ); (More on next slide)‏

14 Use of assert() (continued) for (i = 0; i < nbrOfScores; i++)‏ { printf("Enter score #%d (x >= 0): ", i + 1); scanf("%d", &score); assert( (score >= 0) && (score < INT_MAX) ); assert( (i >= 0) && ( i < nbrOfScores) ); scoreTable[i] = score; sum = sum + score; } // End for printf("\nList of Scores: "); for (i = 0; i < nbrOfScores; i++)‏ printf(" %d ", scoreTable[i]); assert(sum >= 0); average = sum / nbrOfScores; assert(average >= 0); printf("\n\nAverage score: %.2f\n", average); return 0; } // End main

Sample Outputs ( with assert )‏ uxb3% a.out Enter the number of scores (2 - 5): 1 assert-demo.c:25: failed assertion `(nbrOfScores >= MIN_SCORES) && (nbrOfScores <= MAX_SCORES)' Abort uxb3% uxb3% a.out Enter the number of scores (2 - 5): 4 Enter score #1 (x >= 0): -1 assert-demo.c:32: failed assertion `(score >= 0) && (score < INT_MAX)' Abort uxb3% uxb3% a.out Enter the number of scores (2 - 5): 3 Enter score #1 (x >= 0): 89 Enter score #2 (x >= 0): 72 Enter score #3 (x >= 0): 84 List of Scores: Average score: uxb3%

Sample Output ( with no assert )‏ uxb2% a.out Enter the number of scores (2 - 5): 10 Enter score #1 (x >= 0): -5 Enter score #2 (x >= 0): -10 Enter score #3 (x >= 0): -15 Enter score #4 (x >= 0): -20 Enter score #5 (x >= 0): -25 Enter score #6 (x >= 0): 5 Enter score #7 (x >= 0): 10 Enter score #8 (x >= 0): 15 Enter score #9 (x >= 0): 20 Enter score #10 (x >= 0): 25 Enter score #11 (x >= 0): 34 List of Scores: Average score: 5.27 uxb2% Notice the amount of incorrect input and output that is not flagged nor pointed out as wrong

8.3 Signal Masks and Signal Sets

18 The Signal Set A process can temporarily prevent a signal from being delivered by blocking it Blocked signals do not affect the behavior of the process until they are delivered The process signal mask identifies the set of signals that are currently blocked A program specifies operations (such as blocking or unblocking) on groups of signals by using signal sets of type sigset_t Blocking a signal is different from ignoring a signal When a process blocks a signal, the operating system does not deliver the signal until the process unblocks the signal –A process blocks a signal by modifying its signal mask with sigprocmask()‏ When a process ignores a signal, the signal is delivered and the process handles it by throwing it away –A process sets a signal to be ignored by calling sigaction() with a handler of SIG_IGN (described later in these slides)‏

19 Signal Set Functions Signal sets are manipulated by the five functions listed below #include int sigaddset(sigset_t *set, int signalNbr); int sigdelset(sigset_t *set, int signalNbr); int sigemptyset(sigset_t *set); int sigfillset(sigset_t *set); int sigismember(const sigset_t *set, int signalNbr); The first parameter for each function is a pointer to a variable of type sigset_t The sigaddset() function adds signalNbr to the signal set The sigdelset() function removes signalNbr from the signal set The sigemptyset() function initializes a sigset_t variable to contain no signals The sigfillset() function initializes a sigset_t variable to contain all signals –A program initializes a signal set by calling either sigemptyset() or sigfillset() before using it Each of these functions return zero if the operation is successful; otherwise, the function returns –1 and sets errno The sigismember() function reports whether signalNbr is in a sigset_t variable –It returns 1 if sigNbr is in the signal set and zero if it is not

20 Example use of Signal Set Functions #include // ******************************* int main(void)‏ { int status; sigset_t mySet; status = sigemptyset(&mySet); if (status == -1)‏ perror("Failed to empty signal set"); status = sigaddset(&mySet, SIGINT); if (status == -1)‏ perror("Failed to add SIGINT signal to signal set"); status = sigaddset(&mySet, SIGQUIT); if (status == -1)‏ perror("Failed to add SIGQUIT signal to signal set"); if (sigismember(&mySet, SIGINT))‏ fprintf(stderr, "The set contains the SIGINT signal\n"); else fprintf(stderr, "The set does not contain the SIGINT signal\n"); return 0; } // End main

21 sigprocmask() Function A process can examine or modify its process signal mask with the sigprocmask() function #include int sigprocmask(int how, const sigset_t *set, sigset_t *oldSet); –The how parameter is an integer specifying the manner in which the signal mask is to be modified –The set parameter is a pointer to a signal set to be used in the modification If set is NULL, no modification is made –If oldSet is not NULL, the function sets oldSet to a pointer pointing to the signal set used before the modification If successful, the function returns zero; otherwise, it returns –1 and sets errno The sigprocmask() function should only be used by a process with a single thread –When multiple threads exist, the pthread_sigmask() function should be used

22 Using the sigprocmask() Function The how parameter can take on one of the following three values –SIG_BLOCK Add a collection of signals to those currently blocked –SIG_UNBLOCK Delete a collection of signals from those currently blocked –SIG_SETMASK Set the collection of signals being blocked to the specified set Some signals, such as SIGSTOP and SIGKILL cannot be blocked –If an attempt is made to block these signals, the system ignores the request without reporting an error The program on the next slide enters an infinite loop in which it displays a message, blocks the SIGINT signal, does some calculations, unblocks the signal, and does some more calculations –If a user enters Ctrl-C while SIGINT is blocked, the programs finishes the calculations and prints the message "Calculation ran safe from interruption" before terminating –If a user enters Ctrl-C while SIGINT is unblocked, the program terminates immediately (probably in the midst of the unblocked calculation loop)‏

23 Example #1: use of sigprocmask()‏ #include int main(void) { int i; sigset_t intMask; int repeatFactor = 10000; double y = 0.0; int status; status = sigemptyset(&intMask); if (status == -1)‏ { perror("Failed to initialize the signal set"); return 1; } // End if status = sigaddset(&intMask, SIGINT); if (status == -1)‏ { perror("Failed to add SIGINT to the signal set"); return 1; } // End if (More on next slide)‏

24 Example #1: use of sigprocmask() (continued)‏ for ( ; ; ) // Infinite loop { status = sigprocmask(SIG_BLOCK, &intMask, NULL); if (status == -1)‏ break; fprintf(stderr, "Blocked the SIGINT signal\n"); for (i = 0; i < repeatFactor; i++) y = y + sin((double)i); fprintf(stderr, "[SIGINT Blocked] Calculation ran safe from interruption\n"); status = sigprocmask(SIG_UNBLOCK, &intMask, NULL); if (status == -1)‏ break; fprintf(stderr, "Unblocked the SIGINT signal\n"); for (i = 0; i < repeatFactor; i++) y = y + sin((double)i); fprintf(stderr, "[SIGINT Unblocked] Calculation ran, but vulnerable to interruption\n"); } // End for perror("Failed to block or unblock signal mask"); return 1; } // End main

25 Example #2: use of sigprocmask()‏ #include int main(void) { pid_t pid; sigset_t mask, oldMask; int status; status = sigfillset(&mask); if (status == -1)‏ { perror("Failed to fill the signal set"); return 1; } // End if (More on next slide)‏

26 Example #2: use of sigprocmask() (continued)‏ status = sigprocmask(SIG_SETMASK, &mask, &oldMask); if (status == -1) { perror("Failed to block signals using the signal mask"); return 1; } // End if else fprintf(stderr, "\nParent set mask to block all signals before creating child\n\n"); pid = fork(); if (pid == -1) { perror("Failed to create child process"); return 1; } // End if if (pid == 0) // Child process { fprintf(stderr, "Child process is running with inherited signal mask from parent\n\n"); execl("/bin/ls", "ls", "-l", NULL); perror("exec function failed in child process"); return 1; } // End if (More on next slide)‏

27 Example #2: use of sigprocmask() (continued)‏ else // Parent process { fprintf(stderr, "Parent created child process\n\n"); status = sigprocmask(SIG_SETMASK, &oldMask, NULL); if (status == -1) { perror("Parent failed to restore signal mask"); return 1; } // End if else fprintf(stderr, "Parent reset signal mask to original values after creating child\n\n"); fprintf(stderr, "Parent waiting for child process to terminate...\n\n"); status = wait(NULL); if (status == -1) { perror("Parent failed to wait for child"); return 1; } // End if fprintf(stderr, "\nParent has detected termination of child process\n"); } // End else return 0; } // End main

28 Example #2: Sample Output uxb2% a.out Parent set mask to block all signals before creating child Parent created child process Parent reset signal mask to original values after creating child Parent waiting for child process to terminate... Child process is running with inherited signal mask from parent total 6 drwx jjt107 faculty 1024 Jun 21 10:35 Files drwx jjt107 faculty 1024 Jul 24 08:43 Mail drwx--x--x 19 jjt107 faculty 1024 Jul 3 12:32 http Parent has detected termination of child process uxb2%

8.4 Catching and Ignoring Signals

30 sigaction() Function The sigaction() function allows the calling program to set and examine the action associated with a specific signal #include int sigaction(int signalNbr, const struct sigaction *action, struct sigaction *oldAction); –The signalNbr parameter specifies the signal number for the action –The action parameter is a pointer to a struct sigaction structure that specifies the action to be taken –If action is NULL, the call to the function does not change the action associated with the signal –The oldAction parameter is a pointer to a struct sigaction structure that is assigned the previous action associated with the signal If successful, the function returns zero; otherwise, it returns –1 and sets errno

31 struct sigaction The struct sigaction structure has the following contents: struct sigaction { void (*sa_handler)(int); sigset_t sa_mask; int sa_flags; void (*sa_sigaction)(int, siginfo_t *, void *); }; –The sa_handler member is assigned SIG_DFL, SIG_IGN or a pointer to a signal handler function –The sa_mask member is assigned additional signals to be blocked during execution of the handler –The sa_flags member is assigned special flags and options –The sa_sigaction member is assigned a pointer to a real-time signal handler The storage for sa_handler and sa_sigaction may overlap; consequently, a program should use only one of these members to specify the action –If the SA_SIGINFO flag of the sa_flags field is cleared, the sa_handler field specifies the action to be taken for the specified signal –If the SA_SIGINFO flag of the sa_flags field is set, the sa_sigaction field specifies a signal-catching function

32 struct sigaction (continued)‏ This code segment sets the signal handler for SIGINT to mySignalHandler() struct sigaction newAction; int status; newAction.sa_handler = mySignalHandler; // Set the signal handler newAction.sa_flags = 0; // No special actions status = sigemptyset(&newAction.sa_mask); if (status == -1) perror("Failed to initialize signal set"); else { status = sigaction(SIGINT, &newAction, NULL); if (status == -1) perror("Failed to install signal handler for SIGINT"); } // End else

33 Signal Handler A signal handler is an ordinary function that returns void and has one integer parameter When the operating system delivers the signal, it sets the parameter to the number of the signal that was delivered –Most signal handlers ignore this value, but it is possible to use the same signal handler for many signals The usefulness of signal handlers is limited by the inability to pass values to them (this has been corrected for the sa_sigaction handler)‏ Two special values of the sa_handler member of the struct sigaction structure are SIG_DFL and SIG_IGN –SIG_DFL specifies that the sigaction() function should restore the default action for the signal –SIG_IGN specifies that the process should handle the signal by ignoring it (throwing it away)

34 Example use of SIG_IGN The following code segment causes the process to change its new action to ignore the SIGINT signal if the default action is currently in effect for this signal struct sigaction action; int status; status = sigaction(SIGINT, NULL, &action); if (status == -1) perror("Failed to get old handler information for SIGINT"); else if (action.sa_handler == SIG_DFL) { action.sa_handler = SIG_IGN; status = sigaction(SIGINT, &action, NULL); if (status == -1) perror( "Failed to set SIGINT signal handler to ignore signal"); } // End else

35 Example SIGINT Signal Handler #include #define MAX_PRESSES 5 static int pressCount = 0; void catchCtrlC(int signalNbr); int main(void)‏ { // See next slide } // This function is the signal handler void catchCtrlC(int signalNbr)‏ { char message[] = "Ctrl-C was pressed\n"; write(STDERR_FILENO, message, strlen(message) ); pressCount++; // Global variable } // End catchCtrlC The catchCtrlC() function is defined here as a signal handler for the SIGINT signal generated by Ctrl-C. The write() function is used instead of fprintf() because POSIX guarantees that it is async-signal safe, meaning that the function can be called safely from within a signal handler. (More on next slide)‏

36 Example SIGINT Signal Handler (continued)‏ // ***************************************** int main(void) { struct sigaction action; int status; action.sa_handler = catchCtrlC; action.sa_flags = 0; status = sigemptyset(&action.sa_mask); if (status == -1)‏ { perror("Failed to initialize signal set"); exit(1); } // End if status = sigaction(SIGINT, &action, NULL); if (status == -1)‏ { perror("Failed to set signal handler for SIGINT"); exit(1); } // End if while (pressCount < MAX_PRESSES); // Loop has no statements return 0; } // End main uxb3% a.out Ctrl-C was pressed uxb3% Sample Output

8.5 Waiting for Signals

38 Waiting for Signals Signals provide a method for waiting for an event without busy waiting Busy waiting means continually using CPU cycles to test for the occurrence of an event (Typically through the use of a loop)‏ A more efficient approach is to suspend the process until the waited-for event occurs –That way, other processes can use the CPU productively The pause(), sigsuspend(), and sigwait() functions provide three mechanisms for suspending a process until a signal occurs

39 pause() Function The pause() function suspends the calling thread until the delivery of a signal whose action is either to execute a user-defined signal handler or to terminate the process #include int pause(void); If the action is to terminate, pause() does not return If a signal is caught by the process, pause() returns after the signal handler returns The pause() function always returns –1 –If interrupted by a signal, pause() sets errno to EINTR To wait for a particular signal by using pause(), a program must determine which signal caused pause() to return –This information is not directly available, so the signal handler must set a global flag for the program to check after pause() returns

40 Example of pause() and SIGINT Signal Handler int main(void) { struct sigaction action; int status; action.sa_handler = catchCtrlC; action.sa_flags = 0; status = sigemptyset(&action.sa_mask); if (status == -1)‏ { perror("Failed to initialize signal set"); exit(1); } // End if status = sigaction(SIGINT, &action, NULL); if (status == -1)‏ { perror("Failed to set signal handler for SIGINT"); exit(1); } // End if fprintf(stderr, "Program paused...\n"); pause(); return 0; } // End main uxb3% a.out Program paused... Ctrl-C was pressed uxb3% Sample Output

41 Sequence Problems with the pause() Function The following code segment uses pause() to cause a process to wait for a particular signal by having the signal handler set the sigreceived variable to 1 static sig_atomic_t sigreceived = 0;... while (sigreceived == 0) pause(); What happens if a signal occurs after the test of sigreceived but before pause() ? –The pause() function would not return at that time, but would wait until some other signal or the occurrence of the same signal was delivered again to the process The delivery of a signal before the pause() function was called in a program was one of the major problems with the original UNIX signals –There was no simple, reliable way to get around the problem The program must do two operations at once: unblock the signal and call the pause() function –In other words, the two operations need to be atomic (i.e., logically viewed as one instruction)‏ The sigsuspend() function provides a method of achieving this

42 sigsuspend() Function The sigsuspend() function replaces the process signal mask with the set of signals pointed to by the signalMask argument and suspends the process until delivery of a signal whose action is either to execute a signal handler function or to terminate the process. #include int sigsuspend(const sigset_t *signalMask); The sigsuspend() function returns when the signal handler of the caught signal returns The signalMask parameter can be used to unblock the signal that the program is looking for When sigsuspend() returns, the signal mask is reset to the value it had before the sigsuspend() function was called The sigsuspend() function always returns –1 and sets errno

43 Example use of sigsuspend()‏ The program on the next slide shows a way for a process to wait for a single signal Assume that a signal handler has been set up for the sigNbr signal and that the signal handler sets sigReceived to 1 Error checking has been omitted from the code for clarity purposes The maskAll structure contains all signals The maskMost structure contains all signals except sigNbr No signals can be caught between the testing of sigReceived and the suspension of the process, since the signal is blocked at this point The process signal mask had the value contained in maskMost while the process is suspended, so only sigNbr is not blocked When sigsuspend() returns, the signal must have been received

44 Example use of sigsuspend() (continued)‏ #include static sig_atomic_t sigReceived = 0; int main(void)‏ { int status; sigset_t maskAll; sigset_t maskMost; sigset_t maskOld; int sigNbr = SIGUSR1; fprintf(stderr, "PID: %d\n", getpid()); sigfillset(&maskAll); sigfillset(&maskMost); sigdelset(&maskMost, sigNbr); sigprocmask(SIG_SETMASK, &maskAll, &maskOld); if (sigReceived == 0)‏ sigsuspend(&maskMost); sigprocmask(SIG_SETMASK, &maskOld, NULL); return 0; } // End main uxb3% a.out PID 2624 uxb3% a.out PID 3548 uxb3% Command Shell A uxb3% kill –INT 2624 uxb3% uxb3% kill –USR uxb3% kill – uxb3% Command Shell B

45 sigwait() Function The sigwait() function blocks until any of the signals in the structure pointed to by signalMask becomes pending –It then removes that signal from the set of pending signals and unblocks itself #include int sigwait(const sigset_t *signalMask, int *signalNbr); When sigwait() returns, the number of the signal that was removed from the pending signals is stored in the location pointed to by signalNbr If successful, sigwait() returns zero; otherwise, it returns –1 and sets errno Note the difference between sigwait() and sigsuspend()‏ –Both functions have a first parameter that is a pointer to a signal set –For sigsuspend(), this signal set holds the new signal mask and so the signals that are not in the set are the ones that can cause sigsuspend() to return –For sigwait(), this signal set holds the set of signals to be waited for, so the signals in the set are the ones that can cause the sigwait() to return –Unlike sigsuspend(), the sigwait() function does not change the process signal mask –The signals in signalMask should be blocked before sigwait() is called

46 Example use of sigwait()‏ #include int main(void)‏ { int signalCount = 0; int signalNbr; sigset_t signalSet; int status; status = sigfillset(&signalSet); if (status == -1)‏ perror("Failed to fill the signal set"); status = sigprocmask(SIG_BLOCK, &signalSet, NULL); if (status == -1)‏ perror("Failed to block signals"); fprintf(stderr, "This process has ID %ld\n", (long)getpid()); (More on next slide)‏

47 Example use of sigwait() (continued)‏ for ( ; ; ) // Infinite loop { signalNbr = sigwait(&signalSet); // Default version if (signalNbr == -1)‏ { perror("Failed to wait using sigwait"); return 1; } // End if else fprintf(stderr, "Unblocked signal #%d\n", signalNbr); /* status = sigwait(&signalSet, &signalNbr); // POSIX version if (status == -1)‏ { perror("Failed to wait using sigwait"); return 1; } // End if */ signalCount++; fprintf(stderr, "Number of signals so far: %d\n", signalCount); } // End for } // End main

48 Example use of sigwait(): Sample Output uxb3% a.out This process has ID Unblocked signal #16 Number of signals so far: 1 Unblocked signal #17 Number of signals so far: 2 Unblocked signal #2 Number of signals so far: 3 Unblocked signal #14 Number of signals so far: 4 Unblocked signal #16 Number of signals so far: 5 Unblocked signal #2 Number of signals so far: 6 Killed uxb3% uxb3% kill -USR uxb3% kill -USR uxb3% kill -INT uxb3% kill -ALRM uxb3% kill -USR uxb3% kill -INT uxb3% kill -KILL uxb3% Command Shell ACommand Shell B

8.6 Handling Signals: Errors and Async-signal Safety

50 Signals and Function Calls Three difficulties can occur when signals interact with function calls –The first problem concerns whether POSIX functions that are interrupted by signals should be restarted –The second problem occurs when signal handlers call non-reentrant functions –The third problem involves the handling of errors that use errno What happens when a process catches a signal while it is executing a library function? –The answer depends on the type of call Terminal I/O can block the process for an undetermined length of time –There is no limit on how long it takes to get a key value from a keyboard or to read from a pipe –Function calls that perform such operations are sometimes characterized as slow Other operations, such as disk I/O, can block for short periods of time; still others, such as getpid(), do not block at all –Neither of these operations is considered to be slow

51 Slow POSIX Calls The slow POSIX calls are the ones that are interrupted by signals They return when a signal is caught and the signal handler returns The interrupted function returns –1 with errno set to EINTR –If a function sets errno and one of the possible values is EINTR, then the function can be interrupted The program must handle this error explicitly and restart the system call if desired It was originally thought that the operating system needs to interrupt slow calls to allow the user the option of canceling a blocked call This traditional treatment of handling blocked functions has been found to add unneeded complexity to many programs, and it is not used in newly-created POSIX functions

52 Async-signal Safe A function is async-signal safe if it can be safely called from within a signal handler Many POSIX library functions are not async-signal safe because they use static data structures, call malloc() or free(), or use global data structures in a non-reentrant way Consequently, a single process might not correctly execute concurrent calls to these functions Normally this is not a problem in a program with no signal handlers, but signals add concurrency to a program Since signals occur asynchronously, a process may catch a signal while it is executing a library function –Therefore, programmers must be careful when calling library functions from inside signal handlers The next slide lists many of the POSIX functions that are safe to call from within a signal handler –Notice that functions such as scanf(), fscanf(), printf(), and fprintf() from the C standard I/O library are not on the list

53 Some Async-signal Safe Functions sleep sigsuspend sigprocmask sigismember sigfillset sigdelset sigaddset sigaction rmdir read write waitpid wait unlink time stat raisegetpid pipefstat pausefork openexecve mkfifoexecle mkdirclose lstatchown linkchmod killchdir getppidalarm

54 Useful Rules for Signal Handling When in doubt that a function is async-signal safe or not, explicitly restart library function calls within a program Check each library function used in a signal handler to make sure that it is on the list of async-signal safe functions Carefully analyze the potential interactions between a signal handler that changes an external variable and other program code that accesses the variable –Block signals to prevent unwanted interactions As a general rule, signal handlers should save and restore errno if they call functions that might change the value of errno