From bd5c957e5bd0f850ab30c6b1ab94bfbbabe9f006 Mon Sep 17 00:00:00 2001 From: Simon Tatham Date: Sat, 21 Dec 2019 13:31:02 +0000 Subject: [PATCH] winsftp.c: avoid creating multiple netevents. The do_select function is called with a boolean parameter indicating whether we're supposed to start or stop paying attention to network activity on a given socket. So if we freeze and unfreeze the socket in mid-session because of backlog, we'll call do_select(s, false) to freeze it, and do_select(s, true) to unfreeze it. But the implementation of do_select in the Windows SFTP code predated the rigorous handling of socket backlogs, so it assumed that do_select(s, true) would only be called at initialisation time, i.e. only once, and therefore that it was safe to use that flag as a cue to set up the Windows event object to associate with socket activity. Hence, every time the socket was frozen and unfrozen, we would create a new netevent at unfreeze time, leaking the old one. I think perhaps part of the reason why that was hard to figure out was that the boolean parameter was called 'startup' rather than 'enable'. To make it less confusing the next time I read this code, I've also renamed it, and while I was at it, adjusted another related comment. --- windows/window.c | 4 ++-- windows/winplink.c | 4 ++-- windows/winsftp.c | 14 +++++++++----- windows/winstuff.h | 2 +- 4 files changed, 14 insertions(+), 10 deletions(-) diff --git a/windows/window.c b/windows/window.c index f9e09aa3..a17a9854 100644 --- a/windows/window.c +++ b/windows/window.c @@ -1026,10 +1026,10 @@ void cleanup_exit(int code) /* * Set up, or shut down, an AsyncSelect. Called from winnet.c. */ -char *do_select(SOCKET skt, bool startup) +char *do_select(SOCKET skt, bool enable) { int msg, events; - if (startup) { + if (enable) { msg = WM_NETEVENT; events = (FD_CONNECT | FD_READ | FD_WRITE | FD_OOB | FD_CLOSE | FD_ACCEPT); diff --git a/windows/winplink.c b/windows/winplink.c index dce5691f..39ca8fb5 100644 --- a/windows/winplink.c +++ b/windows/winplink.c @@ -190,10 +190,10 @@ static void version(void) exit(0); } -char *do_select(SOCKET skt, bool startup) +char *do_select(SOCKET skt, bool enable) { int events; - if (startup) { + if (enable) { events = (FD_CONNECT | FD_READ | FD_WRITE | FD_OOB | FD_CLOSE | FD_ACCEPT); } else { diff --git a/windows/winsftp.c b/windows/winsftp.c index b120890c..91c3f2fd 100644 --- a/windows/winsftp.c +++ b/windows/winsftp.c @@ -467,19 +467,21 @@ char *dir_file_cat(const char *dir, const char *file) */ static SOCKET sftp_ssh_socket = INVALID_SOCKET; static HANDLE netevent = INVALID_HANDLE_VALUE; -char *do_select(SOCKET skt, bool startup) +char *do_select(SOCKET skt, bool enable) { int events; - if (startup) + if (enable) sftp_ssh_socket = skt; else sftp_ssh_socket = INVALID_SOCKET; + if (netevent == INVALID_HANDLE_VALUE) + netevent = CreateEvent(NULL, false, false, NULL); + if (p_WSAEventSelect) { - if (startup) { + if (enable) { events = (FD_CONNECT | FD_READ | FD_WRITE | FD_OOB | FD_CLOSE | FD_ACCEPT); - netevent = CreateEvent(NULL, false, false, NULL); } else { events = 0; } @@ -732,7 +734,9 @@ char *ssh_sftp_get_cmdline(const char *prompt, bool no_fds_ok) do { ret = do_eventsel_loop(ctx->event); - /* Error return can only occur if netevent==NULL, and it ain't. */ + /* do_eventsel_loop can't return an error (unlike + * ssh_sftp_loop_iteration, which can return -1 if select goes + * wrong or if the socket doesn't exist). */ assert(ret >= 0); } while (ret == 0); diff --git a/windows/winstuff.h b/windows/winstuff.h index 0b191d54..c189c37a 100644 --- a/windows/winstuff.h +++ b/windows/winstuff.h @@ -350,7 +350,7 @@ DECL_WINDOWS_FUNCTION(GLOBAL, int, select, * Provided by each client of winnet.c, and called by winnet.c to turn * on or off WSA*Select for a given socket. */ -char *do_select(SOCKET skt, bool startup); +char *do_select(SOCKET skt, bool enable); /* * Network-subsystem-related functions provided in other Windows modules.