/* * pinger.c: centralised module that deals with sending TS_PING * keepalives, to avoid replicating this code in multiple backends. */ #include "putty.h" struct pinger_tag { int interval; int pending; unsigned long when_set, next; Backend *back; void *backhandle; }; static void pinger_schedule(Pinger pinger); static void pinger_timer(void *ctx, unsigned long now) { Pinger pinger = (Pinger)ctx; if (pinger->pending && now == pinger->next) { pinger->back->special(pinger->backhandle, TS_PING); pinger->pending = FALSE; pinger_schedule(pinger); } } static void pinger_schedule(Pinger pinger) { unsigned long next; if (!pinger->interval) { pinger->pending = FALSE; /* cancel any pending ping */ return; } next = schedule_timer(pinger->interval * TICKSPERSEC, pinger_timer, pinger); if (!pinger->pending || (next - pinger->when_set) < (pinger->next - pinger->when_set)) { pinger->next = next; pinger->when_set = timing_last_clock(); pinger->pending = TRUE; } } Pinger pinger_new(Conf *conf, Backend *back, void *backhandle) { Pinger pinger = snew(struct pinger_tag); pinger->interval = conf_get_int(conf, CONF_ping_interval); pinger->pending = FALSE; pinger->back = back; pinger->backhandle = backhandle; pinger_schedule(pinger); return pinger; } void pinger_reconfig(Pinger pinger, Conf *oldconf, Conf *newconf) { int newinterval = conf_get_int(newconf, CONF_ping_interval); if (conf_get_int(oldconf, CONF_ping_interval) != newinterval) { pinger->interval = newinterval; pinger_schedule(pinger); } } void pinger_free(Pinger pinger) { expire_timer_context(pinger); sfree(pinger); }