您的位置:首页 > 运维架构 > Linux

linux信号总结

2016-01-28 08:55 666 查看
linux中在运行一些进程时有时肯定因为一些内部或外部的原因需要中断,这就是软中断,通过信号来实现。下面介绍一些中断,后续会有增加,可能有不足的地方望指正。

一、传统的信号处理方式signal()函数,这一函数相对比较简单:

void (*signal(int signo, void (*func)(int)))(int);

可分解为:

typedef void sigfunc(int);
sigfunc *signal(int signo, sigfunc *));


第一个参数指定信号的值,第二个参数指定针对前面信号值的处理,可以忽略该信号(参数设为SIG_IGN);可以采用系统默认方式处理信号(参数设为SIG_DFL);也可以自己实现处理方式(参数指定一个函数地址)。


由于signal()函数的功能有限,在此就不详细描述为何有限。unix又引进了sigaction()函数,它包含了signal的所有功能,并且有很多的选择性。下面就来介绍:

二、sigaction()函数

int sigaction(int signo,const struct sigaction * restrict act,struct sigaction * restrict oldact));

第二和第三个参数都是关于sigaction数组的,下面主要介绍:

struct sigaction {
void (*sa_handler)(int);   /* func pointer */
void (*sa_sigaction)(int, siginfo_t *, void *);   /*func pointer */
sigset_t sa_mask;
int sa_flags;
void (*sa_restorer)(void);
}


sa_handler和signal一样是SIG_IGN或SIG_DFL或是处理信号的函数,如果是最后一个,则sa_mask字段说明了一个信号集,在调用该信号捕捉函数之前,这一信号集要加到进程的信号屏蔽字中。仅当从信号捕捉函数返回时再将进程的信号屏蔽字复位为原先值。这样,在调用信号处理程序时就能阻塞某些信号。在信号处理程序被调用时,操作系统建立的新信号屏蔽字包括正被递送的信号。因此保证了在处理一个给定的信号时,如果这种信号再次发生,那么它会被阻塞到对前一个信号的处理结束为止。

sa_flags参数

详解参见:http://blog.chinaunix.net/xmlrpc.php?r=blog/article&uid=28852942&id=3754478

sa_sigaction字段是一个替代的信号处理程序,当在sigaction结构中使用了SA_SIGINFO标志时,使用该信号处理程序。对于sa_sigaction字段和sa_handler字段这两者,其实现可能使用同一存储区,所以应用程序只能一次使用这两个字段中的一个。

通常,按下列方式调用信号处理程序:

void handler(int signo);

但是,如果设置了SA_SIGINFO标志,那么按下列方式调用信号处理程序:

void handler(int signo, siginfo_t *info, void *context);

siginfo_t结构包含了信号产生原因的有关信息。该结构的大致样式如下所示:

struct siginfo {
int      si_signo;        /* signal number */
int      si_code;         /* additional info (depends on signal) */
pid_t    si_pid;          /* sending PRocess ID */
uid_t    si_uid;          /* sending process real user ID */
void    *si_addr;         /* address that caused the fault */
int      si_status;       /* exit value or signal number */
long     si_band;         /* band number for SIGPOLL */
/* possibly other fields also */
};


各种信号的si_code值(包括上面的相关数据结构和标志选项),可通过man sigaction命令进行查看。

若信号是SIGCHLD,则将设置si_pid、si_status和si_uid字段。

若信号是SIGILL或SIGSEGV,则si_addr包含造成故障的根源地址,尽管该地址可能并不准确。

若信号是SIGPOLL,那么si_band字段将包含STREAMS消息的优先级(priority band),该消息产生POLL_IN、POLL_OUT或POLL_MSG事件。

si_errno字段包含错误编号,它对应于引发信号产生的条件,并由实现定义。

信号处理程序的context参数是无类型指针,它可被强制转换为ucntext_t结构类型,用于标识信号传递时进程的上下文。

三、sigprocmask()阻塞信号函数

在任何时候一个进程都有一些信号被阻塞。注意是阻塞而不是忽略。这个信号就称为信号挡板,通过sigprocmask可以修改这个被阻塞的信号集。

int sigprocmask(int how, const sigset_t *restrict set, sigset_t *restrict oldset);


how:用于指定信号修改的方式,可能选择有三种

SIG_BLOCK//将set所指向的信号集中包含的信号加到当前的信号掩码中。即信号掩码和set信号集进行或操作。

SIG_UNBLOCK//将set所指向的信号集中包含的信号从当前的信号掩码中删除。即信号掩码和set进行与操作。

SIG_SETMASK //将set的值设定为新的进程信号掩码。即set对信号掩码进行了赋值操作。

set:为指向信号集的指针,在此专指新设的信号集,如果仅想读取现在的屏蔽值,可将其置为NULL。

oldset:也是指向信号集的指针,在此存放原来的信号集。可用来检测信号掩码中存在什么信号。

sigsetops构造信号集:

sigemptyset()初始化时设置的优先级设为 empty ,与所有信号设置.

sigfillset()初始化设置,包括所有信号.

sigaddset()和sigdelset()信号 signum 中添加和删除分别设置.

注意:

......
sigprocmask(SIG_BLOCK,&sigs,&prevsigs);
//...modify data structure here...
sigprocmask(SIG_SET,*prevsigs,NULL);
......


这里在修改信号挡板时会保存先前的设置 prevsigs,然后在对数据进行修改,最后用保存的设置prevsigs来恢复原来的信号挡板。除非目的就是修改获取修改获取的资源,否则释放资源时恢复获取时的状态是个好习惯。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: