1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-25 01:02:24 +00:00

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.
This commit is contained in:
Simon Tatham 2019-02-07 18:21:06 +00:00
parent 47202c4e16
commit 5c926d9ea4
13 changed files with 272 additions and 148 deletions

3
Recipe
View File

@ -288,7 +288,7 @@ MISCNETCOMMON = timing callback MISC version tree234 CONF
MISCNET = MISCNETCOMMON be_misc settings proxy MISCNET = MISCNETCOMMON be_misc settings proxy
WINMISC = MISCNET winstore winnet winhandl cmdline windefs winmisc winproxy WINMISC = MISCNET winstore winnet winhandl cmdline windefs winmisc winproxy
+ wintime winhsock errsock winsecur winucs miscucs winmiscs + 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 + uxfdsock errsock
UXMISC = MISCNET UXMISCCOMMON uxproxy uxutils 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 + sshrand uxnoise sshsha MISC sshrsa sshdss uxcons uxstore uxmisc
+ sshpubk sshaes sshsh256 sshsh512 IMPORT puttygen.res time tree234 + sshpubk sshaes sshsh256 sshsh512 IMPORT puttygen.res time tree234
+ uxgen notiming CONF sshecc sshecdsag uxnogtk sshauxcrypt sshhmac + uxgen notiming CONF sshecc sshecdsag uxnogtk sshauxcrypt sshhmac
+ uxpoll
puttygen : [U] cmdgen PUTTYGEN_UNIX puttygen : [U] cmdgen PUTTYGEN_UNIX
cgtest : [UT] cgtest PUTTYGEN_UNIX cgtest : [UT] cgtest PUTTYGEN_UNIX

View File

