`
fujinbing
  • 浏览: 238915 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

c语言信号处理

阅读更多

信号安装函数sigaction(int signum,const struct sigaction *act,struct sigaction *oldact)的第二个参数是一个指向sigaction结构的指针(结构体名称与函数名一样,千万别弄混淆了)。在结构sigaction的实例中,指定了对特定信号的处理,信号所传递的信息,信号处理函数执行过程中应屏蔽掉哪些函数等。当然,此指针也可以为NULL,进程会以默认方式处理信号。以下就简单介绍一下sigaction结构以及一般的用法。

        对于内核头文件而言,struct sigaction 结构体定义在kernel/include/asm/signal.h,此头文件又被kernel/include/linux/signal.h包含。
        对于用户空间的头文件而言,struct sigaction定义在 /usr/include/bits/sigaction.h,此头文件又被/usr/include/signal.h包含,所以应用程序中如果用到此结构,只要#include 即可。注意内核中的定义和应用程序中的定义是不一样的,内核空间的sigaction结构只支持函数类型为 __sighandler_t的信号处理函数,不能处理信号传递的额外信息。具体定义如下:

……
/* Type of a signal handler.   */
typedef void (*__sighandler_t)(int);

……
#ifdef __KERNEL__
struct old_sigaction {
          __sighandler_t sa_handler;
         old_sigset_t sa_mask;
         unsigned long sa_flags;
         void (*sa_restorer)(void);
};

struct sigaction {
         __sighandler_t sa_handler;
        unsigned long sa_flags;
        void (*sa_restorer)(void);
        sigset_t sa_mask;   /* mask last for extensibility */
};

struct k_sigaction {
        struct sigaction sa;
};

#else
/* Here we must cater to libcs that poke about in kernel headers.   */

struct sigaction {
          union {
                  __sighandler_t _sa_handler;
                  void (*_sa_sigaction)(int, struct siginfo *, void *);
          } _u;
          sigset_t sa_mask;
          unsigned long sa_flags;
          void (*sa_restorer)(void);
};

#define sa_handler   _u._sa_handler
#define sa_sigaction _u._sa_sigaction

#endif /* __KERNEL__ */

sa_handler的原型是一个参数为int,返回类型为void的函数指针。参数即为信号值,所以信号不能传递除信号值之外的任何信息;

sa_sigaction 的原型是一个带三个参数,类型分别为int,struct siginfo *,void *,返回类型为void的函数指针。第一个参数为信号值;第二个参数是一个指向struct siginfo结构的指针,此结构中包含信号携带的数据值;第三个参数没有使用。

sa_mask指定在信号处理程序执行过程中,哪些信号应当被阻塞。默认当前信号本身被阻塞。

sa_flags 包含了许多标志位,比较重要的一个是SA_SIGINFO,当设定了该标志位时,表示信号附带的参数可以传递到信号处理函数中。即使 sa_sigaction指定信号处理函数,如果不设置SA_SIGINFO,信号处理函数同样不能得到信号传递过来的数据,在信号处理函数中对这些信息的访问都将导致段错误。

sa_restorer已过时,POSIX不支持它,不应再使用。

        因此,当你的信号需要接收附加信息的时候,你必须给sa_sigaction赋信号处理函数指针,同时还要给sa_flags赋SA_SIGINFO,类似下面的代码:
     #include
     ……
     void sig_handler_with_arg(int sig,siginfo_t *sig_info,void *unused){……}
   
     int main(int argc,char **argv)
     {
              struct sigaction sig_act;
              ……
              sigemptyset(&sig_act.sa_mask);
              sig_act.sa_sigaction=sig_handler_with_arg;
              sig_act.sa_flags=SA_SIGINFO;
 
               ……
     }
        如果你的应用程序只需要接收信号,而不需要接收额外信息,那你需要的设置的是sa_handler,而不是sa_sigaction,你的程序可能类似下面的代码:
     #include
     ……
     void sig_handler(int sig){……}
   
     int main(int argc,char **argv)
     {
              struct sigaction sig_act;
              ……
              sigemptyset(&sig_act.sa_mask);
              sig_act.sa_handler=sig_handler;
              sig_act.sa_flags=0;
 
               ……
      }

      如果需要更详细说明,请参阅sigaction的man手册。

补充:

简而言之就是:

//自定义退出函数
    sigact.sa_handler = mysighandler;
    sigemptyset(&sigact.sa_mask);
    sigact.sa_flags = 0;
    sigaction(SIGINT, &sigact, NULL);
    sigaction(SIGTERM, &sigact, NULL);
    sigaction(SIGQUIT, &sigact, NULL);


或者利用signal函数进行信号捕捉:

void (*signal(int signo, void (*handler)(int)))(int);


当signal到来时,程序运行某函数,函数由你自己指定。



附带各种信号定义:

在终端使用kill -l 命令可以显示所有的信号。
$kill -l
1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL
5) SIGTRAP 6) SIGABRT 7) SIGBUS SIGFPE
9) SIGKILL 10) SIGUSR1 11) SIGSEGV 12) SIGUSR2
13) SIGPIPE 14) SIGALRM 15) SIGTERM 16) SIGSTKFLT
17) SIGCHLD 18) SIGCONT 19) SIGSTOP 20) SIGTSTP
21) SIGTTIN 22) SIGTTOU 23) SIGURG 24) SIGXCPU
25) SIGXFSZ 26) SIGVTALRM 27) SIGPROF 28) SIGWINCH
29) SIGIO 30) SIGPWR 31) SIGSYS 34) SIGRTMIN
35) SIGRTMIN+1 36) SIGRTMIN+2 37) SIGRTMIN+3 38) SIGRTMIN+4
39) SIGRTMIN+5 40) SIGRTMIN+6 41) SIGRTMIN+7 42) SIGRTMIN+8
43) SIGRTMIN+9 44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12
47) SIGRTMIN+13 48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14
51) SIGRTMAX-13 52) SIGRTMAX-12 53) SIGRTMAX-11 54) SIGRTMAX-10
55) SIGRTMAX-9 56) SIGRTMAX-8 57) SIGRTMAX-7 58) SIGRTMAX-6
59) SIGRTMAX-5 60) SIGRTMAX-4 61) SIGRTMAX-3 62) SIGRTMAX-2
63) SIGRTMAX-1 64) SIGRTMAX

其中前面31个信号为不可靠信号(非实时的,可能会出现信号的丢失),后面的信号为可靠信号(实时的real_time,对信号
排队,不会丢失)。

1) SIGHUP (挂起) 当运行进程的用户注销时通知该进程,使进程终止

2) SIGINT (中断) 当用户按下时,通知前台进程组终止进程

3) SIGQUIT (退出) 用户按下或时通知进程,使进程终止

4) SIGILL (非法指令) 执行了非法指令,如可执行文件本身出现错误、试图执行数据段、堆栈溢出

5) SIGTRAP 由断点指令或其它trap指令产生. 由debugger使用

6) SIGABRT (异常中止) 调用abort函数生成的信号

7) SIGBUS 非法地址, 包括内存地址对齐(alignment)出错. eg: 访问一个四个字长的整数, 但其地址不是4的倍数.

8) SIGFPE (算术异常) 发生致命算术运算错误,包括浮点运算错误、溢出及除数为0.

9) SIGKILL (确认杀死) 当用户通过kill -9命令向进程发送信号时,可靠的终止进程

10) SIGUSR1 用户使用

11) SIGSEGV (段越界) 当进程尝试访问不属于自己的内存空间导致内存错误时,终止进程

12) SIGUSR2 用户使用

13) SIGPIPE 写至无读进程的管道, 或者Socket通信SOCT_STREAM的读进程已经终止,而再写入。

14) SIGALRM (超时) alarm函数使用该信号,时钟定时器超时响应

15) SIGTERM (软中断) 使用不带参数的kill命令时终止进程

17) SIGCHLD (子进程结束) 当子进程终止时通知父进程

18) SIGCONT (暂停进程继续) 让一个停止(stopped)的进程继续执行. 本信号不能被阻塞.

19) SIGSTOP (停止) 作业控制信号,暂停停止(stopped)进程的执行. 本信号不能被阻塞, 处理或忽略.

20) SIGTSTP (暂停/停止) 交互式停止信号, Ctrl-Z 发出这个信号

21) SIGTTIN 当后台作业要从用户终端读数据时, 终端驱动程序产生SIGTTIN信号

22) SIGTTOU 当后台作业要往用户终端写数据时, 终端驱动程序产生SIGTTOU信号

23) SIGURG 有"紧急"数据或网络上带外数据到达socket时产生.

24) SIGXCPU 超过CPU时间资源限制. 这个限制可以由getrlimit/setrlimit来读取/改变。

25) SIGXFSZ 当进程企图扩大文件以至于超过文件大小资源限制。

26) SIGVTALRM 虚拟时钟信号. 类似于SIGALRM, 但是计算的是该进程占用的CPU时间.

27) SIGPROF (梗概时间超时) setitimer(2)函数设置的梗概统计间隔计时器(profiling interval timer)

28) SIGWINCH 窗口大小改变时发出.

29) SIGIO(异步I/O) 文件描述符准备就绪, 可以开始进行输入/输出操作.

30) SIGPWR 电源失效/重启动

31) SIGSYS 非法的系统调用。

在以上列出的信号中,
程序不可捕获、阻塞或忽略的信号有:SIGKILL,SIGSTOP
不能恢复至默认动作的信号有:SIGILL,SIGTRAP
默认会导致进程流产的信号有:SIGABRT,SIGBUS,SIGFPE,SIGILL,SIGIOT,SIGQUIT,SIGSEGV,SIGTRAP,SIGXCPU,SIGXFSZ
默认会导致进程退出的信号有:SIGALRM,SIGHUP,SIGINT,SIGKILL,SIGPIPE,SIGPOLL,SIGPROF,SIGSYS,SIGTERM,SIGUSR1,SIGUSR2,SIGVTALRM
默认会导致进程停止的信号有:SIGSTOP,SIGTSTP,SIGTTIN,SIGTTOU
默认进程忽略的信号有:SIGCHLD,SIGPWR,SIGURG,SIGWINCH

此外,SIGIO在SVR4是退出,在4.3BSD中是忽略;SIGCONT在进程挂起时是继续,否则是忽略,不能被阻塞。


在Unix/Linux中signal函数是比较复杂的一个,其定义原型如下:
void (*signal(int signo,void (*func)(int))) (int)
这个函数中,最外层的函数体
void (* XXX )(int)表明是一个指针,指向一个函数XXX的指针,XXX所代表的函数需要一个int型的参数,返回void
signal(int signo, void(*func)(int))是signal函数的主体.
需要两个参数int型的signo以及一个指向函数的函数.
void (*func)(int).
正是由于其复杂性,在[Plauger 1992]用typedef来对其进行简化
typedef void Sigfuc(int);//这里可以看成一个返回值 .
再对signal函数进行简化就是这样的了
Sigfunc *signal(int,Sigfuc *);

在signal.h头文件中还有以下几个定义
#define SIG_ERR (void (*)())-1
#define SIG_DFL (void (*)())0
#define SIG_IGN (void (*)())1


处理不当导致的问题

http://www.diybl.com/course/3_program/c++/cppjs/20090831/173152.html
分享到:
评论

相关推荐

    c语言信号处理.pdf

    C语言信号处理是操作系统中进程通信的一种方式,它主要用于通知进程发生了异步事件。信号机制在C语言中被实现为软中断信号,它允许进程间互相传递消息,或者由内核因为内部事件向进程发送通知。信号不携带任何数据,...

    数字信号处理C语言程序集.zip_B2S_C语言_信号处理

    详细讲解了数字信号处理的C语言实现方法及其相对应的C语言程序

    c语言信号处理[参照].pdf

    《C语言信号处理详解》 在计算机编程领域,特别是在软件开发中,进程间的通信和控制是不可或缺的一部分。C语言提供了一种称为“信号”(Signal)的机制,它允许进程间互相传递消息,以响应异步事件。信号,也被称为...

    数字信号处理c语言程序集.rar_C++_C语言信号_信号处理C语言_数字信号处理C

    《数字信号处理C语言程序集》是一份涵盖了C++和C语言在数字信号处理领域的实践应用资源。这个压缩包文件包含了一系列与信号处理相关的C语言程序,为学习者提供了宝贵的编程实例,帮助他们深入理解数字信号处理的概念...

    数字信号处理C语言程序集第三篇随机信号处理程序源码

    在本资源中,我们主要关注的是“数字信号处理”领域,特别是使用C语言进行编程实现的“随机信号处理”部分。数字信号处理是信息技术中的一个关键分支,它涉及到对离散时间或离散值信号的分析、转换和操作。C语言因其...

    数字信号处理C语言版

    数字信号处理方法的实现,使用了C语言,包括FIR,IIR,卷积等

    数字信号处理C语言程序集资料.rar

    《数字信号处理C语言程序集》是一份宝贵的资源,它结合了数字信号处理的理论与C语言编程实践,为学习者提供了深入理解该领域的重要工具。数字信号处理是电子工程、计算机科学和通信技术等领域不可或缺的一部分,而...

    数字信号处理C语言程序集.pdf数字信号处理C语言程序集.pdf

    《数字信号处理C语言程序集》是一本由殷福亮和宋爱军共同编著的专业技术书籍,由辽宁科学技术出版社于1997年7月首次出版,全书共有444页,深入探讨了数字信号处理领域的核心概念、算法以及其实现方式,特别聚焦于...

    数字信号处理(c语言)

    《数字信号处理(c语言)》是一份专注于使用C语言实现数字信号处理技术的教程或代码集合。C语言因其高效、灵活以及对底层硬件的良好控制,常被用于编写此类复杂的算法。下面将详细介绍数字信号处理的基本概念,以及...

    数字信号处理的函数库

    C语言作为一种广泛使用的编程语言,因其效率高、可移植性强,常被用于开发数字信号处理软件。本主题将深入探讨"数字信号处理的函数库",特别是C语言实现的相关知识。 首先,我们要了解数字信号处理的基本概念。数字...

    数字信号处理 C语言 DSP FFT DFT

    在数字信号处理领域,C语言因其高效性和灵活性,常被用于编写实时系统和嵌入式系统的算法实现。本文将深入探讨标题"数字信号处理 C语言 DSP FFT DFT"所涵盖的关键概念,以及压缩包中提供的两个PDF文件可能涉及的内容...

    数字信号处理C语言程序集 数字信号处理C语言程序集

    《数字信号处理C语言程序集》是一份专为学习和实践数字信号处理(Digital Signal Processing,简称DSP)的程序员设计的资源。这份程序集包含了使用C语言编写的多个示例程序,旨在帮助开发者理解并应用数字信号处理的...

    数字信号处理C语言程序

    《数字信号处理C语言程序》是一本深入浅出地探讨如何使用C语言进行数字信号处理的书籍。它涵盖了从基础的数字信号生成到复杂的图像处理和人工神经网络等多个领域,对于学习和实践数字信号处理的工程师和技术人员具有...

    数字信号处理c语言程序集

    数字信号处理c语言程序集

    数字信号处理C语言程序集合

    C语言作为一种高效的编程语言,在数字信号处理(Digital Signal Processing,简称DSP)领域中有着广泛的应用。本集合中的C语言程序,主要目的是提供一套参考代码,帮助读者更好地理解和运用数字信号处理技术。 数字...

    数字信号处理C语言程序集

    标题:“数字信号处理C语言程序集” 描述:“图像处理的C语言程序,常用数字信号产生,处理,数字图像处理” 从这些信息中,我们可以提炼出关于数字信号处理(Digital Signal Processing,简称DSP)以及如何使用...

Global site tag (gtag.js) - Google Analytics