mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 01:02:24 +00:00
Fix a potential time-wraparound issue in pinger.c.
A compiler warning drew my attention to the fact that 'next' in pinger_schedule() was an int, not the unsigned long it should have been. And looking at the code that handles it, it was also taking no care with integer wraparound when checking whether an existing scheduled ping should be moved forward. So now I do something a bit more robust, by remembering what time it _was_ when we set pinger->next, and checking if the new time value falls in the interval between those two times.
This commit is contained in:
parent
46051027fb
commit
b4202c917a
8
pinger.c
8
pinger.c
@ -8,7 +8,7 @@
|
|||||||
struct pinger_tag {
|
struct pinger_tag {
|
||||||
int interval;
|
int interval;
|
||||||
int pending;
|
int pending;
|
||||||
unsigned long next;
|
unsigned long when_set, next;
|
||||||
Backend *back;
|
Backend *back;
|
||||||
void *backhandle;
|
void *backhandle;
|
||||||
};
|
};
|
||||||
@ -28,7 +28,7 @@ static void pinger_timer(void *ctx, unsigned long now)
|
|||||||
|
|
||||||
static void pinger_schedule(Pinger pinger)
|
static void pinger_schedule(Pinger pinger)
|
||||||
{
|
{
|
||||||
int next;
|
unsigned long next;
|
||||||
|
|
||||||
if (!pinger->interval) {
|
if (!pinger->interval) {
|
||||||
pinger->pending = FALSE; /* cancel any pending ping */
|
pinger->pending = FALSE; /* cancel any pending ping */
|
||||||
@ -37,8 +37,10 @@ static void pinger_schedule(Pinger pinger)
|
|||||||
|
|
||||||
next = schedule_timer(pinger->interval * TICKSPERSEC,
|
next = schedule_timer(pinger->interval * TICKSPERSEC,
|
||||||
pinger_timer, pinger);
|
pinger_timer, pinger);
|
||||||
if (!pinger->pending || next < pinger->next) {
|
if (!pinger->pending ||
|
||||||
|
(next - pinger->when_set) < (pinger->next - pinger->when_set)) {
|
||||||
pinger->next = next;
|
pinger->next = next;
|
||||||
|
pinger->when_set = timing_last_clock();
|
||||||
pinger->pending = TRUE;
|
pinger->pending = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
1
putty.h
1
putty.h
@ -1450,6 +1450,7 @@ 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(unsigned long now, unsigned long *next);
|
int run_timers(unsigned long now, unsigned long *next);
|
||||||
void timer_change_notify(unsigned long next);
|
void timer_change_notify(unsigned long next);
|
||||||
|
unsigned long timing_last_clock(void);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Exports from callback.c.
|
* Exports from callback.c.
|
||||||
|
11
timing.c
11
timing.c
@ -148,6 +148,17 @@ unsigned long schedule_timer(int ticks, timer_fn_t fn, void *ctx)
|
|||||||
return when;
|
return when;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unsigned long timing_last_clock(void)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Return the last value we stored in 'now'. In particular,
|
||||||
|
* calling this just after schedule_timer returns the value of
|
||||||
|
* 'now' that was used to decide when the timer you just set would
|
||||||
|
* go off.
|
||||||
|
*/
|
||||||
|
return now;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Call to run any timers whose time has reached the present.
|
* Call to run any timers whose time has reached the present.
|
||||||
* Returns the time (in ticks) expected until the next timer after
|
* Returns the time (in ticks) expected until the next timer after
|
||||||
|
Loading…
Reference in New Issue
Block a user