Время, оставшееся на вызов select()
Вопрос
Я использую select()
на платформе Linux / ARM, чтобы увидеть, получил ли сокет udp пакет.Я хотел бы знать, сколько времени оставалось в вызове select, если он вернется до истечения тайм-аута (обнаружив пакет).
Что - то вроде:
int wait_fd(int fd, int msec)
{
struct timeval tv;
fd_set rws;
tv.tv_sec = msec / 1000ul;
tv.tv_usec = (msec % 1000ul) * 1000ul;
FD_ZERO( & rws);
FD_SET(fd, & rws);
(void)select(fd + 1, & rws, NULL, NULL, & tv);
if (FD_ISSET(fd, &rws)) { /* There is data */
msec = (tv.tv_sec * 1000) + (tv.tv_usec / 1000);
return(msec?msec:1);
} else { /* There is no data */
return(0);
}
}
Решение
Самое безопасное - игнорировать двусмысленное определение select()
и засеките время сами.
Просто получите время до и после выбора и вычтите его из нужного вам интервала.
Другие советы
Если я правильно помню, функция select() обрабатывает время ожидания и параметр ввода-вывода, и когда select возвращает оставшееся время, оно возвращается в переменной timeout.
В противном случае вам придется записать текущее время до вызова и еще раз после и получить разницу между ними.
Из "man select" в OSX:
Timeout is not changed by select(), and may be reused on subsequent calls, however it
is good style to re-ini-tialize it before each invocation of select().
Вам нужно будет вызвать gettimeofday перед вызовом select, а затем gettimeofday при выходе.
[Редактировать] Кажется, что Linux немного отличается:
(ii) The select function may update the timeout parameter to indicate
how much time was left. The pselect function does not change
this parameter.
On Linux, the function select modifies timeout to reflect the amount of
time not slept; most other implementations do not do this. This causes
problems both when Linux code which reads timeout is ported to other
operating systems, and when code is ported to Linux that reuses a
struct timeval for multiple selects in a loop without reinitializing
it. Consider timeout to be undefined after select returns.
Linux select() обновляет аргумент timeout, чтобы отразить прошедшее время.
Обратите внимание, что это не переносимо в другие системы (отсюда и предупреждение в руководстве по OS X, приведенное выше), но работает с Linux.
Гилад
Не используйте select, попробуйте использовать в вашем коде fd больше 1024 и посмотрите, что вы получите.