首页 VxWorks实时操作系统SIGNAL机制的应用

VxWorks实时操作系统SIGNAL机制的应用

举报
开通vip

VxWorks实时操作系统SIGNAL机制的应用VxWorks实时操作系统SIGNAL机制的应用 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 [摘要] 介绍VXWORKS实时操作系统的信号机制以及各种处理方法~特别是利用该机制实现异常情况的恢复和处理。 1. 概述 信号可用来在同一任务内部或不同任务之间实现异步通信,从而改变对多个任务的控制流程。所有任务或中断服务程序均能向指定的任务发送信号,该信号的接收任务将立刻挂起当前的执行线程,而激活任务指定的信号处理程序。信号处理程序是由用户定义的,它关联与特定的信号,而且任务...

VxWorks实时操作系统SIGNAL机制的应用
VxWorks实时操作系统SIGNAL机制的应用 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 [摘要] 介绍VXWORKS实时操作系统的信号机制以及各种处理方法~特别是利用该机制实现异常情况的恢复和处理。 1. 概述 信号可用来在同一任务内部或不同任务之间实现异步通信,从而改变对多个任务的控制流程。所有任务或中断服务程序均能向指定的任务发送信号,该信号的接收任务将立刻挂起当前的执行线程,而激活任务指定的信号处理程序。信号处理程序是由用户定义的,它关联与特定的信号,而且任务接收到该指定信号时的所有必要处理都在该程序中实现。信号的这种机制使得它特别适合于用来实现差错和异常处理。 2. 信号屏蔽 在信号处理时,可通过信号屏蔽来选择需要进行处理的信号,接收到被屏蔽的信号,即使指定了相应的处理程序,也不作任何处理。 为了实现对信号的屏蔽,需要定义数据类型为sigset_t的变量,同时必须包含头文件“signal.h”。 下面介绍实现这一功能的函数: int sigemptyset ( sigset_t *pSet ) 该函数初始化信号集,使得该信号集不包含任何信号; int sigfillset ( sigset_t *pSet ) 该函数初始化信号集,使得该信号集包含所有信号; int sigaddset ( sigset_t *pSet , int signo ) 该函数向信号集中增加新的信号; int sigdelset ( sigset_t *pSet , int signo ) 该函数删除信号集中的信号; int sigismember ( sigset_t *pSet , int signo ) 该函数用来判断信号集是否包含某信号; int sigprocmask ( int how, const sigset_t *pSet, sigset_t *pOset ) 该函数用来设置信号屏蔽;这里,pSet为新的信号集,pOset为当前的 信号集,而how则指示处理方式,其取值与处理方式对应关系如下: SIG_BLOCK 结果信号集为当前信号集和指定信号集二者的并集,通过这种方式, 可向当前信号集增添指定的元素; SIG_UNBLOCK 结果信号集为指定信号集的补集和当前信号集二者的交集;通过这 种方式,可从当前信号集删除指定的元素; 页码:1/8 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 SIG_SETMASK 结果信号集为指定信号集; 3. 信号发送 int raise (int signo ) 任务调用该函数来向自己发送指定的信号; int kill (int tid, int signo ) 该函数可向任何任务发送指定的信号; 4. 信号处理程序 如果一个任务指定了对于某个信号的处理程序,不管该任务是否被挂起时,接收到该信号将立刻调用信号处理程序,信号处理程序执行完毕后,任务恢复到先前的状态,比如:任务在挂起状态接收到该信号,那么处理程序执行完毕后任务将返回到挂起状态。 由于信号处理函数由操作系统调用,所以其函数形式有着严格的要求,包括下面的两种形式: 第一 方案 气瓶 现场处置方案 .pdf气瓶 现场处置方案 .doc见习基地管理方案.doc关于群访事件的化解方案建筑工地扬尘治理专项方案下载 :利用signal函数实现信号与处理函数的关联 void* signal( int signo, void* pHandler () ) () 该函数建立信号signo与处理函数pHandler之间的联系;同时,其返回当前信号处理函数的指针; void sigHandler (int signo); 单参数信号处理程序,这里sigHandler为用户自己定义的函数名(下同),该函数只提供了不同的信号编号signo,进一步的信号代码信息无法获取。 void sigHandler (int signo, int code, struct sigcontext * pContext); 多参数信号处理程序,该函数只提供了不同的信号编号signo和信号代码code信息,可对信号接收所更加详细的处理。 第二方案:利用sigaction函数实现信号与处理函数的关联 int sigaction ( int signo, const struct sigaction *pAct, struct sigaction *pOact ); 该函数建立信号signo与pAct结构中指定的处理函数之间的联系;同时,当前信号处理函数的有关信息将保存在结构pOact中; 另外,为了调用多参数处理程序,pAct结构中的成员sa_flags必须设置SA_SIGINFO; void sigHandler (int signo); 单参数信号处理程序,这里sigHandler为用户自己定义的函数名(下同),该函数只提供了不同的信号编号signo,进一步的信号代码信息无法获取。 void sigHandler (int signo, siginfo_t * pInfo, void * pContext); 多参数信号处理程序,该函数只提供了不同的信号编号signo和信号代码code信息,可对信号接收所更加详细的处理。 这里有必要介绍一下几个数据结构: 页码:2/8 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 typedef struct siginfo { int si_signo; /*信号的编号信息*/ int si_code; /*信号的来源信息*/ union sigval si_value; /*信号的代码信息*/ } siginfo_t; union sigval { int sival_int; /*信号的代码*/ void *sival_ptr; }; struct sigaction { union{ void (*__sa_handler)(int); void (*__sa_sigaction)(int, siginfo_t *, void *); } sa_u; #define sa_handler sa_u.__sa_handler /*单参数处理程序*/ #define sa_sigaction sa_u.__sa_sigaction /*多参数处理程序*/ sigset_t sa_mask; /*信号屏蔽集*/ int sa_flags; /*处理方式*/ }; 下面以Intel i386/i486为例进行进一步的解释: 列出信号定义如下: Signal Code Exception SIGILL ILL_DIVIDE_ERROR Divide error SIGEMT EMT_DEBUG Debugger call SIGILL ILL_NON_MASKABLE NMI interrupt SIGEMT EMT_BREAKPOINT Breakpoint SIGILL ILL_OVERFLOW INTO-detected overflow SIGILL ILL_BOUND Bound range exceeded SIGILL ILL_INVALID_OPCODE Invalid opcode SIGFPE FPE_NO_DEVICE Device not available SIGILL ILL_DOUBLE_FAULT Double fault SIGFPE FPE_CP_OVERRUN Coprocessor segment overrun SIGILL ILL_INVALID_TSS Invalid task state segment SIGBUS BUS_NO_SEGMENT Segment not present SIGBUS BUS_STACK_FAULT Stack exception SIGILL ILL_PROTECTION_FAULT General protection SIGBUS BUS_PAGE_FAULT Page fault SIGILL ILL_RESERVED (intel reserved) SIGFPE FPE_CP_ERROR Coprocessor Error SIGBUS BUS_ALIGNMENT Alignment check 对应于Signal的列也就是前面函数中的signo或第二方案中结构pInfo的成员si_signo,对应于Code的列也就是第一方案中的code和第二方案中结构pInfo 页码:3/8 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 的成员si_value.sival_int。具体实现参见后面的例子程序。 另外,VXWORKS操作系统定义了两个用户自定义信号,其对应的Signal列项为SIGUSR1和SIGUSR2,没有对应的Code列项。 5. 异常处理 当任务在执行过程中出现硬件异常时,将自动的发送相应的信号,用户可以利用这种机制来实现异常处理。利用setjmp和longjmp的相互配合可实现对异常情况的处理。为了利用这两个函数必须包含头文件“setjmp.h”,同时定义类型为jmp_buf公共变量。 int setjmp ( jmp_buf env ) 该函数用于保存任务的运行环境;正常情况下,其返回值为0,出现异 常时,其返回值为非0,具体值由函数longjmp的第二个参数确定。 另外,保存的运行环境不包括公共变量。 void longjmp ( jmp_buf env, int val ) 该函数恢复由保存的setjmp运行环境,并将控制转交给setjmp。 该机制的具体运用见后面的实例程序。 6. 程序实例 实例1:单参数信号处理程序以及信号的发送 #include #include #include #include void sigHandler(int sig); void myMain() { int A[10]; signal(SIGILL,&myHandler); signal(SIGBUS,&myHandler); signal(SIGUSR1,&myHandler); raise(SIGUSR1); A[11]=1; /*人为设置的下标越界差错*/ } void myHandler(int sig) { if(sig==SIGUSR1) printf("SIGUSR"); 页码:4/8 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 else { printf(" ERROR\n"); exit(0); } } 输出结果为: SIGUSR ERROR 实例2:信号屏蔽 #include #include #include #include #include void sigHandler(int sig); void myMain() { int A[10]; sigset_t sigset1,sigset2; sigemptyset(&sigset1); sigaddset(&sigset1,SIGUSR1); sigprocmask(SIG_SETMASK,&sigset1,&sigset2); signal(SIGILL,&sigHandler); signal(SIGBUS,&sigHandler); signal(SIGUSR1,&sigHandler); raise(SIGUSR1); A[11]=10; } void sigHandler(int sig) { if(sig==SIGUSR1) printf("SIGUSR"); else { printf(" ERROR\n"); exit(0); } } 页码:5/8 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 输出结果: ERROR 实例3:第一方案多参数处理程序 #include #include #include #include #include void sigHandler(int sigNum,int code, struct sigcontext *pContext); void myMain() { int A[10]; signal(SIGILL,&sigHandler); signal(SIGBUS,&sigHandler); signal(SIGUSR1,&sigHandler); raise(SIGUSR1); A[11]=10; } void sigHandler(int sigNum,int code, struct sigcontext *pContext) { if(sigNum==SIGUSR1) printf("SIGUSR"); else { printf(" ERROR\n"); if(code==BUS_PAGE_FAULT) printf(" PAGE_FAULT\n"); exit(0); } } 输出结果为: SIGUSR ERROR PAGE_FAULT 实例4:第二方案多参数处理程序 #include #include 页码:6/8 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 #include #include #include void sigHandler(int sigNum,siginfo_t *pInfo,void *pContext); void myMain() { int A[10]; struct sigaction act1,act2; act1.sa_sigaction=&sigHandler; act1.sa_flags=SA_SIGINFO; sigaction(SIGILL,&act1,&act2); sigaction(SIGBUS,&act1,&act2); sigaction(SIGUSR1,&act1,&act2); raise(SIGUSR1); A[11]=10; } void sigHandler(int sigNum,siginfo_t *pInfo,void *pContext) { if(sigNum==SIGUSR1) printf("SIGUSR"); else { printf(" ERROR\n"); if(pInfo->si_value.sival_int==BUS_PAGE_FAULT) printf("PAGE_FAULT\n"); exit(0); } } 输出结果为: SIGUSR ERROR PAGE_FAULT 实例5:异常处理 #include #include #include #include #include jmp_buf env; 页码:7/8 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 void sigHandler(int sig); void myMain() { int i,x,A[10]; signal(SIGILL,&sigHandler); signal(SIGBUS,&sigHandler); signal(SIGEMT,&sigHandler); signal(SIGFPE,&sigHandler); x=10; i=setjmp(env); if(i==0) { x=12; A[11]=10; } else { printf(" x=%d\n",x); exit(0); } } void sigHandler(int sig) { printf("Long Jump!"); longjmp(env,100); printf("Complete!"); } 输出结果: Long Jump! x=10 如果把变量x定义为全局变量,输出的结果为: Long Jump! x=10 上面的现象说明,当x定义为全局变量时,保存运行环境并没有保存变量x,否则在进行运行环境恢复时应该恢复x=10。 另外,信号处理函数sigHandler中位于longjmp后面的操作printf("Complete!") 并没有执行,而执行了任务中的printf(" x=%d\n",x)操作,说明longjmp已经将操作控制转交到了setjmp的位置。 利用这种机制,在主程序的必要位置调用日保存正常的运行环境,再在信号处理程序中调用longjmp以从异常中恢复到先前由setjmp保存的正常运行环境,同时通过setjmp的不同返回值判断是否发生异常以及作异常情况下程序的内部处理。 页码:8/8 ECC/BJ VXWORKS实时操作系统SIGNAL机制的应用 版本:1.0版 页码:9/8
本文档为【VxWorks实时操作系统SIGNAL机制的应用】,请使用软件OFFICE或WPS软件打开。作品中的文字与图均可以修改和编辑, 图片更改请在作品中右键图片并更换,文字修改请直接点击文字进行修改,也可以新增和删除文档中的内容。
该文档来自用户分享,如有侵权行为请发邮件ishare@vip.sina.com联系网站客服,我们会及时删除。
[版权声明] 本站所有资料为用户分享产生,若发现您的权利被侵害,请联系客服邮件isharekefu@iask.cn,我们尽快处理。
本作品所展示的图片、画像、字体、音乐的版权可能需版权方额外授权,请谨慎使用。
网站提供的党政主题相关内容(国旗、国徽、党徽..)目的在于配合国家政策宣传,仅限个人学习分享使用,禁止用于任何广告和商用目的。
下载需要: 免费 已有0 人下载
最新资料
资料动态
专题动态
is_769254
暂无简介~
格式:doc
大小:33KB
软件:Word
页数:14
分类:生活休闲
上传时间:2017-09-21
浏览量:47