linux内核学习-6信号量(关注新浪微博:寂寞侵蚀的岁月(4000多篇技术分享))
浏览数:18 /
时间:2015年06月20日
#include // 调度程序头文件,定义了任务结构task_struct、初始任务0 的数据,// 还有一些有关描述符参数设置和获取的嵌入式汇编函数宏语句。8 #include // 内核头文件。含有一些内核常用函数的原形定义。9 #include // 段操作头文件。定义了有关段寄存器操作的嵌入式汇编函数。1011 #include // 信号头文件。定义信号符号常量,信号结构以及信号操作函数原型。1213 volatile void do_exit(int error_code); // 前面的限定符volatile
要求编译器不要对其进行优化。14// 获取当前任务信号屏蔽位图(屏蔽码)。15 int sys_sgetmask()16 {17 return current->blocked;18 }19// 设置新的信号屏蔽位图。SIGKILL 不能被屏蔽。返回值是原信号屏蔽位图。20 int sys_ssetmask(int newmask)21 {22 int old=current->blocked;2324 current->blocked = newmask & ~(1<<(SIGKILL-1));25 return
old;26 }27// 复制sigaction 数据到fs 数据段to 处。。28 static inline void save_old(char * from,char * to)调用相应功能的C函数处理程序。对信号进行识别预处理。调用do_signal()函数。将信号处理程序句柄插入到用户堆栈中。IRET内核系统调用中断处理程序系统调用int 0x80下一语句信号处理程序用户程序更多电子书教程下载请登陆http://down.zzbaike.com/ebook本站提供的电子书教程均为网上搜集,如果该教程涉及或侵害到您的版权请联系我们。第5
章 内核代码 linux/kernel/11329 {30 int i;3132 verify_area(to, sizeof(struct sigaction)); // 验证to 处的内存是否足够。33 for (i=0 ; i< sizeof(struct sigaction) ; i++) {34 put_fs_byte(*from,to); // 复制到fs 段。一般是用户数据段。35 from++; // put_fs_byte()在include/asm/segment.h 中。36 to++;37
}38 }39// 把sigaction 数据从fs 数据段from 位置复制到to 处。40 static inline void get_new(char * from,char * to)41 {42 int i;4344 for (i=0 ; i< sizeof(struct sigaction) ; i++)45 *(to++) = get_fs_byte(from++);46 }47// signal()系统调用。类似于sigaction()。为指定的信号安装新的信号句柄(信号处理程序)。//
信号句柄可以是用户指定的函数,也可以是SIG_DFL(默认句柄)或SIG_IGN(忽略)。// 参数signum --指定的信号;handler -- 指定的句柄;restorer –原程序当前执行的地址位置。// 函数返回原信号句柄。48 int sys_signal(int signum, long handler, long restorer)49 {50 struct sigaction tmp;5152 if (signum<1 signum="">32 || signum==SIGKILL) //
信号值要在(1-32)范围内,53 return -1; // 并且不得是SIGKILL。54 tmp.sa_handler = (void (*)(int)) handler; // 指定的信号处理句柄。55 tmp.sa_mask = 0; // 执行时的信号屏蔽码。56 tmp.sa_flags = SA_ONESHOT | SA_NOMASK; // 该句柄只使用1 次后就恢复到默认值,// 并允许信号在自己的处理句柄中收到。57 tmp.sa_restorer = (void (*)(void))
restorer; // 保存返回地址。58 handler = (long) current->sigaction[signum-1].sa_handler;59 current->sigaction[signum-1] = tmp;60 return handler;61 }62// sigaction()系统调用。改变进程在收到一个信号时的操作。signum 是除了SIGKILL 以外的任何// 信号。[如果新操作(action)不为空]则新操作被安装。如果oldaction 指针不为空,则原操作//
被保留到oldaction。成功则返回0,否则为-1。63 int sys_sigaction(int signum, const struct sigaction * action,64 struct sigaction * oldaction)65 {66 struct sigaction tmp;67// 信号值要在(1-32)范围内,并且信号SIGKILL 的处理句柄不能被改变。68 if (signum<1 signum="">32 || signum==SIGKILL)69 return -1;//
在信号的sigaction 结构中设置新的操作(动作)。70 tmp = current->sigaction[signum-1];更多电子书教程下载请登陆http://down.zzbaike.com/ebook本站提供的电子书教程均为网上搜集,如果该教程涉及或侵害到您的版权请联系我们。第5 章 内核代码 linux/kernel/11471 get_new((char *) action,72 (char *) (signum-1+current->sigaction));// 如果oldaction
指针不为空的话,则将原操作指针保存到oldaction 所指的位置。73 if (oldaction)74 save_old((char *) &tmp,(char *) oldaction);// 如果允许信号在自己的信号句柄中收到,则令屏蔽码为0,否则设置屏蔽本信号。75 if (current->sigaction[signum-1].sa_flags & SA_NOMASK)76 current->sigaction[signum-1].sa_mask = 0;77 else78 current->sigaction[signum-1].sa_mask
|= (1<<(signum-1));79 return 0;80 }81// 系统调用中断处理程序中真正的信号处理程序(在kernel/system_call.s,119 行)。// 该段代码的主要作用是将信号的处理句柄插入到用户程序堆栈中,并在本系统调用结束// 返回后立刻执行信号句柄程序,然后继续执行用户的程序。82 void do_signal(long signr,long eax, long ebx, long ecx, long edx,83 long fs, long es, long ds,84
long eip, long cs, long eflags,85 unsigned long * esp, long ss)86 {87 unsigned long sa_handler;88 long old_eip=eip;89 struct sigaction * sa = current->sigaction + signr - 1; //current->sigaction[signu-1]。90 int longs;91 unsigned long * tmp_esp;9293 sa_handler
= (unsigned long) sa->sa_handler;// 如果信号句柄为SIG_IGN(忽略),则返回;如果句柄为SIG_DFL(默认处理),则如果信号是// SIGCHLD 则返回,否则终止进程的执行94 if (sa_handler==1)95 return;96 if (!sa_handler) {97 if (signr==SIGCHLD)98 return;99 else100 do_exit(1<<(signr-1)); // [?? 为什么以信号位图为参数?不为什么!??]//
这里应该是do_exit(1<sa_flags & SA_ONESHOT)103 sa->sa_handler = NULL;// 下面这段代码将信号处理句柄插入到用户堆栈中,同时也将sa_restorer,signr,进程屏蔽码(如果// SA_NOMASK 没置位),eax,ecx,edx 作为参数以及原调用系统调用的程序返回指针及标志寄存器值// 压入堆栈。因此在本次系统调用中断(0x80)返回用户程序时会首先执行用户的信号句柄程序,然后// 再继续执行用户程序。// 将用户调用系统调用的代码指针eip
指向该信号处理句柄。104 *(&eip) = sa_handler;// 如果允许信号自己的处理句柄收到信号自己,则也需要将进程的阻塞码压入堆栈。105 longs = (sa->sa_flags & SA_NOMASK)?7:8;// 将原调用程序的用户的堆栈指针向下扩展7(或8)个长字(用来存放调用信号句柄的参数等),更多电子书教程下载请登陆http://down.zzbaike.com/ebook本站提供的电子书教程均为网上搜集,如果该教程涉及或侵害到您的版权请联系我们。第5 章 内核代码 linux/kernel/115//
并检查内存使用情况(例如如果内存超界则分配新页等)。106 *(&esp) -= longs;107 verify_area(esp,longs*4);// 在用户堆栈中从下到上存放sa_restorer, 信号signr, 屏蔽码blocked(如果SA_NOMASK 置位),// eax, ecx, edx, eflags 和用户程序原代码指针。108 tmp_esp=esp;109 put_fs_long((long) sa->sa_restorer,tmp_esp++);110 put_fs_long(signr,tmp_esp++);111
if (!(sa->sa_flags & SA_NOMASK))112 put_fs_long(current->blocked,tmp_esp++);113 put_fs_long(eax,tmp_esp++);114 put_fs_long(ecx,tmp_esp++);115 put_fs_long(edx,tmp_esp++);116 put_fs_long(eflags,tmp_esp++);117 put_fs_long(old_eip,tmp_esp++);118 current->blocked
|= sa->sa_mask; // 进程阻塞码(屏蔽码)添上sa_mask 中的码位。119 }1205.8.3 其它信息5.8.3.1 进程信号说明进程中的信号是用于进程之间通信的一种简单消息,通常是下表中的一个标号数值,并且不携带任何其它的信息。例如当一个子进程终止或结束时,就会产生一个标号为17 的SIGCHILD 信号发送给父进程,以通知父进程有关子进程的当前状态。关于一个进程如何处理收到的信号,一般有两种做法:一是程序的进程不去处理,此时该信号会由系统相应的默认信号处理程序进行处理;第二种做法是进程使用自己的信号处理程序来处理信号。
郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。