如何在交付SIGALRM后重新安装SIGQUIT处理程序 - unix / c?

时间:2013-08-28 02:58:29

标签: c unix signals

我正在尝试设置一些信号处理程序。具体来说,我想安装一个SIGQUIT处理程序,它恢复默认处理,但在n秒后重新安装SIGQUIT处理程序。

当我最初在main()中安装我的信号处理程序时没有问题,但是当我添加代码以尝试在我的SIGALRM处理程序中重新安装我的SIGQUIT处理程序时,我在编译时不断收到以下错误消息:

ish.c: In function ‘SIGALRM_Handler’:
ish.c:42: error: ‘SIGQUIT_Handler’ undeclared (first use in this function)
ish.c:42: error: (Each undeclared identifier is reported only once
ish.c:42: error: for each function it appears in.)

以下是代码的相关摘要。我想要做的就是在触发警报时重新安装SIGQUIT处理程序,但我不确定如何继续?

/* SIGALRM handler */

static void SIGALRM_Handler(int iSig)
{
   void (*pfRet)(int);
   /* reinstall SIGQUIT handler */
   pfRet = signal(SIGQUIT, SIGQUIT_Handler);
   if (pfRet == SIG_ERR) { perror("SIGQUIT_Handler"); exit(EXIT_FAILURE); }
}

/*--------------------------------------------------------------------*/

/* SIGQUIT handler */

static void SIGQUIT_Handler(int iSig)
{
   void (*pfRet)(int);
   /* Print message followed by shell prompt */
   printf("Type Ctrl-\\ again within 5 seconds to exit\n");
   printf("------------------------------------\n");
   printf("%c ", '%');
   fflush(NULL);
   /* install default SIGQUIT handling */
   pfRet = signal(SIGQUIT, SIG_DFL);
   if (pfRet == SIG_ERR) { perror("SIGQUIT_Handler"); exit(EXIT_FAILURE); }
   /* set alarm for 5 seconds later */
   alarm(5);
}

...

int main(int argc, char **argv)
{
   FILE *psFile;
   void (*pfRet)(int);
   sigset_t sSet;
   int iRet;
   assert(argv != NULL);
   /* make sure needed signals are not blocked */
   sigemptyset(&sSet);
   sigaddset(&sSet, SIGINT);
   sigaddset(&sSet, SIGQUIT);
   sigaddset(&sSet, SIGALRM);
   iRet = sigprocmask(SIG_UNBLOCK, &sSet, NULL);
   if (iRet == -1) { perror(argv[0]); exit(EXIT_FAILURE); }
   /* install SIGINT handler */
   pfRet = signal(SIGINT, SIGINT_Handler);
   if (pfRet == SIG_ERR) { perror(argv[0]); exit(EXIT_FAILURE); }
   /* install SIGALRM handler */
   pfRet = signal(SIGALRM, SIGALRM_Handler);
   if (pfRet == SIG_ERR) { perror(argv[0]); exit(EXIT_FAILURE); }
   /* install SIGQUIT handler */
   pfRet = signal(SIGQUIT, SIGQUIT_Handler);
   if (pfRet == SIG_ERR) { perror(argv[0]); exit(EXIT_FAILURE); }
   /* check for configuration file */
   if (argc > 2)
   {
      fprintf(stderr, "Usage: ish [configfile]\n");
      exit(EXIT_FAILURE);
   }
   else if (argc == 2)
   {
      psFile = fopen(argv[1], "r");
      if (psFile != NULL)
      {
         handleProcess(psFile);
      }
      else
      {
         fprintf(stderr, "ish: ");
         perror(argv[1]);
      }
   }
   for(;;)
   {
      handleProcess(stdin);
   }
   return 0;
}

1 个答案:

答案 0 :(得分:2)

您需要在SIGQUIT_Handler()中将SIGALRM_Handler()的定义放在引用之前,或者在它之前添加声明:

static void SIGQUIT_Handler(int iSig);

要么可以工作,但有一个是必要的。为什么?当你写:

signal(SIGQUIT, SIGQUIT_Handler);

第二个参数必须是指向函数的指针,但编译器必须知道该名称表示指向函数的指针,因为它在使用之前已声明。从根本上说,名称SIGQUIT_Handler从未被声明或定义,因此编译器不知道它是什么。