我们在Linux的shell下启动一个前台进程,然后按下ctrl+c使进程中断,那有没有想过它的底层原理是什么?
进程信号是一种“软件中断”,一种事件通知机制。通知进程发生了某个事件,打断进程当前操作,去处理这个事件。
查看信号种类:kill -l
每个信号表示一个事件,总共有62种信号,1-31号信号是借鉴Unix操作系统的非可靠信号(可能会信号丢失),34-64号是可靠信号。
在路径: /usr/include/bits/signum.h,查看这个头文件,这些信号都被定义为宏。
产生——注册——注销——处理
ctrl+z:发送SIGSTOP停止信号
int main(int argc, char *argv[])
{
while(1)
{
printf("_____________\n");
sleep(1);
}
return 0;
}
该进程处于“T”状态,进程由前台进程变为后台进程,
jobs:查看后台停止作业;
fg 作业序号:将该后台停止作业转换到前台。
2.软件产生:
kill(命令):kill -signum pid 杀死一个进程,给进程发送一个信号-默认是终止信号(例如kill -9 就是发送SIGKILL信号),如果不指定信号,则发送默认信号SIGTERM;
kill函数(系统调用):int kill(pid_t pid, int sig);
参数解释:pid:指定要将信号发送给哪个进程;
sig:指定要发送的信号;
返回值:成功返回0,失败返回-1.
int main(int argc, char *argv[])
{
kill(getpid(),SIGINT);
while(1)
{
printf("_____________\n");
sleep(1);
}
return 0;
}
运行程序直接中断。
int raise(int sig):给调用进程自身发送指定信号。
unsigned int alarm(unsigned int seconds):设置一个定时器,多少seconds秒后,给进程发送一个SIGALRM信号(该信号默认处理方式为退出进程)。
int main(int argc, char *argv[])
{
alarm(3);
while(1)
{
printf("_____________\n");
sleep(1);
}
return 0;
}
void abort(void):给进程发送一个SIGABRT信号,abort函数使当前进程接收到信号而异常终止。
在进程的pcb中有个未决信号集合(pending)-还没有被处理的信号的集合——位图,还有一个struct sigqueue专门存放信号信息节点的队列(双向链表)。
注册就是在未决信号集合中标记信号将0置为1,然后给双向链表中添加信号的信息节点。
非可靠信号:若信号已经注册,则不作任何操作;
可靠信号:无论信号是否注册,都会进行注册(置1加添加节点)。
删除在进程pcb中的信号信息节点,重置位图为0。
非可靠信号:删除信息节点,直接位图置0;
可靠信号:删除信息节点之后,确定没有相同节点才会重置位图。
执行信号的处理回调函数(sigaction-handler)。
如何修改信号的处理流程,暨修改处理函数?
1.sighandler_t signal(int signum, sighandler_t handler);
参数解释:
sigunm:信号值;
handler:信号要新指定的处理方式;
handler:SIG_DFL-默认;SIG_IGN-忽略;自定义函数。
自定义函数的类型:typedef void (*sighandler_t)(int);
返回值:成功返回信号原来的处理方式,失败返回-1(SIG_ERR)。
举例忽略SIGINT,暨忽略ctrl+c:
int main(int argc, char *argv[])
{
signal(SIGINT,SIG_IGN);
while(1)
{
printf("_____________\n");
sleep(1);
}
return 0;
}
自定义处理回调函数:
把SIGINT的处理方式改为打印信号值。
void sigcb(int signo)
{
printf("recv signo : %d\n", signo);
}
int main(int argc, char *argv[])
{
signal(SIGINT,sigcb);
while(1)
{
printf("_____________\n");
sleep(1);
}
return 0;
}
暨运行程序后按一次ctrl+c就会打印一次2.
注意:9号信号和19号信号不能被修改处理操作。
如何使程序运行从用户态切换到内核态:
中断、异常、系统调用
阻塞一个信号表示收到这个信号之后暂时不去处理,知道解除阻塞之后进行处理。
进程pcb中有一个信号阻塞集合,在这个集合中标记那个信号则阻塞哪个信号,
int sigprocmask(int how, const sigset_t * set, sigset_t * oldset);
参数解释:
how:要对信号阻塞集合进行的操作类型;
oldset:保存原有集合;
set:新集合;
①SIG_BLOCK:将set集合中的信号添加到阻塞集合中;block |= set;(block:101,set:010;——111)
②SIG_UNBLOCK:从阻塞集合中移除set集合中的信号;
③SIG_SETMASK:将set集合中的信号设置为阻塞集合的信号;block = set,覆盖式设置。
返回值:成功返回0,失败返回-1.
#include
int sigemptyset(sigset_t *set);清空set集合
int sigfillset(sigset_t *set);将所有信号添加到set集合中
int sigaddset(sigset_t *set, int signum);将指定信号添加到集合中
int sigdelset(sigset_t *set, int signum);从集合中移除指定信号
int sigismember(const sigset_t *set, int signum);判断信号是否在集合中
举个栗子:自定义修改两种信号(可靠和非可靠)signal。 先阻塞信号sigprocmask SIG_BLOCK,再使进程暂停getchar,暂停期间给进程发送信号,让进程继续运行,解除信号阻塞sigprocmask SIG_UNBLOCK。
#include
#include
#include
#include
void sigcb(int signo)
{
printf("recv sino:%d\n", signo);
}
int main(int argc, char* argv[])
{
signal(SIGINT, sigcb);//非可靠信号
signal(40,sigcb);//可靠信号
sigset_t set;
sigemptyset(&set);
sigfillset(&set);
sigprocmask(SIG_BLOCK, &set,NULL);
printf("回车解除暂停\n");
getchar();
sigprocmask(SIG_UNBLOCK,&set,NULL);
while(1)
{
sleep(1);
}
return 0;
}