我正在创建一个大约每秒运行一次的计时器,它正在等待按下一个键(我没有这样做)。它正在运行时显示:
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;
}
}
}
答案 0 :(得分:2)
值得一提的是两件事:
select
,read
等功能会被信号中断。您可以在调用SA_RESTART
时设置sigaction
标记。 man signal(7)
:如果在系统调用或库函数调用被阻止时调用信号处理程序,则:
- 在信号处理程序返回后自动重新启动调用;或
- 呼叫失败并显示错误EINTR。
这两种行为中的哪一种取决于接口以及是否使用SA_RESTART标志建立了信号处理程序(请参阅sigaction(2))。 UNIX系统的细节各不相同;下面是Linux的详细信息。
或者,有一种方法可以让计时器不使用timer_create
和timerfd_create
。 select
接受超时参数,该参数可用于指定下一个计时器到期之前的时间。然后,如果发生超时,select
将返回0。此方法适用于其他事件多路分解API,例如poll
和epoll
。