RAdv: Buffer prefixes awhile after they disappear
Keep a cache of all the relevant prefixes we send out. When a prefix appears, insert it into the cache. If it dies, keep it there for a while, marked as dead. Send out the dead prefixes with zero lifetime.
This commit is contained in:
parent
3ac5d1ce4c
commit
e2d2b3ef21
3 changed files with 170 additions and 52 deletions
|
@ -70,36 +70,6 @@ struct radv_opt_dnssl
|
||||||
char domain[];
|
char domain[];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
static struct radv_prefix_config default_prefix = {
|
|
||||||
.onlink = 1,
|
|
||||||
.autonomous = 1,
|
|
||||||
.valid_lifetime = DEFAULT_VALID_LIFETIME,
|
|
||||||
.preferred_lifetime = DEFAULT_PREFERRED_LIFETIME
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
static struct radv_prefix_config *
|
|
||||||
radv_prefix_match(struct radv_iface *ifa, struct ifa *a)
|
|
||||||
{
|
|
||||||
struct radv_proto *p = ifa->ra;
|
|
||||||
struct radv_config *cf = (struct radv_config *) (p->p.cf);
|
|
||||||
struct radv_prefix_config *pc;
|
|
||||||
|
|
||||||
if (a->scope <= SCOPE_LINK)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
WALK_LIST(pc, ifa->cf->pref_list)
|
|
||||||
if ((a->pxlen >= pc->pxlen) && ipa_in_net(a->prefix, pc->prefix, pc->pxlen))
|
|
||||||
return pc;
|
|
||||||
|
|
||||||
WALK_LIST(pc, cf->pref_list)
|
|
||||||
if ((a->pxlen >= pc->pxlen) && ipa_in_net(a->prefix, pc->prefix, pc->pxlen))
|
|
||||||
return pc;
|
|
||||||
|
|
||||||
return &default_prefix;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
radv_prepare_rdnss(struct radv_iface *ifa, list *rdnss_list, char **buf, char *bufend)
|
radv_prepare_rdnss(struct radv_iface *ifa, list *rdnss_list, char **buf, char *bufend)
|
||||||
{
|
{
|
||||||
|
@ -236,9 +206,10 @@ radv_prepare_dnssl(struct radv_iface *ifa, list *dnssl_list, char **buf, char *b
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
radv_prepare_prefix(struct radv_iface *ifa, struct radv_prefix_config *pc,
|
radv_prepare_prefix(struct radv_iface *ifa, struct radv_prefix *prefix,
|
||||||
struct ifa *addr, char **buf, char *bufend)
|
char **buf, char *bufend)
|
||||||
{
|
{
|
||||||
|
struct radv_prefix_config *pc = prefix->config;
|
||||||
struct radv_opt_prefix *op = (void *) *buf;
|
struct radv_opt_prefix *op = (void *) *buf;
|
||||||
|
|
||||||
if (*buf + sizeof(*op) > bufend)
|
if (*buf + sizeof(*op) > bufend)
|
||||||
|
@ -250,7 +221,7 @@ radv_prepare_prefix(struct radv_iface *ifa, struct radv_prefix_config *pc,
|
||||||
|
|
||||||
op->type = OPT_PREFIX;
|
op->type = OPT_PREFIX;
|
||||||
op->length = 4;
|
op->length = 4;
|
||||||
op->pxlen = addr->pxlen;
|
op->pxlen = prefix->len;
|
||||||
op->flags = (pc->onlink ? OPT_PX_ONLINK : 0) |
|
op->flags = (pc->onlink ? OPT_PX_ONLINK : 0) |
|
||||||
(pc->autonomous ? OPT_PX_AUTONOMOUS : 0);
|
(pc->autonomous ? OPT_PX_AUTONOMOUS : 0);
|
||||||
op->valid_lifetime = (ifa->ra->active || !pc->valid_lifetime_sensitive) ?
|
op->valid_lifetime = (ifa->ra->active || !pc->valid_lifetime_sensitive) ?
|
||||||
|
@ -258,7 +229,7 @@ radv_prepare_prefix(struct radv_iface *ifa, struct radv_prefix_config *pc,
|
||||||
op->preferred_lifetime = (ifa->ra->active || !pc->preferred_lifetime_sensitive) ?
|
op->preferred_lifetime = (ifa->ra->active || !pc->preferred_lifetime_sensitive) ?
|
||||||
htonl(pc->preferred_lifetime) : 0;
|
htonl(pc->preferred_lifetime) : 0;
|
||||||
op->reserved = 0;
|
op->reserved = 0;
|
||||||
op->prefix = addr->prefix;
|
op->prefix = prefix->prefix;
|
||||||
ipa_hton(op->prefix);
|
ipa_hton(op->prefix);
|
||||||
*buf += sizeof(*op);
|
*buf += sizeof(*op);
|
||||||
|
|
||||||
|
@ -300,16 +271,10 @@ radv_prepare_ra(struct radv_iface *ifa)
|
||||||
buf += sizeof (*om);
|
buf += sizeof (*om);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ifa *addr;
|
struct radv_prefix *prefix;
|
||||||
WALK_LIST(addr, ifa->iface->addrs)
|
WALK_LIST(prefix, ifa->prefixes)
|
||||||
{
|
{
|
||||||
struct radv_prefix_config *pc;
|
if (radv_prepare_prefix(ifa, prefix, &buf, bufend) < 0)
|
||||||
pc = radv_prefix_match(ifa, addr);
|
|
||||||
|
|
||||||
if (!pc || pc->skip)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (radv_prepare_prefix(ifa, pc, addr, &buf, bufend) < 0)
|
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -419,7 +384,7 @@ radv_err_hook(sock *sk, int err)
|
||||||
int
|
int
|
||||||
radv_sk_open(struct radv_iface *ifa)
|
radv_sk_open(struct radv_iface *ifa)
|
||||||
{
|
{
|
||||||
sock *sk = sk_new(ifa->ra->p.pool);
|
sock *sk = sk_new(ifa->pool);
|
||||||
sk->type = SK_IP;
|
sk->type = SK_IP;
|
||||||
sk->dport = ICMPV6_PROTO;
|
sk->dport = ICMPV6_PROTO;
|
||||||
sk->saddr = ifa->addr->ip;
|
sk->saddr = ifa->addr->ip;
|
||||||
|
|
|
@ -51,6 +51,15 @@ radv_timer(timer *tm)
|
||||||
|
|
||||||
RADV_TRACE(D_EVENTS, "Timer fired on %s", ifa->iface->name);
|
RADV_TRACE(D_EVENTS, "Timer fired on %s", ifa->iface->name);
|
||||||
|
|
||||||
|
/* If some dead prefixes expired, regenerate the prefix list and the packet.
|
||||||
|
* We do so by pretending there was a change on the interface.
|
||||||
|
*
|
||||||
|
* This sets the timer, but we replace it just at the end of this function
|
||||||
|
* (replacing a timer is fine).
|
||||||
|
*/
|
||||||
|
if (ifa->prefix_expires != 0 && ifa->prefix_expires <= now)
|
||||||
|
radv_iface_notify(ifa, RA_EV_GC);
|
||||||
|
|
||||||
radv_send_ra(ifa, 0);
|
radv_send_ra(ifa, 0);
|
||||||
|
|
||||||
/* Update timer */
|
/* Update timer */
|
||||||
|
@ -67,7 +76,129 @@ radv_timer(timer *tm)
|
||||||
tm_start(ifa->timer, after);
|
tm_start(ifa->timer, after);
|
||||||
}
|
}
|
||||||
|
|
||||||
static char* ev_name[] = { NULL, "Init", "Change", "RS" };
|
static struct radv_prefix_config default_prefix = {
|
||||||
|
.onlink = 1,
|
||||||
|
.autonomous = 1,
|
||||||
|
.valid_lifetime = DEFAULT_VALID_LIFETIME,
|
||||||
|
.preferred_lifetime = DEFAULT_PREFERRED_LIFETIME
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct radv_prefix_config dead_prefix = {
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Find a corresponding config for the given prefix */
|
||||||
|
static struct radv_prefix_config *
|
||||||
|
radv_prefix_match(struct radv_iface *ifa, struct ifa *a)
|
||||||
|
{
|
||||||
|
struct radv_proto *p = ifa->ra;
|
||||||
|
struct radv_config *cf = (struct radv_config *) (p->p.cf);
|
||||||
|
struct radv_prefix_config *pc;
|
||||||
|
|
||||||
|
if (a->scope <= SCOPE_LINK)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
WALK_LIST(pc, ifa->cf->pref_list)
|
||||||
|
if ((a->pxlen >= pc->pxlen) && ipa_in_net(a->prefix, pc->prefix, pc->pxlen))
|
||||||
|
return pc;
|
||||||
|
|
||||||
|
WALK_LIST(pc, cf->pref_list)
|
||||||
|
if ((a->pxlen >= pc->pxlen) && ipa_in_net(a->prefix, pc->prefix, pc->pxlen))
|
||||||
|
return pc;
|
||||||
|
|
||||||
|
return &default_prefix;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Go through the list of prefixes, compare them with configs and decide if we
|
||||||
|
* want them or not. */
|
||||||
|
static void
|
||||||
|
prefixes_prepare(struct radv_iface *ifa)
|
||||||
|
{
|
||||||
|
struct radv_proto *p = ifa->ra;
|
||||||
|
/* First mark all the prefixes as unused */
|
||||||
|
struct radv_prefix *pfx;
|
||||||
|
|
||||||
|
WALK_LIST(pfx, ifa->prefixes)
|
||||||
|
pfx->mark = 0;
|
||||||
|
|
||||||
|
/* Now find all the prefixes we want to use and make sure they are in the
|
||||||
|
* list. */
|
||||||
|
struct ifa *addr;
|
||||||
|
WALK_LIST(addr, ifa->iface->addrs)
|
||||||
|
{
|
||||||
|
struct radv_prefix_config *pc = radv_prefix_match(ifa, addr);
|
||||||
|
|
||||||
|
if (!pc || pc->skip)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* Do we have it already? */
|
||||||
|
struct radv_prefix *existing = NULL;
|
||||||
|
WALK_LIST(pfx, ifa->prefixes)
|
||||||
|
if (pfx->len == addr->pxlen &&
|
||||||
|
memcmp(&pfx->prefix, &addr->prefix, sizeof pfx->prefix) == 0)
|
||||||
|
{
|
||||||
|
existing = pfx;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!existing)
|
||||||
|
{
|
||||||
|
RADV_TRACE(D_EVENTS, "Allocating new prefix %I on %s", addr->prefix,
|
||||||
|
ifa->iface->name);
|
||||||
|
existing = mb_allocz(ifa->pool, sizeof *existing);
|
||||||
|
existing->prefix = addr->prefix;
|
||||||
|
existing->len = addr->pxlen;
|
||||||
|
add_tail(&ifa->prefixes, NODE existing);
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
* Update the information (it may have changed, or even bring a prefix back
|
||||||
|
* to life).
|
||||||
|
*/
|
||||||
|
existing->alive = 1;
|
||||||
|
existing->mark = 1;
|
||||||
|
existing->config = pc;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Garbage-collect the prefixes. If something isn't used, it dies (but isn't
|
||||||
|
* dropped just yet). If something is dead and rots there for long enough,
|
||||||
|
* clean it up.
|
||||||
|
*/
|
||||||
|
// XXX: Make these 5 minutes it configurable
|
||||||
|
bird_clock_t rotten = now + 300;
|
||||||
|
struct radv_prefix *next;
|
||||||
|
bird_clock_t expires_soonest = 0;
|
||||||
|
WALK_LIST_DELSAFE(pfx, next, ifa->prefixes) {
|
||||||
|
if (pfx->alive && !pfx->mark)
|
||||||
|
{
|
||||||
|
RADV_TRACE(D_EVENTS, "Marking prefix %I on %s as dead", pfx->prefix,
|
||||||
|
ifa->iface->name);
|
||||||
|
// It just died
|
||||||
|
pfx->alive = 0;
|
||||||
|
pfx->expires = rotten;
|
||||||
|
pfx->config = &dead_prefix;
|
||||||
|
}
|
||||||
|
if (!pfx->alive)
|
||||||
|
if (pfx->expires <= now)
|
||||||
|
{
|
||||||
|
RADV_TRACE(D_EVENTS, "Dropping long dead prefix %I on %s", pfx->prefix,
|
||||||
|
ifa->iface->name);
|
||||||
|
// It's dead and rotten, clean it up
|
||||||
|
rem_node(NODE pfx);
|
||||||
|
mb_free(pfx);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ASSERT(pfx->expires != 0);
|
||||||
|
// Let it rot for a while more, but look when it's ripe.
|
||||||
|
if (expires_soonest == 0 || pfx->expires < expires_soonest)
|
||||||
|
expires_soonest = pfx->expires;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ifa->prefix_expires = expires_soonest;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char* ev_name[] = { NULL, "Init", "Change", "RS", "Garbage collect" };
|
||||||
|
|
||||||
void
|
void
|
||||||
radv_iface_notify(struct radv_iface *ifa, int event)
|
radv_iface_notify(struct radv_iface *ifa, int event)
|
||||||
|
@ -82,6 +213,7 @@ radv_iface_notify(struct radv_iface *ifa, int event)
|
||||||
switch (event)
|
switch (event)
|
||||||
{
|
{
|
||||||
case RA_EV_CHANGE:
|
case RA_EV_CHANGE:
|
||||||
|
case RA_EV_GC:
|
||||||
ifa->plen = 0;
|
ifa->plen = 0;
|
||||||
case RA_EV_INIT:
|
case RA_EV_INIT:
|
||||||
ifa->initial = MAX_INITIAL_RTR_ADVERTISEMENTS;
|
ifa->initial = MAX_INITIAL_RTR_ADVERTISEMENTS;
|
||||||
|
@ -91,6 +223,8 @@ radv_iface_notify(struct radv_iface *ifa, int event)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
prefixes_prepare(ifa);
|
||||||
|
|
||||||
/* Update timer */
|
/* Update timer */
|
||||||
unsigned delta = now - ifa->last;
|
unsigned delta = now - ifa->last;
|
||||||
unsigned after = 0;
|
unsigned after = 0;
|
||||||
|
@ -152,15 +286,17 @@ find_lladdr(struct iface *iface)
|
||||||
static void
|
static void
|
||||||
radv_iface_new(struct radv_proto *p, struct iface *iface, struct radv_iface_config *cf)
|
radv_iface_new(struct radv_proto *p, struct iface *iface, struct radv_iface_config *cf)
|
||||||
{
|
{
|
||||||
pool *pool = p->p.pool;
|
|
||||||
struct radv_iface *ifa;
|
struct radv_iface *ifa;
|
||||||
|
|
||||||
RADV_TRACE(D_EVENTS, "Adding interface %s", iface->name);
|
RADV_TRACE(D_EVENTS, "Adding interface %s", iface->name);
|
||||||
|
|
||||||
|
pool *pool = rp_new(p->p.pool, iface->name);
|
||||||
ifa = mb_allocz(pool, sizeof(struct radv_iface));
|
ifa = mb_allocz(pool, sizeof(struct radv_iface));
|
||||||
|
ifa->pool = pool;
|
||||||
ifa->ra = p;
|
ifa->ra = p;
|
||||||
ifa->cf = cf;
|
ifa->cf = cf;
|
||||||
ifa->iface = iface;
|
ifa->iface = iface;
|
||||||
|
init_list(&ifa->prefixes);
|
||||||
|
|
||||||
add_tail(&p->iface_list, NODE ifa);
|
add_tail(&p->iface_list, NODE ifa);
|
||||||
|
|
||||||
|
@ -198,11 +334,7 @@ radv_iface_remove(struct radv_iface *ifa)
|
||||||
|
|
||||||
rem_node(NODE ifa);
|
rem_node(NODE ifa);
|
||||||
|
|
||||||
rfree(ifa->sk);
|
rfree(ifa->pool);
|
||||||
rfree(ifa->timer);
|
|
||||||
rfree(ifa->lock);
|
|
||||||
|
|
||||||
mb_free(ifa);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
|
@ -121,6 +121,23 @@ struct radv_proto
|
||||||
u8 active; /* Whether radv is active w.r.t. triggers */
|
u8 active; /* Whether radv is active w.r.t. triggers */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct radv_prefix /* One prefix we advertise */
|
||||||
|
{
|
||||||
|
node n;
|
||||||
|
ip_addr prefix;
|
||||||
|
u8 len;
|
||||||
|
/* Is the prefix alive? If not, we advertise it with 0 lifetime, so clients
|
||||||
|
* stop using it. */
|
||||||
|
u8 alive;
|
||||||
|
u8 mark; /* A temporary mark for processing */
|
||||||
|
/* The (absolute) time when we drop this prefix from advertising. It is valid
|
||||||
|
* only if !alive. */
|
||||||
|
bird_clock_t expires;
|
||||||
|
/* The config tied to this prefix. Always valid (we place a dummy config here
|
||||||
|
* when !alive). */
|
||||||
|
struct radv_prefix_config *config;
|
||||||
|
};
|
||||||
|
|
||||||
struct radv_iface
|
struct radv_iface
|
||||||
{
|
{
|
||||||
node n;
|
node n;
|
||||||
|
@ -128,6 +145,9 @@ struct radv_iface
|
||||||
struct radv_iface_config *cf; /* Related config, must be updated in reconfigure */
|
struct radv_iface_config *cf; /* Related config, must be updated in reconfigure */
|
||||||
struct iface *iface;
|
struct iface *iface;
|
||||||
struct ifa *addr; /* Link-local address of iface */
|
struct ifa *addr; /* Link-local address of iface */
|
||||||
|
struct pool *pool; /* A pool for interface-specific things */
|
||||||
|
list prefixes; /* The prefixes we advertise */
|
||||||
|
bird_clock_t prefix_expires; /* When the soonest prefix expires (0 = none dead) */
|
||||||
|
|
||||||
timer *timer;
|
timer *timer;
|
||||||
struct object_lock *lock;
|
struct object_lock *lock;
|
||||||
|
@ -135,12 +155,13 @@ struct radv_iface
|
||||||
|
|
||||||
bird_clock_t last; /* Time of last sending of RA */
|
bird_clock_t last; /* Time of last sending of RA */
|
||||||
u16 plen; /* Length of prepared RA in tbuf, or 0 if not valid */
|
u16 plen; /* Length of prepared RA in tbuf, or 0 if not valid */
|
||||||
byte initial; /* List of active ifaces */
|
byte initial; /* How many RAs are still to be sent as initial */
|
||||||
};
|
};
|
||||||
|
|
||||||
#define RA_EV_INIT 1 /* Switch to initial mode */
|
#define RA_EV_INIT 1 /* Switch to initial mode */
|
||||||
#define RA_EV_CHANGE 2 /* Change of options or prefixes */
|
#define RA_EV_CHANGE 2 /* Change of options or prefixes */
|
||||||
#define RA_EV_RS 3 /* Received RS */
|
#define RA_EV_RS 3 /* Received RS */
|
||||||
|
#define RA_EV_GC 4 /* Internal garbage collection of prefixes */
|
||||||
|
|
||||||
/* Default Router Preferences (RFC 4191) */
|
/* Default Router Preferences (RFC 4191) */
|
||||||
#define RA_PREF_LOW 0x18
|
#define RA_PREF_LOW 0x18
|
||||||
|
|
Loading…
Reference in a new issue