linux自定义信号,并kill测试

原创 2015年07月07日 11:49:26

1. 自定义信号的说明

有时候我们需要在程序中利用信号来控制程序行为,linux为我们提供了2个已经定义的信号SIGUSR1和SIGUSR2,一般的程序利用这2个信号已经能满足需要,不过我最近需要一些其他信号来避免覆盖原来的信号处理函数。
    上网查了一下,看到了下面的程序片段:  

1
2
#define MYSIG_MSG        (SIGUSR2 + 1)
<p>// 定义信号然后注册处理函数</p>

   
    然后到系统里查了一下,MYSIG_MSG其实将其他的信号给覆盖了:
$kill -l 显示 10) SIGUSR1    11) SIGSEGV    12) SIGUSR2 13) SIGPIPE    14) SIGALRM
    虽然SIGPIPE和SIGALRM在这个程序中没有用到,但是这并不是我想要的效果。
    我发现在后面有 34) SIGRTMIN 35) SIGRTMIN+1    36) SIGRTMIN+2 ,man 7 signal页面同样也说明可以用 SIGRTMIN作为自定义信号。然后程序里就多了下面的代码: 


1
#define MYSIG_MSG        (SIGRTMIN+ 1)


    结果当然是出错了咯,但是并不是这个定义方式的问题。在我程序中有下面的代码:


1
2
3
4
switch(signo){
    case MYSIG_MSG:
    break;
}

    
    编译时才发现原来SIGRTMIN并不是一个常量,看了头文件里才知道:


1
2
// centos5.9 /usr/include/bits/signum.h
#define SIGRTMIN        (__libc_current_sigrtmin ())


    原来是函数调用,运行时确定的。
    要用这个SIGRTMIN宏是不行,只能自己定义了:


1
2
#define MYSIGRTMIN    34
#define MYSIG_MSG      (MYSIGRTMIN + 1)


    在找到系统定义的SIGRTMIN值之前,根据man 7 signal里面的说明:
    Linux supports 32 real-time signals, numbered from 32 (SIGRTMIN) to 63 (SIGRTMAX).
    我把自定义的信号值定义成了32,但是一直注册不了这个信号,后来赫然发现在 man 7 signal下面有一行说明,
    However, the glibc POSIX threads implementation internally uses two (for NPTL) or three  (for  LinuxThreads)  real-time signals  (see  pthreads(7)), and adjusts the value of SIGRTMIN suitably (to 34 or 35)
    这个说明在ubuntu12.04里面看见的,估计centos也有类似的情况。同时头文件下面也有:


1
2
3
4
/* These are the hard limits of the kernel.  These values should not be
used directly at user level.  */
#define __SIGRTMIN  32
#define __SIGRTMAX  (_NSIG - 1)

    改成34之后就没有问题了。虽然在man里面说明了程序不应该直接用常量标识信号,不过为了达到我的目的也顾不了了。

总结:

__SIGRTMIN基础上的前3个最好不要用,它是linuxthread用的


Sample:

//自定义信号
#define SIG_MY_DEFINE_TEST    __SIGRTMIN+10


2. 编程例子

#include <sys/wait.h>
#include<sys/types.h>
#include<unistd.h>
#include<stdio.h>
#include<stdlib.h>
#include<errno.h>

#include <signal.h>


//自定义信号
#define SIG_MY_DEFINE_TEST    __SIGRTMIN+10

void    sigactionProcess(int nsig)
{
	//================================================================
	//TODO:ADD YOU CODE

	printf("son process sigactionProcess end !pid is:%d\n", getpid()) ;
	//================================================================
	exit(0);
}
//信号处理函数注册
void    sigactionReg(void)
{
	struct sigaction act,oldact;

	act.sa_handler  = sigactionProcess;
	act.sa_flags = 0;

	//sigaction(SIGINT,&act,&oldact);
	sigaction(SIG_MY_DEFINE_TEST,&act,&oldact);
}



int main()
{
  int ret ;
  pid_t pid;/*pid 进程id号*/

  pid=fork();/*创建一个新进程*/

  if(pid==0) /*返回0为子进程*/
  {
      printf("son get Return pid is %d\n",pid);
      printf("This is son process!  pid is:%d\n",getpid());
      //信号处理函数注册
      sigactionReg();
      //raise(SIGSTOP) ;//子进程暂停
      while(1){}
      printf("son process end !pid is:%d\n", getpid()) ;
      exit(0) ;
  }
  else if(pid>0)/*返回大于0为父进程*/
  {
	  printf("parent get Return pid is %d\n",pid);
      printf("This is parent process!  pid is:%d\n",getpid());
      //
	  usleep(1000*10);
	    //获取到pid子进程没有退出,指定WNOHANG不会阻塞,没有退出会返回0
      if ((waitpid(pid, NULL, WNOHANG)) == 0)
      {
          //if ((ret = kill(pid, SIGINT)) == 0)//向子进程发出SIGKILL信号
          if ((ret = kill(pid, SIG_MY_DEFINE_TEST)) == 0)//向子进程发出SIG_MY_DEFINE_TEST信号
          {
               printf("parent kill %d\n", pid) ;
          }
      }
      waitpid(pid,NULL,0);/*等待子进程退出*/
      exit(0) ;
  }
  else
  {
       perror("fork() error!");
       exit(-1) ;
  }
}

