I have this diff running on the failover-side currently. I'v disabled all extra "set timeout"-settings on failover-side which I used to hold down the exp. time.
The replication of exp. time seems to work. I have a long-lived tcp connection which used to show up with double exp. time on the failover-side. The time for this connection is almost identical now on both sides(1 sec. is a diff in between). The rest of connections seems to be OK too - I don't see doubled exp. time values anymore while scrolling in systat. //maxim On Oct 28, 2011, at 3:49 PM, Mike Belopuhov wrote: > On Fri, Oct 28, 2011 at 11:25 AM, Mike Belopuhov <m...@crypt.org.ru> wrote: >> On Thu, Oct 27, 2011 at 11:18 AM, Mike Belopuhov <m...@crypt.org.ru> wrote: >>>>> On 26-10-2011 20:32, Maxim Bourmistrov wrote: >>>>>> The side question, after observing 'systat -s1 states', is WHY "failover"-side >>>>>> doubles exp. time?? >>>>>> I'm more expected to have it like a "copy" of the current state of the >>>>>> master. >>> >>> i've started looking into the problem, but it's not yet apparent as to >>> why it happens. >>> >> >> ok, so what i've figured out is that expire timeouts are transferred >> as values of "rule->timeout[state->timeout]" and not absolute ones >> (to compensate for a local time difference). but when you insert >> the rule into the state table you have to subtract the timeout value >> as pf_state_expires does a good job itself by adding those back. >> >> so here's a diff. i've tried to solve "no adaptive scaling" caveat as well. >> >> ok? >> > > need to assign a rule pointer before dereferencing it, right? > > Index: if_pfsync.c > =================================================================== > RCS file: /cvs/src/sys/net/if_pfsync.c,v > retrieving revision 1.169 > diff -u -p -u -p -r1.169 if_pfsync.c > --- if_pfsync.c 20 Oct 2011 08:57:26 -0000 1.169 > +++ if_pfsync.c 28 Oct 2011 13:46:12 -0000 > @@ -266,6 +266,8 @@ void pfsync_bulk_status(u_int8_t); > void pfsync_bulk_update(void *); > void pfsync_bulk_fail(void *); > > +u_int32_t pfsync_expires(const struct pf_state *, u_int32_t, u_int8_t); > + > #define PFSYNC_MAX_BULKTRIES 12 > int pfsync_sync_ok; > > @@ -578,13 +580,7 @@ pfsync_state_import(struct pfsync_state > /* copy to state */ > bcopy(&sp->rt_addr, &st->rt_addr, sizeof(st->rt_addr)); > st->creation = time_second - ntohl(sp->creation); > - st->expire = time_second; > - if (sp->expire) { > - /* XXX No adaptive scaling. */ > - st->expire -= r->timeout[sp->timeout] - ntohl(sp->expire); > - } > > - st->expire = ntohl(sp->expire) + time_second; > st->direction = sp->direction; > st->log = sp->log; > st->timeout = sp->timeout; > @@ -603,6 +599,8 @@ pfsync_state_import(struct pfsync_state > st->anchor.ptr = NULL; > st->rt_kif = NULL; > > + st->expire = pfsync_expires(st, ntohl(sp->expire), sp->timeout); > + > st->pfsync_time = time_uptime; > st->sync_state = PFSYNC_S_NONE; > > @@ -948,7 +946,8 @@ pfsync_in_upd(caddr_t buf, int len, int > if (sync < 2) { > pfsync_alloc_scrub_memory(&sp->dst, &st->dst); > pf_state_peer_ntoh(&sp->dst, &st->dst); > - st->expire = ntohl(sp->expire) + time_second; > + st->expire = pfsync_expires(st, ntohl(sp->expire), > + sp->timeout); > st->timeout = sp->timeout; > } > st->pfsync_time = time_uptime; > @@ -1019,10 +1018,12 @@ pfsync_in_upd_c(caddr_t buf, int len, in > else > pf_state_peer_ntoh(&up->dst, &st->dst); > } > + > if (sync < 2) { > pfsync_alloc_scrub_memory(&up->dst, &st->dst); > pf_state_peer_ntoh(&up->dst, &st->dst); > - st->expire = ntohl(up->expire) + time_second; > + st->expire = pfsync_expires(st, ntohl(up->expire), > + up->timeout); > st->timeout = up->timeout; > } > st->pfsync_time = time_uptime; > @@ -2330,4 +2331,37 @@ pfsync_sysctl(int *name, u_int namelen, > default: > return (ENOPROTOOPT); > } > +} > + > +u_int32_t > +pfsync_expires(const struct pf_state *st, u_int32_t expire, u_int8_t ttype) > +{ > + u_int32_t start; > + u_int32_t end; > + u_int32_t states; > + u_int32_t result; > + u_int32_t timeout; > + > + if (expire == 0 || ttype >= PFTM_MAX) > + return (time_second); > + > + timeout = st->rule.ptr->timeout[ttype]; > + result = time_second - timeout + expire; > + start = st->rule.ptr->timeout[PFTM_ADAPTIVE_START]; > + if (start) { > + end = st->rule.ptr->timeout[PFTM_ADAPTIVE_END]; > + states = st->rule.ptr->states_cur; > + } else { > + start = pf_default_rule.timeout[PFTM_ADAPTIVE_START]; > + end = pf_default_rule.timeout[PFTM_ADAPTIVE_END]; > + states = pf_status.states; > + } > + if (end && states > start && start < end) { > + if (states < end) > + return (result + timeout * (end - states) / > + (end - start)); > + else > + return (time_second); > + } > + return (result); > }