《2022年操作系统课程设计方案信号机制实验 .pdf》由会员分享,可在线阅读,更多相关《2022年操作系统课程设计方案信号机制实验 .pdf(10页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、写在前面:这是 操作系统课程设计实验报告( 信号机制实验),前半部分是实验手册上的内容(老师发的),后面有该实验的实验报告,这个是我自己做的,不是网上搜来的,绝对原创,现在免费分享到文库上,目的是为了交流和共享。以后会边做边分享。虽然实验报告已经被老师审核通过,但如果还有不对的地方,请斧正。实验手册内容:(一) 信号机制实验实验目的1、了解什么是信号2、熟悉 LINUX 系统中进程之间软中断通信的基本原理实验内容1、编写程序:用fork()创建两个子进程,再用系统调用signal()让父进程捕捉键盘上来的中断信号(即按c 键);捕捉到中断信号后,父进程用系统调用kill( ) 向两个子进程发出
2、信号,子进程捕捉到信号后分别输出下列信息后终止:Child process1 is killed by parent! Child process2 is killed by parent! 父进程等待两个子进程终止后,输出如下的信息后终止:Parent process is killed! 2、分析利用软中断通信实现进程同步的机理实验指导一、信号1、信号的基本概念每个信号都对应一个正整数常量(称为 signal number,即信号编号。定义在系统头文件 中),代表同一用户的诸进程之间传送事先约定的信息的类型,用于通知某进程发生了某异常事件。每个进程在运行时,都要通过信号机制来检查是否有信号
3、到达。若有,便中断正在执行的程序,转向与该信号相对应的处理程序,以完成对该事件的处理;处理结束后再返回到原来的断点继续执行。实质上,信号机制是对中断机制的一种模拟,故在早期的UNIX 版本中又把它称为软中断。信号与中断的相似点:(1)采用了相同的异步通信方式;(2)当检测出有信号或中断请求时,都暂停正在执行的程序而转去执行相应的处理程序;(3)都在处理完毕后返回到原来的断点;精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 1 页,共 10 页(4)对信号或中断都可进行屏蔽。信号与中断的区别:(1)中断有优先级,而信号没有优先级,所有的信号都是平等的
4、;(2)信号处理程序是在用户态下运行的,而中断处理程序是在核心态下运行;(3)中断响应是及时的,而信号响应通常都有较大的时间延迟。信号机制具有以下三方面的功能:(1)发送信号。发送信号的程序用系统调用kill() 实现;(2)预置对信号的处理方式。接收信号的程序用signal( )来实现对处理方式的预置;(3)收受信号的进程按事先的规定完成对相应事件的处理。2、信号的发送信号的发送,是指由发送进程把信号送到指定进程的信号域的某一位上。如果目标进程正在一个可被中断的优先级上睡眠,核心便将它唤醒,发送进程就此结束。一个进程可能在其信号域中有多个位被置位,代表有多种类型的信号到达,但对于一类信号,进
5、程却只能记住其中的某一个。进程用 kill() 向一个进程或一组进程发送一个信号。3、对信号的处理当一个进程要进入或退出一个低优先级睡眠状态时,或一个进程即将从核心态返回用户态时,核心都要检查该进程是否已收到软中断。当进程处于核心态时,即使收到软中断也不予理睬;只有当它返回到用户态后,才处理软中断信号。 对软中断信号的处理分三种情况进行:(1)如果进程收到的软中断是一个已决定要忽略的信号(function=1),进程不做任何处理便立即返回;(2)进程收到软中断后便退出(function=0);(3)执行用户设置的软中断处理程序。二、所涉及的中断调用1、kill( )系统调用格式int kill
6、(pid,sig) 参数定义int pid,sig。其中, pid 是一个或一组进程的标识符,参数sig是要发送的软中断信号。(1)pid0 时,核心将信号发送给进程pid。(2)pid=0 时,核心将信号发送给与发送进程同组的所有进程。(3)pid=-1 时,核心将信号发送给所有用户标识符真正等于发送进程的有效用户标识号的进程。2、signal( )预置对信号的处理方式,允许调用进程控制软中断信号。系统调用格式signal(sig,function) 头文件为#include 参数定义精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 2 页,共 1
7、0 页signal(sig,function) int sig。void (*func) () 其中 sig 用于指定信号的类型, sig 为 0 则表示没有收到任何信号,余者如下表:值名 字说明01 SIGHUP 挂起( hangup)02 SIGINT 中断,当用户从键盘按 c键或break键时 (输入方式 Ctrl+C )03 SIGQUIT 退出,当用户从键盘按quit 键时04 SIGILL 非法指令05 SIGTRAP 跟踪陷阱( trace trap ),启动进程,跟踪代码的执行06 SIGIOT IOT 指令07 SIGEMT EMT 指令08 SIGFPE 浮点运算溢出09
8、SIGKILL 杀死、终止进程10 SIGBUS 总线错误11 SIGSEGV 段违例( segmentation violation),进程试图去访问其虚地址空间以外的位置12 SIGSYS 系统调用中参数错,如系统调用号非法13 SIGPIPE 向某个非读管道中写入数据14 SIGALRM 闹钟。当某进程希望在某时间后接收信号时发此信号15 SIGTERM 软件终止( software termination)16 SIGUSR1 用户自定义信号 1 17 SIGUSR2 用户自定义信号 2 18 SIGCLD 某个子进程死19 SIGPWR 电源故障function:在该进程中的一个函数
9、地址,在核心返回用户态时,它以软中断信号的序号作为参数调用该函数,对除了信号SIGKILL ,SIGTRAP 和 SIGPWR以外的信号,核心自动地重新设置软中断信号处理程序的值为SIG_DFL (进程终止) ,一个进程不能捕获SIGKILL 信号。function 的解释如下:(1)function=1(即 SIG_IGN )时,进程对sig 类信号不予理睬,亦即 屏蔽了该类信号;(2)function=0(即 SIG_DFL) 时,缺省值,进程在收到sig 信号后应终止自己;(3)function 为非 0、非 1 类整数时, function 的值即作为信号处理程序的指针。三、参考程序#
10、include #include #include void waiting( ),stop( )。精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 3 页,共 10 页int wait_mark。main( ) int p1,p2,stdout。while(p1=fork()=-1) 。 /*创建子进程 p1*/ if (p10) while(p2=fork()=-1) 。 /*创建子进程 p2*/ if(p20) wait_mark=1。signal(SIGINT,stop)。 /*接收到 c 信号,转 stop*/ waiting( )。kil
11、l(p1,16) 。 /*向 p1 发软中断信号 16*/ kill(p2,17) 。 /*向 p2 发软中断信号 17*/wait(0)。 /*同步*/ wait(0)。printf(Parent process is killed!n)。exit(0)。 else wait_mark=1。signal(17,stop)。 /*接收到软中断信号17,转 stop*/ waiting( )。lockf(stdout,1,0)。printf(Child process 2 is killed by parent!n) 。lockf(stdout,0,0)。exit(0)。 else wait_m
12、ark=1。signal(16,stop)。 /*接收到软中断信号16,转 stop*/ waiting( )。lockf(stdout,1,0)。printf(Child process 1 is killed by parent!n) 。lockf(stdout,0,0)。exit(0)。 void waiting( ) while(wait_mark!=0) 。 精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 4 页,共 10 页void stop( ) wait_mark=0。 四、运行结果屏幕上无反应,按下 C后,显示 Parent pr
13、ocess is killed! 五、分析原因上述程序中, signal()都放在一段程序的前面部位,而不是在其他接收信号处。这是因为 signal( )的执行只是为进程指定信号值16或 17的作用,以及分配相应的与stop()过程链接的指针。因而,signal()函数必须在程序前面部分执行。本方法通信效率低,当通信数据量较大时一般不用此法。六、思考1、该程序段前面部分用了两个wait(0),它们起什么作用?2、该程序段中每个进程退出时都用了语句exit(0),为什么?3、为何预期的结果并未显示出?4、(重点) 程序该如何修改才能得到正确结果?(实验报告中写出完整程序)5、*不修改程序如何得到
14、期望的输出?(给出合理的思路即可)以下为实验报告部分:实验 信号机制姓名 LeeQQ 615824191 实验目的1、了解什么是信号2、熟悉 LINUX 系统中进程之间软中断通信的基本原理实验内容1、编写程序:用fork()创建两个子进程,再用系统调用signal()让父进程捕捉键盘上来的中断信号(即按c 键);捕捉到中断信号后,父进程用系统调用kill( ) 向两个子进程发出信号,子进程捕捉到信号后分别输出下列信息后终止:Child process1 is killed by parent! Child process2 is killed by parent! 父进程等待两个子进程终止后,
15、输出如下的信息后终止:精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 5 页,共 10 页Parent process is killed! 2、分析利用软中断通信实现进程同步的机理实验步骤1.将参考程序用 vi 编辑器录入,然后编译执行#include #include #include void waiting( ),stop( )。int wait_mark。main( ) int p1,p2,stdout。while(p1=fork()=-1) 。 /*创建子进程 p1*/ if (p10) while(p2=fork()=-1) 。 /*
16、创建子进程 p2*/ if(p20) wait_mark=1。signal(SIGINT,stop)。 /*接收到 c 信号,转 stop*/ waiting( )。kill(p1,16) 。 /*向 p1发软中断信号 16*/ kill(p2,17) 。 /*向 p2发软中断信号 17*/ wait(0)。 /*同步*/ wait(0)。printf(Parent process is killed!n)。exit(0)。 else wait_mark=1。signal(17,stop)。 /*接收到软中断信号17,转 stop*/ waiting( )。lockf(stdout,1,0)。
17、printf(Child process 2 is killed by parent!n)。lockf(stdout,0,0)。exit(0)。 else wait_mark=1。signal(16,stop)。 /*接收到软中断信号16,转 stop*/ 精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 6 页,共 10 页waiting( )。lockf(stdout,1,0)。printf(Child process 1 is killed by parent!n)。lockf(stdout,0,0)。exit(0)。 void waiting
18、( ) while(wait_mark!=0) 。 void stop( ) wait_mark=0。 2.运行结果屏幕上无反应,按下 C 后,显示 Parent process is killed! 3.分析原因上述程序中, signal()都放在一段程序的前面部位,而不是在其他接收信号处。这是因为signal( )的执行只是为进程指定信号值16 或 17 的作用,以及分配相应的与stop()过程链接的指针。因而,signal()函数必须在程序前面部分执行。思考1. 该程序段前面部分用了两个wait(0),它们起什么作用?该程序段前面部分用的两个wait(0) 是为了让父进程等待子进程结束后
19、再结束自己,在此过程中,父进程处于挂起状态。2. 该程序段中每个进程退出时都用了语句exit(0),为什么?用 exit(0) 是为了让子进程自我终止,正常退出。3. 为何预期的结果并未显示出?p1、p2 都会捕捉该中断信号。对于父进程,当它捕捉到中断信号后就会就会转向程序中指定的函数“stop()”,当“ stop()”执行完毕后,父进程被唤醒,从中断处继续运行。对于子进程,由于没有给它们指定收到中断信精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 7 页,共 10 页号后的动作,它们会在捕捉到中断信号后执行默认操作,即结束自己。所以当我们发出中
20、断信号后,父进程按预计方式正常执行,而p1、p2 就被自己结束了,从而也就不会有预计的结果了。4. 程序该如何修改才能得到正确结果?为了使程序运行得到正确的结果,可以在每个子进程程序段开头加上忽略“ C”中断信号的语句,即: signal(SIGINT,SIG_IGN) 。#include #include #include #include void waiting() 。void stop() 。int wait_mark 。main() int p1,p2,stdout 。 while(p1 = fork() = -1) 。 if (p1 0) while(p2 = fork() = -
21、1) 。if (p2 0) wait_mark = 1 。 signal(SIGINT,stop) 。 waiting() 。 kill(p1,16) 。 kill(p2,17) 。 wait(0) 。 wait(0) 。 printf(Parent process is killed !n) 。 exit(0) 。 else signal(SIGINT,SIG_IGN) 。 wait_mark = 1 。 signal(17,stop) 。 waiting() 。 lockf(stdout,1,0) 。 printf(Child process 2 is killed by parent !
22、n) 。 lockf(stdout,0,0) 。 exit(0) 。 else signal(SIGINT,SIG_IGN) 。wait_mark = 1 。精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 8 页,共 10 页signal(16,stop) 。waiting() 。lockf(stdout,1,0) 。printf(Child process 1 is killed by parent !n) 。lockf(stdout,0,0) 。exit(0) 。 void waiting() while(wait_mark != 0) 。 v
23、oid stop() wait_mark = 0 。 4.不修改程序如何得到期望的输出?在不修改程序的情况下要输出期望的结果,可以单独向父进程发送SIGINT 信号,这样即可避免子进程由于收到SIGINT 信号执行默认操作而自我终止,具体实现方法:(该实验中程序名为: demo2)首先让程序在后台运行,命令:./demo2& 执行该命令后,会在后台生成3 个进程,使用ps 命令可以查看到它们的 PID (相对小的PID 应该为父进程的PID,原因是创建时间相对早)。然后向后台的父进程发送SIGINT 信号,命令:kill SIGINT 28664 (其中 28664为父进程的 PID) 具体实验结果见下图:精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 9 页,共 10 页心得与体会通过该实验,我对UNIX/LINUX 下软中断的实现原理与机制有了一个初步的认识,可以用kill() 函数和 signal() 函数实现简单的软中断程序,并且领会到了程序中wait_mark 所起的作用。在研究第五个思考题时,学习了进程在前台和后台间切换和向后台进程发送信号的方法,收获很多新知识和新方法。精选学习资料 - - - - - - - - - 名师归纳总结 - - - - - - -第 10 页,共 10 页
限制150内