1
0
mirror of https://github.com/samba-team/samba.git synced 2024-12-27 03:21:53 +03:00
samba-mirror/lib/util/select.c
Daniel Kobras 5907b0cc1e sys_poll_intr: fix timeout arithmetic
Callers of sys_poll_intr() assume timeout to be in milliseconds like
poll(2) expects, but implementation used nanosecond units. Also make
sure timeout doesn't become infinite by mistake during time arithmetic.

Signed-off-by: Daniel Kobras <d.kobras@science-computing.de>
Reviewed-by: Volker Lendecke <vl@samba.org>
Reviewed-by: Stefan Metzmacher <metze@samba.org>
Reviewed-by: Jeremy Allison <jra@samba.org>

Autobuild-User(master): Jeremy Allison <jra@samba.org>
Autobuild-Date(master): Tue Jul 22 00:12:24 CEST 2014 on sn-devel-104
2014-07-22 00:12:24 +02:00

61 lines
1.5 KiB
C

/*
Unix SMB/Netbios implementation.
Version 3.0
Samba select/poll implementation
Copyright (C) Andrew Tridgell 1992-1998
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "includes.h"
#include "system/filesys.h"
#include "system/select.h"
#include "lib/util/select.h"
int sys_poll_intr(struct pollfd *fds, int num_fds, int timeout)
{
int orig_timeout = timeout;
struct timespec start;
int ret;
clock_gettime_mono(&start);
while (true) {
struct timespec now;
int64_t elapsed;
ret = poll(fds, num_fds, timeout);
if (ret != -1) {
break;
}
if (errno != EINTR) {
break;
}
/* Infinite timeout, no need to adjust. */
if (timeout < 0) {
continue;
}
clock_gettime_mono(&now);
elapsed = nsec_time_diff(&now, &start) / 1000000;
timeout = orig_timeout - elapsed;
/* Unlikely, but might happen eg. when getting traced.
* Make sure we're not hanging in this case.
*/
if (timeout < 0) {
timeout = 0;
}
};
return ret;
}