1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 01:48:00 +00:00

Two related changes to timing code:

First, make absolute times unsigned.  This means that it's safe to 
depend on their overflow behaviour (which is undefined for signed 
integers).  This requires a little extra care in handling comparisons, 
but I think I've correctly adjusted them all.

Second, functions registered with schedule_timer() are guaranteed to be 
called with precisely the time that was returned by schedule_timer().  
Thus, it's only necessary to check these values for equality rather than 
doing risky range checks, so do that.

The timing code still does lots that's undefined, unnecessary, or just
wrong, but this is a good start.

[originally from svn r9667]
This commit is contained in:
Ben Harris 2012-09-18 21:42:48 +00:00
parent 41ad182710
commit d5836982e2
16 changed files with 88 additions and 61 deletions

View File

@ -11,7 +11,7 @@
#include "putty.h" #include "putty.h"
long schedule_timer(int ticks, timer_fn_t fn, void *ctx) unsigned long schedule_timer(int ticks, timer_fn_t fn, void *ctx)
{ {
return 0; return 0;
} }

View File

@ -8,18 +8,18 @@
struct pinger_tag { struct pinger_tag {
int interval; int interval;
int pending; int pending;
long next; unsigned long next;
Backend *back; Backend *back;
void *backhandle; void *backhandle;
}; };
static void pinger_schedule(Pinger pinger); static void pinger_schedule(Pinger pinger);
static void pinger_timer(void *ctx, long now) static void pinger_timer(void *ctx, unsigned long now)
{ {
Pinger pinger = (Pinger)ctx; Pinger pinger = (Pinger)ctx;
if (pinger->pending && now - pinger->next >= 0) { if (pinger->pending && now == pinger->next) {
pinger->back->special(pinger->backhandle, TS_PING); pinger->back->special(pinger->backhandle, TS_PING);
pinger->pending = FALSE; pinger->pending = FALSE;
pinger_schedule(pinger); pinger_schedule(pinger);

View File

@ -1383,11 +1383,11 @@ char *get_random_data(int bytes); /* used in cmdgen.c */
* GETTICKCOUNT() and compare the result with the returned `next' * GETTICKCOUNT() and compare the result with the returned `next'
* value to find out how long you have to make your next wait().) * value to find out how long you have to make your next wait().)
*/ */
typedef void (*timer_fn_t)(void *ctx, long now); typedef void (*timer_fn_t)(void *ctx, unsigned long now);
long schedule_timer(int ticks, timer_fn_t fn, void *ctx); unsigned long schedule_timer(int ticks, timer_fn_t fn, void *ctx);
void expire_timer_context(void *ctx); void expire_timer_context(void *ctx);
int run_timers(long now, long *next); int run_timers(unsigned long now, unsigned long *next);
void timer_change_notify(long next); void timer_change_notify(unsigned long next);
/* /*
* Define no-op macros for the jump list functions, on platforms that * Define no-op macros for the jump list functions, on platforms that

14
ssh.c
View File

@ -772,7 +772,7 @@ static int ssh_do_close(Ssh ssh, int notify_exit);
static unsigned long ssh_pkt_getuint32(struct Packet *pkt); static unsigned long ssh_pkt_getuint32(struct Packet *pkt);
static int ssh2_pkt_getbool(struct Packet *pkt); static int ssh2_pkt_getbool(struct Packet *pkt);
static void ssh_pkt_getstring(struct Packet *pkt, char **p, int *length); static void ssh_pkt_getstring(struct Packet *pkt, char **p, int *length);
static void ssh2_timer(void *ctx, long now); static void ssh2_timer(void *ctx, unsigned long now);
static void do_ssh2_transport(Ssh ssh, void *vin, int inlen, static void do_ssh2_transport(Ssh ssh, void *vin, int inlen,
struct Packet *pktin); struct Packet *pktin);
static void ssh2_msg_unexpected(Ssh ssh, struct Packet *pktin); static void ssh2_msg_unexpected(Ssh ssh, struct Packet *pktin);
@ -981,7 +981,7 @@ struct ssh_tag {
unsigned long incoming_data_size, outgoing_data_size, deferred_data_size; unsigned long incoming_data_size, outgoing_data_size, deferred_data_size;
unsigned long max_data_size; unsigned long max_data_size;
int kex_in_progress; int kex_in_progress;
long next_rekey, last_rekey; unsigned long next_rekey, last_rekey;
char *deferred_rekey_reason; /* points to STATIC string; don't free */ char *deferred_rekey_reason; /* points to STATIC string; don't free */
/* /*
@ -9482,7 +9482,7 @@ static void ssh2_protocol_setup(Ssh ssh)
ssh->packet_dispatch[SSH2_MSG_DEBUG] = ssh2_msg_debug; ssh->packet_dispatch[SSH2_MSG_DEBUG] = ssh2_msg_debug;
} }
static void ssh2_timer(void *ctx, long now) static void ssh2_timer(void *ctx, unsigned long now)
{ {
Ssh ssh = (Ssh)ctx; Ssh ssh = (Ssh)ctx;
@ -9490,7 +9490,7 @@ static void ssh2_timer(void *ctx, long now)
return; return;
if (!ssh->kex_in_progress && conf_get_int(ssh->conf, CONF_ssh_rekey_time) != 0 && if (!ssh->kex_in_progress && conf_get_int(ssh->conf, CONF_ssh_rekey_time) != 0 &&
now - ssh->next_rekey >= 0) { now == ssh->next_rekey) {
do_ssh2_transport(ssh, "timeout", -1, NULL); do_ssh2_transport(ssh, "timeout", -1, NULL);
} }
} }
@ -9773,10 +9773,10 @@ static void ssh_reconfig(void *handle, Conf *conf)
rekey_time = conf_get_int(conf, CONF_ssh_rekey_time); rekey_time = conf_get_int(conf, CONF_ssh_rekey_time);
if (conf_get_int(ssh->conf, CONF_ssh_rekey_time) != rekey_time && if (conf_get_int(ssh->conf, CONF_ssh_rekey_time) != rekey_time &&
rekey_time != 0) { rekey_time != 0) {
long new_next = ssh->last_rekey + rekey_time*60*TICKSPERSEC; unsigned long new_next = ssh->last_rekey + rekey_time*60*TICKSPERSEC;
long now = GETTICKCOUNT(); unsigned long now = GETTICKCOUNT();
if (new_next - now < 0) { if (now - ssh->last_rekey > rekey_time*60*TICKSPERSEC) {
rekeying = "timeout shortened"; rekeying = "timeout shortened";
} else { } else {
ssh->next_rekey = schedule_timer(new_next - now, ssh2_timer, ssh); ssh->next_rekey = schedule_timer(new_next - now, ssh2_timer, ssh);

View File

@ -199,9 +199,9 @@ static void random_add_heavynoise_bitbybit(void *noise, int length)
pool.poolpos = i; pool.poolpos = i;
} }
static void random_timer(void *ctx, long now) static void random_timer(void *ctx, unsigned long now)
{ {
if (random_active > 0 && now - next_noise_collection >= 0) { if (random_active > 0 && now == next_noise_collection) {
noise_regular(); noise_regular();
next_noise_collection = next_noise_collection =
schedule_timer(NOISE_REGULAR_INTERVAL, random_timer, &pool); schedule_timer(NOISE_REGULAR_INTERVAL, random_timer, &pool);

View File

@ -1065,32 +1065,32 @@ static termline *lineptr(Terminal *term, int y, int lineno, int screen)
static void term_schedule_tblink(Terminal *term); static void term_schedule_tblink(Terminal *term);
static void term_schedule_cblink(Terminal *term); static void term_schedule_cblink(Terminal *term);
static void term_timer(void *ctx, long now) static void term_timer(void *ctx, unsigned long now)
{ {
Terminal *term = (Terminal *)ctx; Terminal *term = (Terminal *)ctx;
int update = FALSE; int update = FALSE;
if (term->tblink_pending && now - term->next_tblink >= 0) { if (term->tblink_pending && now == term->next_tblink) {
term->tblinker = !term->tblinker; term->tblinker = !term->tblinker;
term->tblink_pending = FALSE; term->tblink_pending = FALSE;
term_schedule_tblink(term); term_schedule_tblink(term);
update = TRUE; update = TRUE;
} }
if (term->cblink_pending && now - term->next_cblink >= 0) { if (term->cblink_pending && now == term->next_cblink) {
term->cblinker = !term->cblinker; term->cblinker = !term->cblinker;
term->cblink_pending = FALSE; term->cblink_pending = FALSE;
term_schedule_cblink(term); term_schedule_cblink(term);
update = TRUE; update = TRUE;
} }
if (term->in_vbell && now - term->vbell_end >= 0) { if (term->in_vbell && now == term->vbell_end) {
term->in_vbell = FALSE; term->in_vbell = FALSE;
update = TRUE; update = TRUE;
} }
if (update || if (update ||
(term->window_update_pending && now - term->next_update >= 0)) (term->window_update_pending && now == term->next_update))
term_update(term); term_update(term);
} }

View File

@ -36,13 +36,13 @@
struct timer { struct timer {
timer_fn_t fn; timer_fn_t fn;
void *ctx; void *ctx;
long now; unsigned long now;
long when_set; unsigned long when_set;
}; };
static tree234 *timers = NULL; static tree234 *timers = NULL;
static tree234 *timer_contexts = NULL; static tree234 *timer_contexts = NULL;
static long now = 0L; static unsigned long now = 0L;
static int compare_timers(void *av, void *bv) static int compare_timers(void *av, void *bv)
{ {
@ -106,9 +106,9 @@ static void init_timers(void)
} }
} }
long schedule_timer(int ticks, timer_fn_t fn, void *ctx) unsigned long schedule_timer(int ticks, timer_fn_t fn, void *ctx)
{ {
long when; unsigned long when;
struct timer *t, *first; struct timer *t, *first;
init_timers(); init_timers();
@ -153,7 +153,7 @@ long schedule_timer(int ticks, timer_fn_t fn, void *ctx)
* Returns the time (in ticks) expected until the next timer after * Returns the time (in ticks) expected until the next timer after
* that triggers. * that triggers.
*/ */
int run_timers(long anow, long *next) int run_timers(unsigned long anow, unsigned long *next)
{ {
struct timer *first; struct timer *first;
@ -174,8 +174,8 @@ int run_timers(long anow, long *next)
*/ */
delpos234(timers, 0); delpos234(timers, 0);
sfree(first); sfree(first);
} else if (first->now - now <= 0 || } else if (now - first->when_set - 10 >
now - (first->when_set - 10) < 0) { first->now - first->when_set - 10) {
/* /*
* This timer is active and has reached its running * This timer is active and has reached its running
* time. Run it. * time. Run it.

View File

@ -1400,13 +1400,18 @@ void notify_remote_exit(void *frontend)
static gint timer_trigger(gpointer data) static gint timer_trigger(gpointer data)
{ {
long now = GPOINTER_TO_LONG(data); unsigned long now = GPOINTER_TO_LONG(data);
long next; unsigned long next, then;
long ticks; long ticks;
if (run_timers(now, &next)) { if (run_timers(now, &next)) {
ticks = next - GETTICKCOUNT(); then = now;
timer_id = gtk_timeout_add(ticks > 0 ? ticks : 1, timer_trigger, now = GETTICKCOUNT();
if (now - then > next - then)
ticks = 0;
else
ticks = next - now;
timer_id = gtk_timeout_add(ticks, timer_trigger,
LONG_TO_GPOINTER(next)); LONG_TO_GPOINTER(next));
} }
@ -1417,7 +1422,7 @@ static gint timer_trigger(gpointer data)
return FALSE; return FALSE;
} }
void timer_change_notify(long next) void timer_change_notify(unsigned long next)
{ {
long ticks; long ticks;

View File

@ -70,7 +70,7 @@ void notify_remote_exit(void *frontend)
{ {
} }
void timer_change_notify(long next) void timer_change_notify(unsigned long next)
{ {
} }

View File

@ -593,7 +593,7 @@ int main(int argc, char **argv)
int errors; int errors;
int use_subsystem = 0; int use_subsystem = 0;
int got_host = FALSE; int got_host = FALSE;
long now; unsigned long now;
struct winsize size; struct winsize size;
fdlist = NULL; fdlist = NULL;
@ -1016,12 +1016,17 @@ int main(int argc, char **argv)
} }
do { do {
long next, ticks; unsigned long next, then;
long ticks;
struct timeval tv, *ptv; struct timeval tv, *ptv;
if (run_timers(now, &next)) { if (run_timers(now, &next)) {
ticks = next - GETTICKCOUNT(); then = now;
if (ticks < 0) ticks = 0; /* just in case */ now = GETTICKCOUNT();
if (now - then > next - then)
ticks = 0;
else
ticks = next - now;
tv.tv_sec = ticks / 1000; tv.tv_sec = ticks / 1000;
tv.tv_usec = ticks % 1000 * 1000; tv.tv_usec = ticks % 1000 * 1000;
ptv = &tv; ptv = &tv;

View File

@ -443,7 +443,7 @@ static int ssh_sftp_do_select(int include_stdin, int no_fds_ok)
fd_set rset, wset, xset; fd_set rset, wset, xset;
int i, fdcount, fdsize, *fdlist; int i, fdcount, fdsize, *fdlist;
int fd, fdstate, rwx, ret, maxfd; int fd, fdstate, rwx, ret, maxfd;
long now = GETTICKCOUNT(); unsigned long now = GETTICKCOUNT();
fdlist = NULL; fdlist = NULL;
fdcount = fdsize = 0; fdcount = fdsize = 0;
@ -489,13 +489,17 @@ static int ssh_sftp_do_select(int include_stdin, int no_fds_ok)
FD_SET_MAX(0, maxfd, rset); FD_SET_MAX(0, maxfd, rset);
do { do {
long next, ticks; unsigned long next, then;
long ticks;
struct timeval tv, *ptv; struct timeval tv, *ptv;
if (run_timers(now, &next)) { if (run_timers(now, &next)) {
ticks = next - GETTICKCOUNT(); then = now;
if (ticks <= 0) now = GETTICKCOUNT();
ticks = 1; /* just in case */ if (now - then > next - then)
ticks = 0;
else
ticks = next - now;
tv.tv_sec = ticks / 1000; tv.tv_sec = ticks / 1000;
tv.tv_usec = ticks % 1000 * 1000; tv.tv_usec = ticks % 1000 * 1000;
ptv = &tv; ptv = &tv;

View File

@ -41,7 +41,7 @@ void notify_remote_exit(void *frontend)
{ {
} }
void timer_change_notify(long next) void timer_change_notify(unsigned long next)
{ {
} }

View File

@ -2012,10 +2012,14 @@ void notify_remote_exit(void *fe)
} }
} }
void timer_change_notify(long next) void timer_change_notify(unsigned long next)
{ {
long ticks = next - GETTICKCOUNT(); unsigned long now = GETTICKCOUNT();
if (ticks <= 0) ticks = 1; /* just in case */ long ticks;
if (now - next < INT_MAX)
ticks = 0;
else
ticks = next - now;
KillTimer(hwnd, TIMING_TIMER_ID); KillTimer(hwnd, TIMING_TIMER_ID);
SetTimer(hwnd, TIMING_TIMER_ID, ticks, NULL); SetTimer(hwnd, TIMING_TIMER_ID, ticks, NULL);
timing_next_time = next; timing_next_time = next;
@ -2042,7 +2046,7 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
switch (message) { switch (message) {
case WM_TIMER: case WM_TIMER:
if ((UINT_PTR)wParam == TIMING_TIMER_ID) { if ((UINT_PTR)wParam == TIMING_TIMER_ID) {
long next; unsigned long next;
KillTimer(hwnd, TIMING_TIMER_ID); KillTimer(hwnd, TIMING_TIMER_ID);
if (run_timers(timing_next_time, &next)) { if (run_timers(timing_next_time, &next)) {
@ -5354,9 +5358,9 @@ static int flashing = 0;
* Timer for platforms where we must maintain window flashing manually * Timer for platforms where we must maintain window flashing manually
* (e.g., Win95). * (e.g., Win95).
*/ */
static void flash_window_timer(void *ctx, long now) static void flash_window_timer(void *ctx, unsigned long now)
{ {
if (flashing && now - next_flash >= 0) { if (flashing && now == next_flash) {
flash_window(1); flash_window(1);
} }
} }

View File

@ -289,7 +289,7 @@ int main(int argc, char **argv)
int errors; int errors;
int got_host = FALSE; int got_host = FALSE;
int use_subsystem = 0; int use_subsystem = 0;
long now, next; unsigned long now, next, then;
sklist = NULL; sklist = NULL;
skcount = sksize = 0; skcount = sksize = 0;
@ -634,8 +634,12 @@ int main(int argc, char **argv)
} }
if (run_timers(now, &next)) { if (run_timers(now, &next)) {
ticks = next - GETTICKCOUNT(); then = now;
if (ticks < 0) ticks = 0; /* just in case */ now = GETTICKCOUNT();
if (now - then > next - then)
ticks = 0;
else
ticks = next - now;
} else { } else {
ticks = INFINITE; ticks = INFINITE;
} }

View File

@ -331,11 +331,11 @@ static void serial_size(void *handle, int width, int height)
return; return;
} }
static void serbreak_timer(void *ctx, long now) static void serbreak_timer(void *ctx, unsigned long now)
{ {
Serial serial = (Serial)ctx; Serial serial = (Serial)ctx;
if (now >= serial->clearbreak_time && serial->port) { if (now == serial->clearbreak_time && serial->port) {
ClearCommBreak(serial->port); ClearCommBreak(serial->port);
serial->break_in_progress = FALSE; serial->break_in_progress = FALSE;
logevent(serial->frontend, "Finished serial break"); logevent(serial->frontend, "Finished serial break");

View File

@ -486,15 +486,20 @@ extern int select_result(WPARAM, LPARAM);
int do_eventsel_loop(HANDLE other_event) int do_eventsel_loop(HANDLE other_event)
{ {
int n, nhandles, nallhandles, netindex, otherindex; int n, nhandles, nallhandles, netindex, otherindex;
long next, ticks; unsigned long next, then;
long ticks;
HANDLE *handles; HANDLE *handles;
SOCKET *sklist; SOCKET *sklist;
int skcount; int skcount;
long now = GETTICKCOUNT(); unsigned long now = GETTICKCOUNT();
if (run_timers(now, &next)) { if (run_timers(now, &next)) {
ticks = next - GETTICKCOUNT(); then = now;
if (ticks < 0) ticks = 0; /* just in case */ now = GETTICKCOUNT();
if (now - then > next - then)
ticks = 0;
else
ticks = next - now;
} else { } else {
ticks = INFINITE; ticks = INFINITE;
} }