Linux中的信号及产生原因这里就不再多说。若有不了解可查看Linux中信号signal及其产生原因
进程信号
linux中的信号种类:62种 1~31非可靠信号 34~64可靠信号
进程信号是进程之间事件异步通信的一种方式,属于软中断。
通俗的来说软中断是由进程产生的,硬中断是由硬件产生的。
硬中断是外部设备对CPU的中断,软中断是中断底半部的一种处理机制,信号则是由内核(或其他进程)对某个进程的中断。
产生信号的方式
1.终端按键 ctrl+c产生SIGINT信号 ctrl+/产生 SIGQUIT信号
2.系统函数调用
调用 kill 命令 向3044进程发送SIGSEGV信号 ,SIGSEGV会引起一个段信号错误。
kill命令是调用kill函数实现的
#include<signal.h>
//向指定进程发送指定信号
int kill(pid_t pid, int sig);
//向当前进程发送指定信号
int raise(int signo)
//异常终止一个进程
void abort(void)
//函数首先解除进程对SIGABRT信号的阻止,然后向调用进程发送该信号。
//abort()函数会导致进程的异常终止除非SIGABRT信号被捕捉并且信号处理句柄没有返回。
3.软件条件产生信号
SIGPIPE,SIGALRM都是由软件产生的信号
SIGPIPE 管道的读端关闭,一个进程写入管道时就会发生SIGPIPE
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<wait.h>
#include<signal.h>
void sigcb(int arg)
{
printf("SIGPIPE");
}
int main()
{
int pipe_fd[2];
pipe(pipe_fd); //创建匿名管道
if(fork() == 0)
{
//子进程
//关闭读端
close(pipe_fd[0]);
write(pipe_fd[1],"hello",6);
}
//关闭读端
close(pipe_fd[0]);
wait(NULL);
return 0;
}
4.硬件异常产生的信号
硬件异常被硬件检测到通知内核,内核向当前进程发送适当的信号。
常见的硬件异常有 除0,内存非法访问 SIGFPE ,SIGSEGV等。
信号的处理方式
- 忽略此信号
- 执行信号的默认动作
- 捕捉函数:signal(SIG***, sigcb), sigcb为这个信号的处理函数。当处理这个信号时,从内核态调用到用户态执行这个函数。
信号的常见概念
- 实际执行信号的处理动作称为信号递达
- 信号从产生到递达的过程称为信号未决
- 进程可以阻塞指定信号,直到进程解除对指定信号的阻塞,才执行递达动作
信号在内核中的表示
block为阻塞信号表,对应0/1标记对应位数信号是否被阻塞;
pending为未决信号表,表示是否收到对应位数的信号;
handler表标识相应信号的处理动作(忽略、默认处理或自定义处理);