0

我目前正在尝试设置一些信号处理程序。具体来说,我想安装一个 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;
}
4

1 回答 1

2

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

static void SIGQUIT_Handler(int iSig);

任何一种都可以,但其中一种是必要的。为什么?当你写:

signal(SIGQUIT, SIGQUIT_Handler);

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

于 2013-08-28T03:02:07.660 回答