测试

parent get Return pid is 10990
This is parent process!  pid is:10986
son get Return pid is 0
This is son process!  pid is:10990
parent kill 10990
son process sigactionProcess end !pid is:10990






参考:

   Linux下发送自定义信号 C++代码
   http://bbs.chinaunix.net/forum.php?mod=viewthread&tid=989499
   
   [转]linux 自定义信号(1)
   http://m.blog.csdn.net/blog/jmflovezlf/18217213
   
       linux自定义信号的处理
       http://my.oschina.net/u/566882/blog/174414
       
        linux进程通信---几个发送信号的函数(kill,raise,alarm,pause)
        http://blog.csdn.net/zzyoucan/article/details/9235685           
       
        Linux进程 Fork函数
        http://1793109.blog.51cto.com/1783109/593183  

版权声明:本文为博主原创文章,未经博主允许不得转载。

Linux编程,用户自定义信号

linux的进程之间使用用户自定义信号通信非常方面。 普通的系统信号 用户的自定义信号...
  • luopeiyuan1990
  • luopeiyuan1990
  • 2013年01月10日 13:42
  • 2744

linux 如何自定义信号

从来没试过linux自定义信号,查了下,说是系统只提供了SIGUSR1和SIGUSR2两个,就两个够吗?更要命的是如果要自定义信号如#define SIG_MYSIG   ....的话要改内核才行,哥...
  • killmice
  • killmice
  • 2016年12月19日 16:58
  • 716

linux信号集操作函数

信号集操作函数 内核通过读取未决信号集来判断信号是否应被处理。信号屏蔽字mask可以影响未决信号集。而我们可以在应用程序中自定义set来改变mask。已达到屏蔽指定信号的目的。 信号集设定 si...
  • oguro
  • oguro
  • 2016年12月24日 10:14
  • 314

linux信号通信---信号集函数组

信号处理的方法主要有两种,一种是使用signal()函数,另一种是使用信号集函数组。实例#include #include #include /*自定义的信号处理函数*/ void my_...
  • qq_27312943
  • qq_27312943
  • 2018年01月18日 21:47
  • 19

Linux自定义信号

信号同样是用于进程通信的,他是一种异步通信方式。我们之前的管道不是,明显,读进程要等管道里面有数据才能运行,否则他要等待。信号处理则不同,进程不知道什么时候信号会到来,先看一小段代码,看看信号的程序:...
  • luopeiyuan1990
  • luopeiyuan1990
  • 2013年11月22日 16:06
  • 3218

linux自定义信号处理

有时候我们需要在程序中利用信号来控制程序行为,linux为我们提供了2个已经定义的信号SIGUSR1和SIGUSR2,一般的程序利用这2个信号已经能满足需要,不过我最近需要一些其他信号来避免覆盖原来的...
  • bblueske
  • bblueske
  • 2013年11月05日 20:08
  • 461

linux下C 信号集处理函数

sigset_t  为信号集 可 sizeof(sigset_t) 查看  128K int sigemptyset(sigset_t * set) 清空设置 ...
  • ebw123
  • ebw123
  • 2016年01月19日 22:53
  • 599

Linux进程间通信(五)---信号通信之signal()、信号集函数组及其基础实验

上一节介绍进程间通信方式之一信号通信中的信号产生和捕捉函数,这一节介绍信号处理函数signal()函数和信号集函数组,接上一节http://blog.csdn.net/mybelief321/arti...
  • mybelief321
  • mybelief321
  • 2013年06月13日 10:16
  • 4221

Unix:用户自定义信号-SIGUSR1和SIGUSR2

http://blog.sina.com.cn/s/blog_ac9fdc0b0101jteb.html Unix:用户自定义信号-SIGUSR1和SIGUSR2...
  • qiezikuaichuan
  • qiezikuaichuan
  • 2015年10月13日 09:22
  • 1131

Linux下信号的基本使用与分析

1.基本概念信号是事件发生时对进程的通知机制,一个进程可以想另一个进程发送信号,此做法可以实现进程间的同步,然而给进程发信号的通常都是内核(1)内核一般都会在发生以下事件时给进程发信号 .硬件发生异...
  • Shreck66
  • Shreck66
  • 2015年07月26日 23:30
  • 1392
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:linux自定义信号,并kill测试
举报原因:
原因补充:

(最多只允许输入30个字)