linux-0.11内核中关于信号处理这部分内容主要涉及四个文件system_call.s,signal.h,signal.c,exit.c。其中signal.h,signal.c中定义和实现了内核中所有有关信号处理的函数,当然exit.c中还有两个与信号相关的函数send_sig与tell_father。
linux-0.11内核中定义了22种不同的信号,20种是posix.1中规定的,还有两个专用的:SIGUNUSED(未定义)和SIGSTKFLT(堆栈错)。
对于一个进程来说,收到一个信号后,有三种处理方式:
- 忽略该信号,但SIGKILL和SIGSTOP除外
- 捕获该信号,自定义处理函数
- 执行默认操作,内核为每种信号都提供一种默认操作——终止进程的执行
信号处理过程
当进程调用系统调用结束返回后,对信号进行识别处理,若进程结构的信号位图表明该进程接收到信号,则调用信号处理函数do_signal,该函数就会把信号的处理句柄插入到用户程序堆栈中。然后执行信号处理程序,之后再继续执行用户的程序。
在这个过程中,由系统调用signal或sigaction向进程设置信号处理句柄。信号的识别由system_call.s中的system_call函数完成,并负责调用do_signal。do_signal的主要作用是将信号处理函数句柄插入到用户堆栈中(用户程序执行系统调用时进入了内核堆栈),并在do_signal返回后立刻执行信号处理函数。do_signal返回后,system_call.s把堆栈eip以下的所有值弹出(见下面图2,及system_call.s中120-127行代码),iret之后把剩下的cs:eip,eflags,ss:esp弹出,转向用户态。
下面是