选择中断的系统调用

时间:2016-01-28 09:15:05

标签: c linux timer embedded-linux

我正在创建一个大约每秒运行一次的计时器,它正在等待按下一个键(我没有这样做)。它正在运行时显示:

select : interrupted system call 
select : interrupted system call 
select : interrupted system call 
select : interrupted system call 

你能告诉我为什么会这样吗:

struct sigaction s1;
static timer_t tid3;    
sigfillset(&s1.sa_mask);
s1.sa_flags = SA_SIGINFO;
s1.sa_sigaction = SignalHandler;
if (sigaction(SIGU, &s1, NULL) == -1) 
{
  perror("s1 failed");
  exit( EXIT_FAILURE );
}
printf("\nTimer %d is setting up \n",TimerIdentity);    
tid3=SetTimer(SIGU, 1000, 1);

// ---------- SET timer values -------------------
static struct sigevent sigev;
static timer_t tid;
static struct itimerspec itval;
static struct itimerspec oitval;
sigev.sigev_notify = SIGEV_SIGNAL;
sigev.sigev_signo = signo;
sigev.sigev_value.sival_ptr = &tid;

if (timer_create(CLOCK_REALTIME, &sigev, &tid) == 0) 
{
    itval.it_value.tv_sec = sec/1000;
    itval.it_value.tv_nsec = (long)(sec % 1000) * (1000000L);
    //itval.it_value.tv_nsec = 0;

    if (mode == 1) 
    {
        itval.it_interval.tv_sec = itval.it_value.tv_sec;
        itval.it_interval.tv_nsec = itval.it_value.tv_nsec;
    }
    if (timer_settime(tid, 0, &itval, NULL) == 0) 
    {
        printf("Timer_settime \n");
    }
    else
    {
        perror("time_settime error!");
    }
}

//---------------- SIGNAL HANDLER ---------------- 

void SignalHandler(int signo, siginfo_t* info, void* context)
{
    else if (signo == SIGU) // for keypad being pressed
   {
      calltimer3function();
   }

}

//-----------------calltimer3function------------------------  

unsigned char key5_debounce=0,key5_debounce_count=0;
calltimer3function()
{
    if(!key5_debounce)
   {
       if((GPIORead(INPUT_SW5)==0))
       {
          key5_debounce=1;
       }
   }
   if(key5_debounce)
   {
       if((GPIORead(INPUT_SW5)==0))
       {
          key5_debounce_count++;
       }
       else
       key5_debounce=0;

        if(key5_debounce_count>=KEY_DEBOUNCE)
        {
           printf("key5 pressed\n");
           extr_count=1;
           printf("\nDisplay menu called");
           display_menu();

          key5_debounce=0;
          key5_debounce_count=0;
        }

   }
}

1 个答案:

答案 0 :(得分:2)

值得一提的是两件事:

  1. 阻止selectread等功能会被信号中断。您可以在调用SA_RESTART时设置sigaction标记。 man signal(7)
  2.   

    如果在系统调用或库函数调用被阻止时调用信号处理程序,则:

         
        
    • 在信号处理程序返回后自动重新启动调用;或
    •   
    • 呼叫失败并显示错误EINTR。
    •   
         

    这两种行为中的哪一种取决于接口以及是否使用SA_RESTART标志建立了信号处理程序(请参阅sigaction(2))。 UNIX系统的细节各不相同;下面是Linux的详细信息。

    1. 在信号处理程序中,您应该只调用异步信号安全功能。或者使用the self-pipe trick来避免在信号处理程序中做任何事情。
    2. 或者,有一种方法可以让计时器不使用timer_createtimerfd_createselect接受超时参数,该参数可用于指定下一个计时器到期之前的时间。然后,如果发生超时,select将返回0。此方法适用于其他事件多路分解API,例如pollepoll