From 5c926d9ea4a9e0a0a2384f06c7583648cdff3ed6 Mon Sep 17 00:00:00 2001 From: Simon Tatham Date: Thu, 7 Feb 2019 18:21:06 +0000 Subject: [PATCH] Switch to using poll(2) in place of select(2). I've always thought poll was more hassle to set up, because if you want to reuse part of your pollfds list between calls then you have to index every fd by its position in the list as well as the fd number itself, which gives you twice as many indices to keep track of than if the fd is always its own key. But the problem is that select is fundamentally limited to the range of fds that can fit in an fd_set, which is not the range of fds that can _exist_, so I've had a change of heart and now have to go with poll. For the moment, I've surrounded it with a 'pollwrapper' structure that lets me treat it more or less like select, containing a tree234 that maps each fd to its location in the list, and also translating between the simple select r/w/x classification and the richer poll flags. That's let me do the migration with minimal disruption to the call sites. In future perhaps I can start using poll more directly, and/or using the richer flag system (though the latter might be fiddly because of sometimes being constrained to use the glib event loop). But this will do for now. --- Recipe | 3 +- cmdline.c | 2 +- configure.ac | 2 +- putty.h | 6 +-- unix/unix.h | 19 +++++++ unix/uxcons.c | 18 +++---- unix/uxnet.c | 2 +- unix/uxpgnt.c | 55 +++++++++----------- unix/uxplink.c | 69 +++++++++++------------- unix/uxpoll.c | 136 ++++++++++++++++++++++++++++++++++++++++++++++++ unix/uxsel.c | 4 +- unix/uxserver.c | 48 +++++++---------- unix/uxsftp.c | 56 +++++++++----------- 13 files changed, 272 insertions(+), 148 deletions(-) create mode 100644 unix/uxpoll.c diff --git a/Recipe b/Recipe index b62a2db1..23e30af6 100644 --- a/Recipe +++ b/Recipe @@ -288,7 +288,7 @@ MISCNETCOMMON = timing callback MISC version tree234 CONF MISCNET = MISCNETCOMMON be_misc settings proxy WINMISC = MISCNET winstore winnet winhandl cmdline windefs winmisc winproxy + wintime winhsock errsock winsecur winucs miscucs winmiscs -UXMISCCOMMON = MISCNETCOMMON uxstore uxsel uxnet uxpeer uxmisc time +UXMISCCOMMON = MISCNETCOMMON uxstore uxsel uxpoll uxnet uxpeer uxmisc time + uxfdsock errsock UXMISC = MISCNET UXMISCCOMMON uxproxy uxutils @@ -365,6 +365,7 @@ PUTTYGEN_UNIX = sshrsag sshdssg sshprime sshdes ARITH sshmd5 version sshprng + sshrand uxnoise sshsha MISC sshrsa sshdss uxcons uxstore uxmisc + sshpubk sshaes sshsh256 sshsh512 IMPORT puttygen.res time tree234 + uxgen notiming CONF sshecc sshecdsag uxnogtk sshauxcrypt sshhmac + + uxpoll puttygen : [U] cmdgen PUTTYGEN_UNIX cgtest : [UT] cgtest PUTTYGEN_UNIX diff --git a/cmdline.c b/cmdline.c index 06f1520f..b3826a05 100644 --- a/cmdline.c +++ b/cmdline.c @@ -117,7 +117,7 @@ int cmdline_get_passwd_input(prompts_t *p) * transfer tools (psftp, pscp) can't do a great deal with protocol * selections (ever tried running scp over telnet?) or with port * forwarding (even if it wasn't a hideously bad idea, they don't - * have the select() infrastructure to make them work). + * have the select/poll infrastructure to make them work). */ int cmdline_tooltype = 0; diff --git a/configure.ac b/configure.ac index 9548ba87..35552ed2 100644 --- a/configure.ac +++ b/configure.ac @@ -89,7 +89,7 @@ case "$gtk_version_desired" in *) AC_ERROR([Invalid GTK version specified]) esac -AC_CHECK_HEADERS([utmpx.h sys/select.h],,,[ +AC_CHECK_HEADERS([utmpx.h],,,[ #include #include ]) diff --git a/putty.h b/putty.h index a261bf50..54206ccb 100644 --- a/putty.h +++ b/putty.h @@ -2052,7 +2052,7 @@ bool open_for_write_would_lose_data(const Filename *fn); * The reason for this is that an OS's system clock might not agree * exactly with the timing mechanisms it supplies to wait for a * given interval. I'll illustrate this by the simple example of - * Unix Plink, which uses timeouts to select() in a way which for + * Unix Plink, which uses timeouts to poll() in a way which for * these purposes can simply be considered to be a wait() function. * Suppose, for the sake of argument, that this wait() function * tends to return early by 1%. Then a possible sequence of actions @@ -2124,12 +2124,12 @@ unsigned long timing_last_clock(void); * instead request notifications when a callback is available, so that * it knows to ask its delegate event loop to do the same thing. Also, * if a front end needs to know whether a callback is pending without - * actually running it (e.g. so as to put a zero timeout on a select() + * actually running it (e.g. so as to put a zero timeout on a poll() * call) then it can call toplevel_callback_pending(), which will * return true if at least one callback is in the queue. * * run_toplevel_callbacks() returns true if it ran any actual code. - * This can be used as a means of speculatively terminating a select + * This can be used as a means of speculatively terminating a poll * loop, as in PSFTP, for example - if a callback has run then perhaps * it might have done whatever the loop's caller was waiting for. */ diff --git a/unix/unix.h b/unix/unix.h index 684080f7..284d1430 100644 --- a/unix/unix.h +++ b/unix/unix.h @@ -426,4 +426,23 @@ char *gtk_askpass_main(const char *display, const char *wintitle, */ extern const SftpServerVtable unix_live_sftpserver_vt; +/* + * uxpoll.c. + */ +typedef struct pollwrapper pollwrapper; +pollwrapper *pollwrap_new(void); +void pollwrap_free(pollwrapper *pw); +void pollwrap_clear(pollwrapper *pw); +void pollwrap_add_fd_events(pollwrapper *pw, int fd, int events); +void pollwrap_add_fd_rwx(pollwrapper *pw, int fd, int rwx); +int pollwrap_poll_instant(pollwrapper *pw); +int pollwrap_poll_endless(pollwrapper *pw); +int pollwrap_poll_timeout(pollwrapper *pw, int milliseconds); +int pollwrap_get_fd_events(pollwrapper *pw, int fd); +int pollwrap_get_fd_rwx(pollwrapper *pw, int fd); +static inline bool pollwrap_check_fd_rwx(pollwrapper *pw, int fd, int rwx) +{ + return (pollwrap_get_fd_rwx(pw, fd) & rwx) != 0; +} + #endif /* PUTTY_UNIX_H */ diff --git a/unix/uxcons.c b/unix/uxcons.c index f57ac26f..3a66844c 100644 --- a/unix/uxcons.c +++ b/unix/uxcons.c @@ -13,9 +13,6 @@ #include #include #include -#ifndef HAVE_NO_SYS_SELECT_H -#include -#endif #include "putty.h" #include "storage.h" @@ -122,12 +119,13 @@ void timer_change_notify(unsigned long next) /* * Wrapper around Unix read(2), suitable for use on a file descriptor * that's been set into nonblocking mode. Handles EAGAIN/EWOULDBLOCK - * by means of doing a one-fd select and then trying again; all other - * errors (including errors from select) are returned to the caller. + * by means of doing a one-fd poll and then trying again; all other + * errors (including errors from poll) are returned to the caller. */ static int block_and_read(int fd, void *buf, size_t len) { int ret; + pollwrapper *pw = pollwrap_new(); while ((ret = read(fd, buf, len)) < 0 && ( #ifdef EAGAIN @@ -138,18 +136,18 @@ static int block_and_read(int fd, void *buf, size_t len) #endif false)) { - fd_set rfds; - FD_ZERO(&rfds); - FD_SET(fd, &rfds); + pollwrap_clear(pw); + pollwrap_add_fd_rwx(pw, fd, SELECT_R); do { - ret = select(fd+1, &rfds, NULL, NULL, NULL); + ret = pollwrap_poll_endless(pw); } while (ret < 0 && errno == EINTR); assert(ret != 0); if (ret < 0) return ret; - assert(FD_ISSET(fd, &rfds)); + assert(pollwrap_check_fd_rwx(pw, fd, SELECT_R)); } + pollwrap_free(pw); return ret; } diff --git a/unix/uxnet.c b/unix/uxnet.c index b8c9c505..16680fb9 100644 --- a/unix/uxnet.c +++ b/unix/uxnet.c @@ -1403,7 +1403,7 @@ static void net_select_result(int fd, int event) case SELECT_W: /* writable */ if (!s->connected) { /* - * select() reports a socket as _writable_ when an + * select/poll reports a socket as _writable_ when an * asynchronous connect() attempt either completes or * fails. So first we must find out which. */ diff --git a/unix/uxpgnt.c b/unix/uxpgnt.c index 5a1f6a51..e5f14c97 100644 --- a/unix/uxpgnt.c +++ b/unix/uxpgnt.c @@ -861,20 +861,17 @@ void run_agent(void) now = GETTICKCOUNT(); + pollwrapper *pw = pollwrap_new(); + while (!time_to_die) { - fd_set rset, wset, xset; - int maxfd; int rwx; int ret; unsigned long next; - FD_ZERO(&rset); - FD_ZERO(&wset); - FD_ZERO(&xset); - maxfd = 0; + pollwrap_clear(pw); if (signalpipe[0] >= 0) { - FD_SET_MAX(signalpipe[0], maxfd, rset); + pollwrap_add_fd_rwx(pw, signalpipe[0], SELECT_R); } /* Count the currently active fds. */ @@ -886,30 +883,21 @@ void run_agent(void) sgrowarray(fdlist, fdsize, i); /* - * Add all currently open fds to the select sets, and store - * them in fdlist as well. + * Add all currently open fds to pw, and store them in fdlist + * as well. */ int fdcount = 0; for (fd = first_fd(&fdstate, &rwx); fd >= 0; fd = next_fd(&fdstate, &rwx)) { fdlist[fdcount++] = fd; - if (rwx & 1) - FD_SET_MAX(fd, maxfd, rset); - if (rwx & 2) - FD_SET_MAX(fd, maxfd, wset); - if (rwx & 4) - FD_SET_MAX(fd, maxfd, xset); + pollwrap_add_fd_rwx(pw, fd, rwx); } if (toplevel_callback_pending()) { - struct timeval tv; - tv.tv_sec = 0; - tv.tv_usec = 0; - ret = select(maxfd, &rset, &wset, &xset, &tv); + ret = pollwrap_poll_instant(pw); } else if (run_timers(now, &next)) { unsigned long then; long ticks; - struct timeval tv; then = now; now = GETTICKCOUNT(); @@ -917,22 +905,27 @@ void run_agent(void) ticks = 0; else ticks = next - now; - tv.tv_sec = ticks / 1000; - tv.tv_usec = ticks % 1000 * 1000; - ret = select(maxfd, &rset, &wset, &xset, &tv); - if (ret == 0) + + bool overflow = false; + if (ticks > INT_MAX) { + ticks = INT_MAX; + overflow = true; + } + + ret = pollwrap_poll_timeout(pw, ticks); + if (ret == 0 && !overflow) now = next; else now = GETTICKCOUNT(); } else { - ret = select(maxfd, &rset, &wset, &xset, NULL); + ret = pollwrap_poll_endless(pw); } if (ret < 0 && errno == EINTR) continue; if (ret < 0) { - perror("select"); + perror("poll"); exit(1); } @@ -952,20 +945,22 @@ void run_agent(void) for (i = 0; i < fdcount; i++) { fd = fdlist[i]; + int rwx = pollwrap_get_fd_rwx(pw, fd); /* * We must process exceptional notifications before * ordinary readability ones, or we may go straight * past the urgent marker. */ - if (FD_ISSET(fd, &xset)) + if (rwx & SELECT_X) select_result(fd, SELECT_X); - if (FD_ISSET(fd, &rset)) + if (rwx & SELECT_R) select_result(fd, SELECT_R); - if (FD_ISSET(fd, &wset)) + if (rwx & SELECT_W) select_result(fd, SELECT_W); } - if (signalpipe[0] >= 0 && FD_ISSET(signalpipe[0], &rset)) { + if (signalpipe[0] >= 0 && + pollwrap_check_fd_rwx(pw, signalpipe[0], SELECT_R)) { char c[1]; if (read(signalpipe[0], c, 1) <= 0) /* ignore error */; diff --git a/unix/uxplink.c b/unix/uxplink.c index ee7b6853..a654fa23 100644 --- a/unix/uxplink.c +++ b/unix/uxplink.c @@ -14,9 +14,6 @@ #include #include #include -#ifndef HAVE_NO_SYS_SELECT_H -#include -#endif #define PUTTY_DO_GLOBALS /* actually _define_ globals */ #include "putty.h" @@ -873,36 +870,33 @@ int main(int argc, char **argv) sending = false; now = GETTICKCOUNT(); + pollwrapper *pw = pollwrap_new(); + while (1) { - fd_set rset, wset, xset; - int maxfd; int rwx; int ret; unsigned long next; - FD_ZERO(&rset); - FD_ZERO(&wset); - FD_ZERO(&xset); - maxfd = 0; + pollwrap_clear(pw); - FD_SET_MAX(signalpipe[0], maxfd, rset); + pollwrap_add_fd_rwx(pw, signalpipe[0], SELECT_R); if (!sending && backend_connected(backend) && backend_sendok(backend) && backend_sendbuffer(backend) < MAX_STDIN_BACKLOG) { /* If we're OK to send, then try to read from stdin. */ - FD_SET_MAX(STDIN_FILENO, maxfd, rset); + pollwrap_add_fd_rwx(pw, STDIN_FILENO, SELECT_R); } if (bufchain_size(&stdout_data) > 0) { /* If we have data for stdout, try to write to stdout. */ - FD_SET_MAX(STDOUT_FILENO, maxfd, wset); + pollwrap_add_fd_rwx(pw, STDOUT_FILENO, SELECT_W); } if (bufchain_size(&stderr_data) > 0) { /* If we have data for stderr, try to write to stderr. */ - FD_SET_MAX(STDERR_FILENO, maxfd, wset); + pollwrap_add_fd_rwx(pw, STDERR_FILENO, SELECT_W); } /* Count the currently active fds. */ @@ -914,31 +908,22 @@ int main(int argc, char **argv) sgrowarray(fdlist, fdsize, i); /* - * Add all currently open fds to the select sets, and store - * them in fdlist as well. + * Add all currently open fds to pw, and store them in fdlist + * as well. */ int fdcount = 0; for (fd = first_fd(&fdstate, &rwx); fd >= 0; fd = next_fd(&fdstate, &rwx)) { fdlist[fdcount++] = fd; - if (rwx & 1) - FD_SET_MAX(fd, maxfd, rset); - if (rwx & 2) - FD_SET_MAX(fd, maxfd, wset); - if (rwx & 4) - FD_SET_MAX(fd, maxfd, xset); + pollwrap_add_fd_rwx(pw, fd, rwx); } if (toplevel_callback_pending()) { - struct timeval tv; - tv.tv_sec = 0; - tv.tv_usec = 0; - ret = select(maxfd, &rset, &wset, &xset, &tv); + ret = pollwrap_poll_instant(pw); } else if (run_timers(now, &next)) { do { unsigned long then; long ticks; - struct timeval tv; then = now; now = GETTICKCOUNT(); @@ -946,42 +931,48 @@ int main(int argc, char **argv) ticks = 0; else ticks = next - now; - tv.tv_sec = ticks / 1000; - tv.tv_usec = ticks % 1000 * 1000; - ret = select(maxfd, &rset, &wset, &xset, &tv); - if (ret == 0) + + bool overflow = false; + if (ticks > INT_MAX) { + ticks = INT_MAX; + overflow = true; + } + + ret = pollwrap_poll_timeout(pw, ticks); + if (ret == 0 && !overflow) now = next; else now = GETTICKCOUNT(); } while (ret < 0 && errno == EINTR); } else { - ret = select(maxfd, &rset, &wset, &xset, NULL); + ret = pollwrap_poll_endless(pw); } if (ret < 0 && errno == EINTR) continue; if (ret < 0) { - perror("select"); + perror("poll"); exit(1); } for (i = 0; i < fdcount; i++) { fd = fdlist[i]; + int rwx = pollwrap_get_fd_rwx(pw, fd); /* * We must process exceptional notifications before * ordinary readability ones, or we may go straight * past the urgent marker. */ - if (FD_ISSET(fd, &xset)) + if (rwx & SELECT_X) select_result(fd, SELECT_X); - if (FD_ISSET(fd, &rset)) + if (rwx & SELECT_R) select_result(fd, SELECT_R); - if (FD_ISSET(fd, &wset)) + if (rwx & SELECT_W) select_result(fd, SELECT_W); } - if (FD_ISSET(signalpipe[0], &rset)) { + if (pollwrap_check_fd_rwx(pw, signalpipe[0], SELECT_R)) { char c[1]; struct winsize size; if (read(signalpipe[0], c, 1) <= 0) @@ -991,7 +982,7 @@ int main(int argc, char **argv) backend_size(backend, size.ws_col, size.ws_row); } - if (FD_ISSET(STDIN_FILENO, &rset)) { + if (pollwrap_check_fd_rwx(pw, STDIN_FILENO, SELECT_R)) { char buf[4096]; int ret; @@ -1013,11 +1004,11 @@ int main(int argc, char **argv) } } - if (FD_ISSET(STDOUT_FILENO, &wset)) { + if (pollwrap_check_fd_rwx(pw, STDOUT_FILENO, SELECT_W)) { backend_unthrottle(backend, try_output(false)); } - if (FD_ISSET(STDERR_FILENO, &wset)) { + if (pollwrap_check_fd_rwx(pw, STDERR_FILENO, SELECT_W)) { backend_unthrottle(backend, try_output(true)); } diff --git a/unix/uxpoll.c b/unix/uxpoll.c new file mode 100644 index 00000000..bbc5490d --- /dev/null +++ b/unix/uxpoll.c @@ -0,0 +1,136 @@ +#include + +#include "putty.h" +#include "tree234.h" + +struct pollwrapper { + struct pollfd *fds; + size_t nfd, fdsize; + tree234 *fdtopos; +}; + +typedef struct pollwrap_fdtopos pollwrap_fdtopos; +struct pollwrap_fdtopos { + int fd; + size_t pos; +}; + +static int pollwrap_fd_cmp(void *av, void *bv) +{ + pollwrap_fdtopos *a = (pollwrap_fdtopos *)av; + pollwrap_fdtopos *b = (pollwrap_fdtopos *)bv; + return a->fd < b->fd ? -1 : a->fd > b->fd ? +1 : 0; +} + +pollwrapper *pollwrap_new(void) +{ + pollwrapper *pw = snew(pollwrapper); + pw->fdsize = 16; + pw->nfd = 0; + pw->fds = snewn(pw->fdsize, struct pollfd); + pw->fdtopos = newtree234(pollwrap_fd_cmp); + return pw; +} + +void pollwrap_free(pollwrapper *pw) +{ + pollwrap_clear(pw); + freetree234(pw->fdtopos); + sfree(pw->fds); + sfree(pw); +} + +void pollwrap_clear(pollwrapper *pw) +{ + pw->nfd = 0; + for (pollwrap_fdtopos *f2p; + (f2p = delpos234(pw->fdtopos, 0)) != NULL ;) + sfree(f2p); +} + +void pollwrap_add_fd_events(pollwrapper *pw, int fd, int events) +{ + pollwrap_fdtopos *f2p, f2p_find; + + assert(fd >= 0); + + f2p_find.fd = fd; + f2p = find234(pw->fdtopos, &f2p_find, NULL); + if (!f2p) { + sgrowarray(pw->fds, pw->fdsize, pw->nfd); + size_t index = pw->nfd++; + pw->fds[index].fd = fd; + pw->fds[index].events = pw->fds[index].revents = 0; + + f2p = snew(pollwrap_fdtopos); + f2p->fd = fd; + f2p->pos = index; + pollwrap_fdtopos *added = add234(pw->fdtopos, f2p); + assert(added == f2p); + } + + pw->fds[f2p->pos].events |= events; +} + +#define SELECT_R_IN (POLLIN | POLLRDNORM | POLLRDBAND) +#define SELECT_W_IN (POLLOUT | POLLWRNORM | POLLWRBAND) +#define SELECT_X_IN (POLLPRI) + +#define SELECT_R_OUT (SELECT_R_IN | POLLERR | POLLHUP) +#define SELECT_W_OUT (SELECT_W_IN | POLLERR) +#define SELECT_X_OUT (SELECT_X_IN) + +void pollwrap_add_fd_rwx(pollwrapper *pw, int fd, int rwx) +{ + int events = 0; + if (rwx & SELECT_R) + events |= SELECT_R_IN; + if (rwx & SELECT_W) + events |= SELECT_W_IN; + if (rwx & SELECT_X) + events |= SELECT_X_IN; + pollwrap_add_fd_events(pw, fd, events); +} + +int pollwrap_poll_instant(pollwrapper *pw) +{ + return poll(pw->fds, pw->nfd, 0); +} + +int pollwrap_poll_endless(pollwrapper *pw) +{ + return poll(pw->fds, pw->nfd, -1); +} + +int pollwrap_poll_timeout(pollwrapper *pw, int milliseconds) +{ + assert(milliseconds >= 0); + return poll(pw->fds, pw->nfd, milliseconds); +} + +int pollwrap_get_fd_events(pollwrapper *pw, int fd) +{ + pollwrap_fdtopos *f2p, f2p_find; + + assert(fd >= 0); + + f2p_find.fd = fd; + f2p = find234(pw->fdtopos, &f2p_find, NULL); + if (!f2p) + return 0; + + return pw->fds[f2p->pos].revents; +} + +int pollwrap_get_fd_rwx(pollwrapper *pw, int fd) +{ + int revents = pollwrap_get_fd_events(pw, fd); + int rwx = 0; + if (revents & SELECT_R_OUT) + rwx |= SELECT_R; + if (revents & SELECT_W_OUT) + rwx |= SELECT_W; + if (revents & SELECT_X_OUT) + rwx |= SELECT_X; + return rwx; +} diff --git a/unix/uxsel.c b/unix/uxsel.c index d32eac63..c16d5714 100644 --- a/unix/uxsel.c +++ b/unix/uxsel.c @@ -4,8 +4,8 @@ * This module is a sort of all-purpose interchange for file * descriptors. At one end it talks to uxnet.c and pty.c and * anything else which might have one or more fds that need - * select()-type things doing to them during an extended program - * run; at the other end it talks to pterm.c or uxplink.c or + * select() or poll()-type things doing to them during an extended + * program run; at the other end it talks to pterm.c or uxplink.c or * anything else which might have its own means of actually doing * those select()-type things. */ diff --git a/unix/uxserver.c b/unix/uxserver.c index 206cc1d4..0e7a4512 100644 --- a/unix/uxserver.c +++ b/unix/uxserver.c @@ -37,9 +37,6 @@ #include #include #include -#ifndef HAVE_NO_SYS_SELECT_H -#include -#endif #define PUTTY_DO_GLOBALS /* actually _define_ globals */ #include "putty.h" @@ -558,17 +555,13 @@ int main(int argc, char **argv) now = GETTICKCOUNT(); + pollwrapper *pw = pollwrap_new(); while (!finished) { - fd_set rset, wset, xset; - int maxfd; int rwx; int ret; unsigned long next; - FD_ZERO(&rset); - FD_ZERO(&wset); - FD_ZERO(&xset); - maxfd = 0; + pollwrap_clear(pw); /* Count the currently active fds. */ i = 0; @@ -586,24 +579,15 @@ int main(int argc, char **argv) for (fd = first_fd(&fdstate, &rwx); fd >= 0; fd = next_fd(&fdstate, &rwx)) { fdlist[fdcount++] = fd; - if (rwx & 1) - FD_SET_MAX(fd, maxfd, rset); - if (rwx & 2) - FD_SET_MAX(fd, maxfd, wset); - if (rwx & 4) - FD_SET_MAX(fd, maxfd, xset); + pollwrap_add_fd_rwx(pw, fd, rwx); } if (toplevel_callback_pending()) { - struct timeval tv; - tv.tv_sec = 0; - tv.tv_usec = 0; - ret = select(maxfd, &rset, &wset, &xset, &tv); + ret = pollwrap_poll_instant(pw); } else if (run_timers(now, &next)) { do { unsigned long then; long ticks; - struct timeval tv; then = now; now = GETTICKCOUNT(); @@ -611,38 +595,44 @@ int main(int argc, char **argv) ticks = 0; else ticks = next - now; - tv.tv_sec = ticks / 1000; - tv.tv_usec = ticks % 1000 * 1000; - ret = select(maxfd, &rset, &wset, &xset, &tv); - if (ret == 0) + + bool overflow = false; + if (ticks > INT_MAX) { + ticks = INT_MAX; + overflow = true; + } + + ret = pollwrap_poll_timeout(pw, ticks); + if (ret == 0 && !overflow) now = next; else now = GETTICKCOUNT(); } while (ret < 0 && errno == EINTR); } else { - ret = select(maxfd, &rset, &wset, &xset, NULL); + ret = pollwrap_poll_endless(pw); } if (ret < 0 && errno == EINTR) continue; if (ret < 0) { - perror("select"); + perror("poll"); exit(1); } for (i = 0; i < fdcount; i++) { fd = fdlist[i]; + int rwx = pollwrap_get_fd_rwx(pw, fd); /* * We must process exceptional notifications before * ordinary readability ones, or we may go straight * past the urgent marker. */ - if (FD_ISSET(fd, &xset)) + if (rwx & SELECT_X) select_result(fd, SELECT_X); - if (FD_ISSET(fd, &rset)) + if (rwx & SELECT_R) select_result(fd, SELECT_R); - if (FD_ISSET(fd, &wset)) + if (rwx & SELECT_W) select_result(fd, SELECT_W); } diff --git a/unix/uxsftp.c b/unix/uxsftp.c index 3152f85e..69abf725 100644 --- a/unix/uxsftp.c +++ b/unix/uxsftp.c @@ -13,9 +13,6 @@ #include #include #include -#ifndef HAVE_NO_SYS_SELECT_H -#include -#endif #include "putty.h" #include "ssh.h" @@ -448,10 +445,9 @@ char *dir_file_cat(const char *dir, const char *file) */ static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok) { - fd_set rset, wset, xset; int i, *fdlist; size_t fdsize; - int fd, fdcount, fdstate, rwx, ret, maxfd; + int fd, fdcount, fdstate, rwx, ret; unsigned long now = GETTICKCOUNT(); unsigned long next; bool done_something = false; @@ -459,6 +455,8 @@ static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok) fdlist = NULL; fdsize = 0; + pollwrapper *pw = pollwrap_new(); + do { /* Count the currently active fds. */ @@ -472,10 +470,7 @@ static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok) /* Expand the fdlist buffer if necessary. */ sgrowarray(fdlist, fdsize, i); - FD_ZERO(&rset); - FD_ZERO(&wset); - FD_ZERO(&xset); - maxfd = 0; + pollwrap_clear(pw); /* * Add all currently open fds to the select sets, and store @@ -485,29 +480,20 @@ static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok) for (fd = first_fd(&fdstate, &rwx); fd >= 0; fd = next_fd(&fdstate, &rwx)) { fdlist[fdcount++] = fd; - if (rwx & 1) - FD_SET_MAX(fd, maxfd, rset); - if (rwx & 2) - FD_SET_MAX(fd, maxfd, wset); - if (rwx & 4) - FD_SET_MAX(fd, maxfd, xset); + pollwrap_add_fd_rwx(pw, fd, rwx); } if (include_stdin) - FD_SET_MAX(0, maxfd, rset); + pollwrap_add_fd_rwx(pw, 0, SELECT_R); if (toplevel_callback_pending()) { - struct timeval tv; - tv.tv_sec = 0; - tv.tv_usec = 0; - ret = select(maxfd, &rset, &wset, &xset, &tv); + ret = pollwrap_poll_instant(pw); if (ret == 0) done_something |= run_toplevel_callbacks(); } else if (run_timers(now, &next)) { do { unsigned long then; long ticks; - struct timeval tv; then = now; now = GETTICKCOUNT(); @@ -515,38 +501,44 @@ static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok) ticks = 0; else ticks = next - now; - tv.tv_sec = ticks / 1000; - tv.tv_usec = ticks % 1000 * 1000; - ret = select(maxfd, &rset, &wset, &xset, &tv); - if (ret == 0) + + bool overflow = false; + if (ticks > INT_MAX) { + ticks = INT_MAX; + overflow = true; + } + + ret = pollwrap_poll_timeout(pw, ticks); + if (ret == 0 && !overflow) now = next; else now = GETTICKCOUNT(); } while (ret < 0 && errno == EINTR); } else { do { - ret = select(maxfd, &rset, &wset, &xset, NULL); + ret = pollwrap_poll_endless(pw); } while (ret < 0 && errno == EINTR); } } while (ret == 0 && !done_something); if (ret < 0) { - perror("select"); + perror("poll"); exit(1); } for (i = 0; i < fdcount; i++) { fd = fdlist[i]; + int rwx = pollwrap_get_fd_rwx(pw, fd); /* * We must process exceptional notifications before * ordinary readability ones, or we may go straight * past the urgent marker. */ - if (FD_ISSET(fd, &xset)) + if (rwx & SELECT_X) select_result(fd, SELECT_X); - if (FD_ISSET(fd, &rset)) + if (rwx & SELECT_R) select_result(fd, SELECT_R); - if (FD_ISSET(fd, &wset)) + if (rwx & SELECT_W) select_result(fd, SELECT_W); } @@ -554,7 +546,9 @@ static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok) run_toplevel_callbacks(); - return FD_ISSET(0, &rset) ? 1 : 0; + int toret = pollwrap_check_fd_rwx(pw, 0, SELECT_R) ? 1 : 0; + pollwrap_free(pw); + return toret; } /*