Вопрос

Я использую 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 и посмотрите, что вы получите.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top