El tiempo restante en una seleccione() la llamada
Pregunta
Estoy usando select()
en Linux/BRAZO de la plataforma para ver si un socket udp ha recibido un paquete.Me gustaría saber cuánto tiempo se quedan en el seleccione llamar si se devuelve antes de que el tiempo de espera (después de haber detectado un paquete).
Algo a lo largo de las líneas de:
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);
}
}
Solución
Lo más seguro es ignorar la ambigüedad de la definición de select()
y el tiempo es a ti mismo.
Acaba de obtener el tiempo antes y después de la selección y restar que desde el intervalo deseado.
Otros consejos
Si recuerdo correctamente, la función select () trata el tiempo de espera y un parámetro de E / S y cuando select devuelve el tiempo restante se devuelve en la variable de tiempo de espera.
De lo contrario, tendrá que registrar la hora actual antes de llamar, y nuevamente después y obtener la diferencia entre los dos.
De " man select " en 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().
Deberá llamar a gettimeofday antes de llamar a select y luego a gettimeofday al salir.
[Editar] Parece que Linux es ligeramente diferente:
(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 () actualiza el argumento de tiempo de espera para reflejar el tiempo que ha pasado.
Tenga en cuenta que esto no es portátil en otros sistemas (de ahí la advertencia en el manual de OS X citado anteriormente) pero funciona con Linux.
Gilad
No use select, intente con fd mayor que 1024 con su código y vea lo que obtendrá.