唯有前进值得敬仰

---等得越久,相聚时越幸福
  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

linux-0.11内核 信号处理小结

Posted on 2010-07-08 10:26  绿豆芽33  阅读(1353)  评论(0编辑  收藏  举报

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(堆栈错)。

对于一个进程来说,收到一个信号后,有三种处理方式:

  1. 忽略该信号,但SIGKILL和SIGSTOP除外
  2. 捕获该信号,自定义处理函数
  3. 执行默认操作,内核为每种信号都提供一种默认操作——终止进程的执行

信号处理过程

当进程调用系统调用结束返回后,对信号进行识别处理,若进程结构的信号位图表明该进程接收到信号,则调用信号处理函数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弹出,转向用户态。

下面是老师关于信号处理程序的调用方式给出的两张图,贴出来有助于理解