signal函数

#include <signal.h>

  1. void (*signal(int signo, void (*func)(int)))(int)
  2. 成功返回前一个信号布署,错误返回SIG_ERR。

 

signal 函数的原型指定函数需要两个参数并返回一个无返回的函数的指针。signal函数的第一个参数,signo是一个整型。第二个参数是一个接受一个整型参数而无返回的函数的指针。signal返回的指针指向接受单个整型参数(信号号)而无返回的函数。当我们调用signal来建立信号处理器时,第二个参数是一个函数指针。signal的返回值是前一个信号处理器的指针。

如果我们检查系统的头文件<signal.h>,我们很可能发现下面形式的声明:


/* Fake signal functions.  */
#define SIG_ERR ((__sighandler_t) -1)           /* Error return.  */
#define SIG_DFL ((__sighandler_t) 0)            /* Default action.  */
#define SIG_IGN ((__sighandler_t) 1)            /* Ignore signal.  */


这些常量被用来代替“指向带一个整型参数而无返回的函数的指针”,signal的第二个参数和signal的返回值。这三个值不必是-1、0、1。它们必须是三个不可能是任何声明的函数的地址的值。多数UNIX系统使用了展示的值。


下面的代码展示了一个简单的信号处理器,它捕获两个用户定义的信号的任一个并打印信号号:

  1. #include <signal.h>
  1. static void sig_usr(int); /* one handler for both signals */
  1. int  main(void)
  1. {
  2.     if (signal(SIGUSR1, sig_usr) == SIG_ERR) {
  3.         printf("can't catch SIGUSR1\n");
  4.         exit(1);
  5.     }
  6.     if (signal(SIGUSR2, sig_usr) == SIG_ERR) {
  7.         printf("can't catch SIGUSR2\n");
  8.         exit(1);
  9.     }
  10.     for (;;)
  11.         pause();
  12. }

  13. static void
  14. sig_usr(int signo) /* argument is signal numer */
  15. {
  16.     if (signo == SIGUSR1)
  17.         printf("received SIGUSR1\n");
  18.     else if (signo == SIGUSR2)
  19.         printf("received SIGUSR2\n");
  20.     else
  21.         printf("received signal %d\n", signo);
  22. }


在10.10节,我们描述pause函数,它简单的暂停调用进程,直到一个信号被收到。

我们在后台运行这个程序,然后使用kill命令向它发送信号。注意在UNIX系统的术语“kill”是一个不恰当的名字。kill命令和kill函数只是向一个进程或进程组发送一个信号。信号是否终止进程取决于哪个信号被发送和进程是否捕获这个信号。运行结果为:


$ ./a.out &
[1] 3594
$ kill -USR1 3594
received SIGUSR1
$ kill -USR2 3594
received SIGUSR2
$ kill 3594
[1]+  已终止               ./a.out


当我们发送SIGTERM信号时,进程被终止,因为它不捕获这个信号,而这个信号的默认动作是终止。


程序启动


当一个程序执行时,所有信号的状态都是默认或忽略。通常,所有信号被设置被它们的默认动作,除非调用exec的进程正忽略这个信号。确切地说,exec函数改变任何被被捕获的的信号的布署为它们的默认动作,而不改变其它信号的状态。(自然地,一个被一个调用exec的进程捕获的信号不能被新程序的相同函数捕获,因为调用者的信号处理函数很可能在新的被执行的程序文件里没有意义。)


一个确切的例子是一个交互外壳如何为后台进程处理中断和退出信号。对于一个不支持工作控制的外壳,当我们在后台执行一个进程时,比如cc main.c &,外壳自动把后台进程的中断和退出信号的布署设置为忽略。这样如果我们输入中断字符,它不会影响后台进程。如果不这么做而我们输入中断字符,那么它不会只终止前台进程,还有所有的后台进程。


许多捕获这两个信号的交互程序有如下的代码:


void sig_int(int), sig_quit(int);
if (signal(SIGINT, SIG_IGN) != SIG_IGN)
  signal(SIGINT, sig_int);
if (signal(SIGQUIT, SIG_IGN) != SIG_IGN)
  signal(SIGQUIT, sig_quit);


这样做,进程只捕获当前没有被忽略的信号。


这两个signal的调用也显示了signal函数的局限:我们不能不改变布署来决定当前的布署。我们在本章后面看到sigaction函数如何允许我们不改变它也能决定一个信号的布署。

 

 

一些常用的Signal 如下:

SignalDescription
SIGABRT由调用abort函数产生,进程非正常退出
SIGALRM用alarm函数设置的timer超时或setitimer函数设置的interval timer超时
SIGBUS某种特定的硬件异常,通常由内存访问引起
SIGCANCEL由Solaris Thread Library内部使用,通常不会使用
SIGCHLD进程Terminate或Stop的时候,SIGCHLD会发送给它的父进程。缺省情况下该Signal会被忽略
SIGCONT当被stop的进程恢复运行的时候,自动发送
SIGEMT和实现相关的硬件异常
SIGFPE数学相关的异常,如被0除,浮点溢出,等等
SIGFREEZESolaris专用,Hiberate或者Suspended时候发送
SIGHUP发送给具有Terminal的Controlling Process,当terminal被disconnect时候发送
SIGILL非法指令异常
SIGINFOBSD signal。由Status Key产生,通常是CTRL+T。发送给所有Foreground Group的进程
SIGINT由Interrupt Key产生,通常是CTRL+C或者DELETE。发送给所有ForeGround Group的进程
SIGIO异步IO事件
SIGIOT实现相关的硬件异常,一般对应SIGABRT
SIGKILL无法处理和忽略。中止某个进程
SIGLWP由Solaris Thread Libray内部使用
SIGPIPE在reader中止之后写Pipe的时候发送
SIGPOLL当某个事件发送给Pollable Device的时候发送
SIGPROFSetitimer指定的Profiling Interval Timer所产生
SIGPWR和系统相关。和UPS相关。
SIGQUIT输入Quit Key的时候(CTRL+\)发送给所有Foreground Group的进程
SIGSEGV非法内存访问
SIGSTKFLTLinux专用,数学协处理器的栈异常
SIGSTOP中止进程。无法处理和忽略。
SIGSYS非法系统调用
SIGTERM请求中止进程,kill命令缺省发送
SIGTHAWSolaris专用,从Suspend恢复时候发送
SIGTRAP实现相关的硬件异常。一般是调试异常
SIGTSTPSuspend Key,一般是Ctrl+Z。发送给所有Foreground Group的进程
SIGTTIN当Background Group的进程尝试读取Terminal的时候发送
SIGTTOU当Background Group的进程尝试写Terminal的时候发送
SIGURG当out-of-band data接收的时候可能发送
SIGUSR1用户自定义signal 1
SIGUSR2用户自定义signal 2
SIGVTALRMsetitimer函数设置的Virtual Interval Timer超时的时候
SIGWAITINGSolaris Thread Library内部实现专用
SIGWINCH当Terminal的窗口大小改变的时候,发送给Foreground Group的所有进程
SIGXCPU当CPU时间限制超时的时候
SIGXFSZ进程超过文件大小限制
SIGXRESSolaris专用,进程超过资源限制的时候发送


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值