linux信号总结
2016-01-28 08:55
666 查看
linux中在运行一些进程时有时肯定因为一些内部或外部的原因需要中断,这就是软中断,通过信号来实现。下面介绍一些中断,后续会有增加,可能有不足的地方望指正。
一、传统的信号处理方式signal()函数,这一函数相对比较简单:
void (*signal(int signo, void (*func)(int)))(int);
可分解为:
由于signal()函数的功能有限,在此就不详细描述为何有限。unix又引进了sigaction()函数,它包含了signal的所有功能,并且有很多的选择性。下面就来介绍:
二、sigaction()函数
int sigaction(int signo,const struct sigaction * restrict act,struct sigaction * restrict oldact));
第二和第三个参数都是关于sigaction数组的,下面主要介绍:
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结构包含了信号产生原因的有关信息。该结构的大致样式如下所示:
各种信号的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可以修改这个被阻塞的信号集。
how:用于指定信号修改的方式,可能选择有三种
SIG_BLOCK//将set所指向的信号集中包含的信号加到当前的信号掩码中。即信号掩码和set信号集进行或操作。
SIG_UNBLOCK//将set所指向的信号集中包含的信号从当前的信号掩码中删除。即信号掩码和set进行与操作。
SIG_SETMASK //将set的值设定为新的进程信号掩码。即set对信号掩码进行了赋值操作。
set:为指向信号集的指针,在此专指新设的信号集,如果仅想读取现在的屏蔽值,可将其置为NULL。
oldset:也是指向信号集的指针,在此存放原来的信号集。可用来检测信号掩码中存在什么信号。
sigsetops构造信号集:
sigemptyset()初始化时设置的优先级设为 empty ,与所有信号设置.
sigfillset()初始化设置,包括所有信号.
sigaddset()和sigdelset()信号 signum 中添加和删除分别设置.
注意:
这里在修改信号挡板时会保存先前的设置 prevsigs,然后在对数据进行修改,最后用保存的设置prevsigs来恢复原来的信号挡板。除非目的就是修改获取修改获取的资源,否则释放资源时恢复获取时的状态是个好习惯。
一、传统的信号处理方式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来恢复原来的信号挡板。除非目的就是修改获取修改获取的资源,否则释放资源时恢复获取时的状态是个好习惯。
相关文章推荐
- Ubuntu 14.04 文件服务器--samba的安装和配置
- CentOS 7下MySQL服务启动失败的解决思路
- Linux 基金会引发巨大争议,它到底是企业的还是草根的?
- ubuntu安装和配置SVN
- 最适合和最不适合新手使用的几款 Linux 发行版
- Linux下的parted工具的使用 GPT分区安装系统
- 在linux环境上如何部署SolrCloud
- 桌面版Linux提示Enter Password for Default Keyring to Unlock的解决方法
- Linux-IPC之信号量
- Linux curl命令
- CentOS7安装iptables防火墙
- Linux调试技术
- linux查看是否为虚拟机
- Linux系统项目部署一些命令总结
- Linux监控和安全运维 2.0 zabbix配置邮件告警
- linux mint 安装 SecureCRT
- centos挂存储
- CentOs 关机注意事项
- 阿岳之_centos日志功能rsyslog介绍篇
- Linux内核总结