@ -117,7 +117,7 @@ int cmdline_get_passwd_input(prompts_t *p)
* transfer tools (psftp, pscp) can't do a great deal with protocol * transfer tools (psftp, pscp) can't do a great deal with protocol
* selections (ever tried running scp over telnet?) or with port * selections (ever tried running scp over telnet?) or with port
* forwarding (even if it wasn't a hideously bad idea, they don't * 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; int cmdline_tooltype = 0;

View File

@ -89,7 +89,7 @@ case "$gtk_version_desired" in
*) AC_ERROR([Invalid GTK version specified]) *) AC_ERROR([Invalid GTK version specified])
esac esac
AC_CHECK_HEADERS([utmpx.h sys/select.h],,,[ AC_CHECK_HEADERS([utmpx.h],,,[
#include <sys/types.h> #include <sys/types.h>
#include <utmp.h>]) #include <utmp.h>])

View File

@ -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 * 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 * exactly with the timing mechanisms it supplies to wait for a
* given interval. I'll illustrate this by the simple example of * 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. * these purposes can simply be considered to be a wait() function.
* Suppose, for the sake of argument, that this wait() function * Suppose, for the sake of argument, that this wait() function
* tends to return early by 1%. Then a possible sequence of actions * 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 * instead request notifications when a callback is available, so that
* it knows to ask its delegate event loop to do the same thing. Also, * 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 * 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 * call) then it can call toplevel_callback_pending(), which will
* return true if at least one callback is in the queue. * return true if at least one callback is in the queue.
* *
* run_toplevel_callbacks() returns true if it ran any actual code. * 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 * 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. * it might have done whatever the loop's caller was waiting for.
*/ */

View File

@ -426,4 +426,23 @@ char *gtk_askpass_main(const char *display, const char *wintitle,
*/ */
extern const SftpServerVtable unix_live_sftpserver_vt; 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 */ #endif /* PUTTY_UNIX_H */

View File

@ -13,9 +13,6 @@
#include <unistd.h> #include <unistd.h>
#include <fcntl.h> #include <fcntl.h>
#include <sys/time.h> #include <sys/time.h>
#ifndef HAVE_NO_SYS_SELECT_H
#include <sys/select.h>
#endif
#include "putty.h" #include "putty.h"
#include "storage.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 * Wrapper around Unix read(2), suitable for use on a file descriptor
* that's been set into nonblocking mode. Handles EAGAIN/EWOULDBLOCK * that's been set into nonblocking mode. Handles EAGAIN/EWOULDBLOCK
* by means of doing a one-fd select and then trying again; all other * by means of doing a one-fd poll and then trying again; all other
* errors (including errors from select) are returned to the caller. * errors (including errors from poll) are returned to the caller.
*/ */
static int block_and_read(int fd, void *buf, size_t len) static int block_and_read(int fd, void *buf, size_t len)
{ {
int ret; int ret;
pollwrapper *pw = pollwrap_new();
while ((ret = read(fd, buf, len)) < 0 && ( while ((ret = read(fd, buf, len)) < 0 && (
#ifdef EAGAIN #ifdef EAGAIN
@ -138,18 +136,18 @@ static int block_and_read(int fd, void *buf, size_t len)
#endif #endif
false)) { false)) {
fd_set rfds; pollwrap_clear(pw);
FD_ZERO(&rfds); pollwrap_add_fd_rwx(pw, fd, SELECT_R);
FD_SET(fd, &rfds);
do { do {
ret = select(fd+1, &rfds, NULL, NULL, NULL); ret = pollwrap_poll_endless(pw);
} while (ret < 0 && errno == EINTR); } while (ret < 0 && errno == EINTR);
assert(ret != 0); assert(ret != 0);
if (ret < 0) if (ret < 0)
return ret; return ret;
assert(FD_ISSET(fd, &rfds)); assert(pollwrap_check_fd_rwx(pw, fd, SELECT_R));
} }
pollwrap_free(pw);
return ret; return ret;
} }

View File

@ -1403,7 +1403,7 @@ static void net_select_result(int fd, int event)
case SELECT_W: /* writable */ case SELECT_W: /* writable */
if (!s->connected) { 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 * asynchronous connect() attempt either completes or
* fails. So first we must find out which. * fails. So first we must find out which.
*/ */

View File

@ -861,20 +861,17 @@ void run_agent(void)
now = GETTICKCOUNT(); now = GETTICKCOUNT();
pollwrapper *pw = pollwrap_new();
while (!time_to_die) { while (!time_to_die) {
fd_set rset, wset, xset;
int maxfd;
int rwx; int rwx;
int ret; int ret;
unsigned long next; unsigned long next;
FD_ZERO(&rset); pollwrap_clear(pw);
FD_ZERO(&wset);
FD_ZERO(&xset);
maxfd = 0;
if (signalpipe[0] >= 0) { 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. */ /* Count the currently active fds. */
@ -886,30 +883,21 @@ void run_agent(void)
sgrowarray(fdlist, fdsize, i); sgrowarray(fdlist, fdsize, i);
/* /*
* Add all currently open fds to the select sets, and store * Add all currently open fds to pw, and store them in fdlist
* them in fdlist as well. * as well.
*/ */
int fdcount = 0; int fdcount = 0;
for (fd = first_fd(&fdstate, &rwx); fd >= 0; for (fd = first_fd(&fdstate, &rwx); fd >= 0;
fd = next_fd(&fdstate, &rwx)) { fd = next_fd(&fdstate, &rwx)) {
fdlist[fdcount++] = fd; fdlist[fdcount++] = fd;
if (rwx & 1) pollwrap_add_fd_rwx(pw, fd, rwx);
FD_SET_MAX(fd, maxfd, rset);
if (rwx & 2)
FD_SET_MAX(fd, maxfd, wset);
if (rwx & 4)
FD_SET_MAX(fd, maxfd, xset);
} }
if (toplevel_callback_pending()) { if (toplevel_callback_pending()) {
struct timeval tv; ret = pollwrap_poll_instant(pw);
tv.tv_sec = 0;
tv.tv_usec = 0;
ret = select(maxfd, &rset, &wset, &xset, &tv);
} else if (run_timers(now, &next)) { } else if (run_timers(now, &next)) {
unsigned long then; unsigned long then;
long ticks; long ticks;
struct timeval tv;
then = now; then = now;
now = GETTICKCOUNT(); now = GETTICKCOUNT();
@ -917,22 +905,27 @@ void run_agent(void)
ticks = 0; ticks = 0;
else else
ticks = next - now; ticks = next - now;
tv.tv_sec = ticks / 1000;
tv.tv_usec = ticks % 1000 * 1000; bool overflow = false;
ret = select(maxfd, &rset, &wset, &xset, &tv); if (ticks > INT_MAX) {
if (ret == 0) ticks = INT_MAX;
overflow = true;
}
ret = pollwrap_poll_timeout(pw, ticks);
if (ret == 0 && !overflow)
now = next; now = next;
else else
now = GETTICKCOUNT(); now = GETTICKCOUNT();
} else { } else {
ret = select(maxfd, &rset, &wset, &xset, NULL); ret = pollwrap_poll_endless(pw);
} }
if (ret < 0 && errno == EINTR) if (ret < 0 && errno == EINTR)
continue; continue;
if (ret < 0) { if (ret < 0) {
perror("select"); perror("poll");
exit(1); exit(1);
} }
@ -952,20 +945,22 @@ void run_agent(void)
for (i = 0; i < fdcount; i++) { for (i = 0; i < fdcount; i++) {
fd = fdlist[i]; fd = fdlist[i];
int rwx = pollwrap_get_fd_rwx(pw, fd);
/* /*
* We must process exceptional notifications before * We must process exceptional notifications before
* ordinary readability ones, or we may go straight * ordinary readability ones, or we may go straight
* past the urgent marker. * past the urgent marker.
*/ */
if (FD_ISSET(fd, &xset)) if (rwx & SELECT_X)
select_result(fd, SELECT_X); select_result(fd, SELECT_X);
if (FD_ISSET(fd, &rset)) if (rwx & SELECT_R)
select_result(fd, SELECT_R); select_result(fd, SELECT_R);
if (FD_ISSET(fd, &wset)) if (rwx & SELECT_W)
select_result(fd, 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]; char c[1];
if (read(signalpipe[0], c, 1) <= 0) if (read(signalpipe[0], c, 1) <= 0)
/* ignore error */; /* ignore error */;

View File

@ -14,9 +14,6 @@
#include <pwd.h> #include <pwd.h>
#include <sys/ioctl.h> #include <sys/ioctl.h>
#include <sys/time.h> #include <sys/time.h>
#ifndef HAVE_NO_SYS_SELECT_H
#include <sys/select.h>
#endif
#define PUTTY_DO_GLOBALS /* actually _define_ globals */ #define PUTTY_DO_GLOBALS /* actually _define_ globals */
#include "putty.h" #include "putty.h"
@ -873,36 +870,33 @@ int main(int argc, char **argv)
sending = false; sending = false;
now = GETTICKCOUNT(); now = GETTICKCOUNT();
pollwrapper *pw = pollwrap_new();
while (1) { while (1) {
fd_set rset, wset, xset;
int maxfd;
int rwx; int rwx;
int ret; int ret;
unsigned long next; unsigned long next;
FD_ZERO(&rset); pollwrap_clear(pw);
FD_ZERO(&wset);
FD_ZERO(&xset);
maxfd = 0;
FD_SET_MAX(signalpipe[0], maxfd, rset); pollwrap_add_fd_rwx(pw, signalpipe[0], SELECT_R);
if (!sending && if (!sending &&
backend_connected(backend) && backend_connected(backend) &&
backend_sendok(backend) && backend_sendok(backend) &&
backend_sendbuffer(backend) < MAX_STDIN_BACKLOG) { backend_sendbuffer(backend) < MAX_STDIN_BACKLOG) {
/* If we're OK to send, then try to read from stdin. */ /* 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 (bufchain_size(&stdout_data) > 0) {
/* If we have data for stdout, try to write to stdout. */ /* 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 (bufchain_size(&stderr_data) > 0) {
/* If we have data for stderr, try to write to stderr. */ /* 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. */ /* Count the currently active fds. */
@ -914,31 +908,22 @@ int main(int argc, char **argv)
sgrowarray(fdlist, fdsize, i); sgrowarray(fdlist, fdsize, i);
/* /*
* Add all currently open fds to the select sets, and store * Add all currently open fds to pw, and store them in fdlist
* them in fdlist as well. * as well.
*/ */
int fdcount = 0; int fdcount = 0;
for (fd = first_fd(&fdstate, &rwx); fd >= 0; for (fd = first_fd(&fdstate, &rwx); fd >= 0;
fd = next_fd(&fdstate, &rwx)) { fd = next_fd(&fdstate, &rwx)) {
fdlist[fdcount++] = fd; fdlist[fdcount++] = fd;
if (rwx & 1) pollwrap_add_fd_rwx(pw, fd, rwx);
FD_SET_MAX(fd, maxfd, rset);
if (rwx & 2)
FD_SET_MAX(fd, maxfd, wset);
if (rwx & 4)
FD_SET_MAX(fd, maxfd, xset);
} }
if (toplevel_callback_pending()) { if (toplevel_callback_pending()) {
struct timeval tv; ret = pollwrap_poll_instant(pw);
tv.tv_sec = 0;
tv.tv_usec = 0;
ret = select(maxfd, &rset, &wset, &xset, &tv);
} else if (run_timers(now, &next)) { } else if (run_timers(now, &next)) {
do { do {
unsigned long then; unsigned long then;
long ticks; long ticks;
struct timeval tv;
then = now; then = now;
now = GETTICKCOUNT(); now = GETTICKCOUNT();
@ -946,42 +931,48 @@ int main(int argc, char **argv)
ticks = 0; ticks = 0;
else else
ticks = next - now; ticks = next - now;
tv.tv_sec = ticks / 1000;
tv.tv_usec = ticks % 1000 * 1000; bool overflow = false;
ret = select(maxfd, &rset, &wset, &xset, &tv); if (ticks > INT_MAX) {
if (ret == 0) ticks = INT_MAX;
overflow = true;
}
ret = pollwrap_poll_timeout(pw, ticks);
if (ret == 0 && !overflow)
now = next; now = next;
else else
now = GETTICKCOUNT(); now = GETTICKCOUNT();
} while (ret < 0 && errno == EINTR); } while (ret < 0 && errno == EINTR);
} else { } else {
ret = select(maxfd, &rset, &wset, &xset, NULL); ret = pollwrap_poll_endless(pw);
} }
if (ret < 0 && errno == EINTR) if (ret < 0 && errno == EINTR)
continue; continue;
if (ret < 0) { if (ret < 0) {
perror("select"); perror("poll");
exit(1); exit(1);
} }
for (i = 0; i < fdcount; i++) { for (i = 0; i < fdcount; i++) {
fd = fdlist[i]; fd = fdlist[i];
int rwx = pollwrap_get_fd_rwx(pw, fd);
/* /*
* We must process exceptional notifications before * We must process exceptional notifications before
* ordinary readability ones, or we may go straight * ordinary readability ones, or we may go straight
* past the urgent marker. * past the urgent marker.
*/ */
if (FD_ISSET(fd, &xset)) if (rwx & SELECT_X)
select_result(fd, SELECT_X); select_result(fd, SELECT_X);
if (FD_ISSET(fd, &rset)) if (rwx & SELECT_R)
select_result(fd, SELECT_R); select_result(fd, SELECT_R);
if (FD_ISSET(fd, &wset)) if (rwx & SELECT_W)
select_result(fd, 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]; char c[1];
struct winsize size; struct winsize size;
if (read(signalpipe[0], c, 1) <= 0) 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); 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]; char buf[4096];
int ret; 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)); 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)); backend_unthrottle(backend, try_output(true));
} }

136
unix/uxpoll.c Normal file
View File

@ -0,0 +1,136 @@
#include <poll.h>
#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;
}

View File

@ -4,8 +4,8 @@
* This module is a sort of all-purpose interchange for file * This module is a sort of all-purpose interchange for file
* descriptors. At one end it talks to uxnet.c and pty.c and * descriptors. At one end it talks to uxnet.c and pty.c and
* anything else which might have one or more fds that need * anything else which might have one or more fds that need
* select()-type things doing to them during an extended program * select() or poll()-type things doing to them during an extended
* run; at the other end it talks to pterm.c or uxplink.c or * 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 * anything else which might have its own means of actually doing
* those select()-type things. * those select()-type things.
*/ */

View File

@ -37,9 +37,6 @@
#include <pwd.h> #include <pwd.h>
#include <sys/ioctl.h> #include <sys/ioctl.h>
#include <sys/time.h> #include <sys/time.h>
#ifndef HAVE_NO_SYS_SELECT_H
#include <sys/select.h>
#endif
#define PUTTY_DO_GLOBALS /* actually _define_ globals */ #define PUTTY_DO_GLOBALS /* actually _define_ globals */
#include "putty.h" #include "putty.h"
@ -558,17 +555,13 @@ int main(int argc, char **argv)
now = GETTICKCOUNT(); now = GETTICKCOUNT();
pollwrapper *pw = pollwrap_new();
while (!finished) { while (!finished) {
fd_set rset, wset, xset;
int maxfd;
int rwx; int rwx;
int ret; int ret;
unsigned long next; unsigned long next;
FD_ZERO(&rset); pollwrap_clear(pw);
FD_ZERO(&wset);
FD_ZERO(&xset);
maxfd = 0;
/* Count the currently active fds. */ /* Count the currently active fds. */
i = 0; i = 0;
@ -586,24 +579,15 @@ int main(int argc, char **argv)
for (fd = first_fd(&fdstate, &rwx); fd >= 0; for (fd = first_fd(&fdstate, &rwx); fd >= 0;
fd = next_fd(&fdstate, &rwx)) { fd = next_fd(&fdstate, &rwx)) {
fdlist[fdcount++] = fd; fdlist[fdcount++] = fd;
if (rwx & 1) pollwrap_add_fd_rwx(pw, fd, rwx);
FD_SET_MAX(fd, maxfd, rset);
if (rwx & 2)
FD_SET_MAX(fd, maxfd, wset);
if (rwx & 4)
FD_SET_MAX(fd, maxfd, xset);
} }
if (toplevel_callback_pending()) { if (toplevel_callback_pending()) {
struct timeval tv; ret = pollwrap_poll_instant(pw);
tv.tv_sec = 0;
tv.tv_usec = 0;
ret = select(maxfd, &rset, &wset, &xset, &tv);
} else if (run_timers(now, &next)) { } else if (run_timers(now, &next)) {
do { do {
unsigned long then; unsigned long then;
long ticks; long ticks;
struct timeval tv;
then = now; then = now;
now = GETTICKCOUNT(); now = GETTICKCOUNT();
@ -611,38 +595,44 @@ int main(int argc, char **argv)
ticks = 0; ticks = 0;
else else
ticks = next - now; ticks = next - now;
tv.tv_sec = ticks / 1000;
tv.tv_usec = ticks % 1000 * 1000; bool overflow = false;
ret = select(maxfd, &rset, &wset, &xset, &tv); if (ticks > INT_MAX) {
if (ret == 0) ticks = INT_MAX;
overflow = true;
}
ret = pollwrap_poll_timeout(pw, ticks);
if (ret == 0 && !overflow)
now = next; now = next;
else else
now = GETTICKCOUNT(); now = GETTICKCOUNT();
} while (ret < 0 && errno == EINTR); } while (ret < 0 && errno == EINTR);
} else { } else {
ret = select(maxfd, &rset, &wset, &xset, NULL); ret = pollwrap_poll_endless(pw);
} }
if (ret < 0 && errno == EINTR) if (ret < 0 && errno == EINTR)
continue; continue;
if (ret < 0) { if (ret < 0) {
perror("select"); perror("poll");
exit(1); exit(1);
} }
for (i = 0; i < fdcount; i++) { for (i = 0; i < fdcount; i++) {
fd = fdlist[i]; fd = fdlist[i];
int rwx = pollwrap_get_fd_rwx(pw, fd);
/* /*
* We must process exceptional notifications before * We must process exceptional notifications before
* ordinary readability ones, or we may go straight * ordinary readability ones, or we may go straight
* past the urgent marker. * past the urgent marker.
*/ */
if (FD_ISSET(fd, &xset)) if (rwx & SELECT_X)
select_result(fd, SELECT_X); select_result(fd, SELECT_X);
if (FD_ISSET(fd, &rset)) if (rwx & SELECT_R)
select_result(fd, SELECT_R); select_result(fd, SELECT_R);
if (FD_ISSET(fd, &wset)) if (rwx & SELECT_W)
select_result(fd, SELECT_W); select_result(fd, SELECT_W);
} }

View File

@ -13,9 +13,6 @@
#include <errno.h> #include <errno.h>
#include <assert.h> #include <assert.h>
#include <glob.h> #include <glob.h>
#ifndef HAVE_NO_SYS_SELECT_H
#include <sys/select.h>
#endif
#include "putty.h" #include "putty.h"
#include "ssh.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) static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok)
{ {
fd_set rset, wset, xset;
int i, *fdlist; int i, *fdlist;
size_t fdsize; size_t fdsize;
int fd, fdcount, fdstate, rwx, ret, maxfd; int fd, fdcount, fdstate, rwx, ret;
unsigned long now = GETTICKCOUNT(); unsigned long now = GETTICKCOUNT();
unsigned long next; unsigned long next;
bool done_something = false; bool done_something = false;
@ -459,6 +455,8 @@ static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok)
fdlist = NULL; fdlist = NULL;
fdsize = 0; fdsize = 0;
pollwrapper *pw = pollwrap_new();
do { do {
/* Count the currently active fds. */ /* 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. */ /* Expand the fdlist buffer if necessary. */
sgrowarray(fdlist, fdsize, i); sgrowarray(fdlist, fdsize, i);
FD_ZERO(&rset); pollwrap_clear(pw);
FD_ZERO(&wset);
FD_ZERO(&xset);
maxfd = 0;
/* /*
* Add all currently open fds to the select sets, and store * 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; for (fd = first_fd(&fdstate, &rwx); fd >= 0;
fd = next_fd(&fdstate, &rwx)) { fd = next_fd(&fdstate, &rwx)) {
fdlist[fdcount++] = fd; fdlist[fdcount++] = fd;
if (rwx & 1) pollwrap_add_fd_rwx(pw, fd, rwx);
FD_SET_MAX(fd, maxfd, rset);
if (rwx & 2)
FD_SET_MAX(fd, maxfd, wset);
if (rwx & 4)
FD_SET_MAX(fd, maxfd, xset);
} }
if (include_stdin) if (include_stdin)
FD_SET_MAX(0, maxfd, rset); pollwrap_add_fd_rwx(pw, 0, SELECT_R);
if (toplevel_callback_pending()) { if (toplevel_callback_pending()) {
struct timeval tv; ret = pollwrap_poll_instant(pw);
tv.tv_sec = 0;
tv.tv_usec = 0;
ret = select(maxfd, &rset, &wset, &xset, &tv);
if (ret == 0) if (ret == 0)
done_something |= run_toplevel_callbacks(); done_something |= run_toplevel_callbacks();
} else if (run_timers(now, &next)) { } else if (run_timers(now, &next)) {
do { do {
unsigned long then; unsigned long then;
long ticks; long ticks;
struct timeval tv;
then = now; then = now;
now = GETTICKCOUNT(); now = GETTICKCOUNT();
@ -515,38 +501,44 @@ static int ssh_sftp_do_select(bool include_stdin, bool no_fds_ok)
ticks = 0; ticks = 0;
else else
ticks = next - now; ticks = next - now;
tv.tv_sec = ticks / 1000;
tv.tv_usec = ticks % 1000 * 1000; bool overflow = false;
ret = select(maxfd, &rset, &wset, &xset, &tv); if (ticks > INT_MAX) {
if (ret == 0) ticks = INT_MAX;
overflow = true;
}
ret = pollwrap_poll_timeout(pw, ticks);
if (ret == 0 && !overflow)
now = next; now = next;
else else
now = GETTICKCOUNT(); now = GETTICKCOUNT();
} while (ret < 0 && errno == EINTR); } while (ret < 0 && errno == EINTR);
} else { } else {
do { do {
ret = select(maxfd, &rset, &wset, &xset, NULL); ret = pollwrap_poll_endless(pw);
} while (ret < 0 && errno == EINTR); } while (ret < 0 && errno == EINTR);
} }
} while (ret == 0 && !done_something); } while (ret == 0 && !done_something);
if (ret < 0) { if (ret < 0) {
perror("select"); perror("poll");
exit(1); exit(1);
} }
for (i = 0; i < fdcount; i++) { for (i = 0; i < fdcount; i++) {
fd = fdlist[i]; fd = fdlist[i];
int rwx = pollwrap_get_fd_rwx(pw, fd);
/* /*
* We must process exceptional notifications before * We must process exceptional notifications before
* ordinary readability ones, or we may go straight * ordinary readability ones, or we may go straight
* past the urgent marker. * past the urgent marker.
*/ */
if (FD_ISSET(fd, &xset)) if (rwx & SELECT_X)
select_result(fd, SELECT_X); select_result(fd, SELECT_X);
if (FD_ISSET(fd, &rset)) if (rwx & SELECT_R)
select_result(fd, SELECT_R); select_result(fd, SELECT_R);
if (FD_ISSET(fd, &wset)) if (rwx & SELECT_W)
select_result(fd, 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(); 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;
} }
/* /*