2020-02-07 19:15:13 +00:00
|
|
|
#include "putty.h"
|
|
|
|
|
|
|
|
void cli_main_loop(cliloop_pre_t pre, cliloop_post_t post, void *ctx)
|
|
|
|
{
|
|
|
|
SOCKET *sklist = NULL;
|
|
|
|
size_t skcount = 0, sksize = 0;
|
|
|
|
unsigned long now, next, then;
|
|
|
|
now = GETTICKCOUNT();
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
int nhandles;
|
|
|
|
HANDLE *handles;
|
2021-01-19 20:26:04 +00:00
|
|
|
DWORD n;
|
2020-02-07 19:15:13 +00:00
|
|
|
DWORD ticks;
|
|
|
|
|
|
|
|
const HANDLE *extra_handles = NULL;
|
|
|
|
size_t n_extra_handles = 0;
|
|
|
|
if (!pre(ctx, &extra_handles, &n_extra_handles))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (toplevel_callback_pending()) {
|
|
|
|
ticks = 0;
|
|
|
|
next = now;
|
|
|
|
} else if (run_timers(now, &next)) {
|
|
|
|
then = now;
|
|
|
|
now = GETTICKCOUNT();
|
|
|
|
if (now - then > next - then)
|
|
|
|
ticks = 0;
|
|
|
|
else
|
|
|
|
ticks = next - now;
|
|
|
|
} else {
|
|
|
|
ticks = INFINITE;
|
|
|
|
/* no need to initialise next here because we can never
|
|
|
|
* get WAIT_TIMEOUT */
|
|
|
|
}
|
|
|
|
|
|
|
|
handles = handle_get_events(&nhandles);
|
2021-01-19 20:26:04 +00:00
|
|
|
size_t winselcli_index = -(size_t)1;
|
|
|
|
size_t extra_base = nhandles;
|
|
|
|
if (winselcli_event != INVALID_HANDLE_VALUE) {
|
|
|
|
winselcli_index = extra_base++;
|
2021-01-26 12:41:26 +00:00
|
|
|
handles = sresize(handles, extra_base, HANDLE);
|
2021-01-19 20:26:04 +00:00
|
|
|
handles[winselcli_index] = winselcli_event;
|
|
|
|
}
|
2020-02-07 19:15:13 +00:00
|
|
|
size_t total_handles = extra_base + n_extra_handles;
|
|
|
|
handles = sresize(handles, total_handles, HANDLE);
|
|
|
|
for (size_t i = 0; i < n_extra_handles; i++)
|
|
|
|
handles[extra_base + i] = extra_handles[i];
|
|
|
|
|
|
|
|
n = WaitForMultipleObjects(total_handles, handles, false, ticks);
|
|
|
|
|
|
|
|
size_t extra_handle_index = n_extra_handles;
|
|
|
|
|
|
|
|
if ((unsigned)(n - WAIT_OBJECT_0) < (unsigned)nhandles) {
|
|
|
|
handle_got_event(handles[n - WAIT_OBJECT_0]);
|
2021-01-19 20:26:04 +00:00
|
|
|
} else if (winselcli_event != INVALID_HANDLE_VALUE &&
|
|
|
|
n == WAIT_OBJECT_0 + winselcli_index) {
|
2020-02-07 19:15:13 +00:00
|
|
|
WSANETWORKEVENTS things;
|
|
|
|
SOCKET socket;
|
|
|
|
int i, socketstate;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We must not call select_result() for any socket
|
|
|
|
* until we have finished enumerating within the tree.
|
|
|
|
* This is because select_result() may close the socket
|
|
|
|
* and modify the tree.
|
|
|
|
*/
|
|
|
|
/* Count the active sockets. */
|
|
|
|
i = 0;
|
|
|
|
for (socket = first_socket(&socketstate);
|
|
|
|
socket != INVALID_SOCKET;
|
|
|
|
socket = next_socket(&socketstate)) i++;
|
|
|
|
|
|
|
|
/* Expand the buffer if necessary. */
|
|
|
|
sgrowarray(sklist, sksize, i);
|
|
|
|
|
|
|
|
/* Retrieve the sockets into sklist. */
|
|
|
|
skcount = 0;
|
|
|
|
for (socket = first_socket(&socketstate);
|
|
|
|
socket != INVALID_SOCKET;
|
|
|
|
socket = next_socket(&socketstate)) {
|
|
|
|
sklist[skcount++] = socket;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now we're done enumerating; go through the list. */
|
|
|
|
for (i = 0; i < skcount; i++) {
|
|
|
|
WPARAM wp;
|
|
|
|
socket = sklist[i];
|
|
|
|
wp = (WPARAM) socket;
|
|
|
|
if (!p_WSAEnumNetworkEvents(socket, NULL, &things)) {
|
|
|
|
static const struct { int bit, mask; } eventtypes[] = {
|
|
|
|
{FD_CONNECT_BIT, FD_CONNECT},
|
|
|
|
{FD_READ_BIT, FD_READ},
|
|
|
|
{FD_CLOSE_BIT, FD_CLOSE},
|
|
|
|
{FD_OOB_BIT, FD_OOB},
|
|
|
|
{FD_WRITE_BIT, FD_WRITE},
|
|
|
|
{FD_ACCEPT_BIT, FD_ACCEPT},
|
|
|
|
};
|
|
|
|
int e;
|
|
|
|
|
|
|
|
noise_ultralight(NOISE_SOURCE_IOID, socket);
|
|
|
|
|
|
|
|
for (e = 0; e < lenof(eventtypes); e++)
|
|
|
|
if (things.lNetworkEvents & eventtypes[e].mask) {
|
|
|
|
LPARAM lp;
|
|
|
|
int err = things.iErrorCode[eventtypes[e].bit];
|
|
|
|
lp = WSAMAKESELECTREPLY(eventtypes[e].mask, err);
|
|
|
|
select_result(wp, lp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (n >= WAIT_OBJECT_0 + extra_base &&
|
|
|
|
n < WAIT_OBJECT_0 + extra_base + n_extra_handles) {
|
|
|
|
extra_handle_index = n - (WAIT_OBJECT_0 + extra_base);
|
|
|
|
}
|
|
|
|
|
|
|
|
run_toplevel_callbacks();
|
|
|
|
|
|
|
|
if (n == WAIT_TIMEOUT) {
|
|
|
|
now = next;
|
|
|
|
} else {
|
|
|
|
now = GETTICKCOUNT();
|
|
|
|
}
|
|
|
|
|
|
|
|
sfree(handles);
|
|
|
|
|
|
|
|
if (!post(ctx, extra_handle_index))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cliloop_null_pre(void *vctx, const HANDLE **eh, size_t *neh)
|
|
|
|
{ return true; }
|
|
|
|
bool cliloop_null_post(void *vctx, size_t ehi) { return true; }
|