linux信号总结

本文介绍了Linux中处理进程中断的信号机制,包括传统的signal()函数、增强型的sigaction()函数以及sigprocmask()函数。详细阐述了信号处理程序的设置、信号掩码的操作以及信号集的管理,提供了理解和使用这些函数的关键知识点。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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来恢复原来的信号挡板。除非目的就是修改获取修改获取的资源,否则释放资源时恢复获取时的状态是个好习惯。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值