mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2024-12-22 17:51:53 +00:00
Kernel: refeed routes and delete them from kernel before actual shutdown
This commit is contained in:
parent
e65a5257b2
commit
5cdc1b679c
@ -346,7 +346,7 @@ krt_is_installed(struct krt_proto *p, net *n)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static uint
|
static uint
|
||||||
rte_feed_count(net *n)
|
rte_feed_count_valid(net *n)
|
||||||
{
|
{
|
||||||
uint count = 0;
|
uint count = 0;
|
||||||
for (struct rte_storage *e = n->routes; e; e = e->next)
|
for (struct rte_storage *e = n->routes; e; e = e->next)
|
||||||
@ -356,7 +356,7 @@ rte_feed_count(net *n)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
rte_feed_obtain(net *n, const rte **feed, uint count)
|
rte_feed_obtain_valid(net *n, const rte **feed, uint count)
|
||||||
{
|
{
|
||||||
uint i = 0;
|
uint i = 0;
|
||||||
for (struct rte_storage *e = n->routes; e; e = e->next)
|
for (struct rte_storage *e = n->routes; e; e = e->next)
|
||||||
@ -383,12 +383,12 @@ krt_export_net(struct krt_proto *p, net *net)
|
|||||||
|
|
||||||
if (c->ra_mode == RA_MERGED)
|
if (c->ra_mode == RA_MERGED)
|
||||||
{
|
{
|
||||||
uint count = rte_feed_count(net);
|
uint count = rte_feed_count_valid(net);
|
||||||
if (!count)
|
if (!count)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
const rte **feed = alloca(count * sizeof(rte *));
|
const rte **feed = alloca(count * sizeof(rte *));
|
||||||
rte_feed_obtain(net, feed, count);
|
rte_feed_obtain_valid(net, feed, count);
|
||||||
return rt_export_merged(c, feed, count, krt_filter_lp, 1);
|
return rt_export_merged(c, feed, count, krt_filter_lp, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -436,6 +436,10 @@ krt_same_dest(rte *k, rte *e)
|
|||||||
void
|
void
|
||||||
krt_got_route(struct krt_proto *p, rte *e, s8 src)
|
krt_got_route(struct krt_proto *p, rte *e, s8 src)
|
||||||
{
|
{
|
||||||
|
/* Ignore when flushing from table */
|
||||||
|
if (p->flush_routes == 1)
|
||||||
|
return;
|
||||||
|
|
||||||
rte *new = NULL;
|
rte *new = NULL;
|
||||||
e->pflags = 0;
|
e->pflags = 0;
|
||||||
|
|
||||||
@ -465,8 +469,8 @@ krt_got_route(struct krt_proto *p, rte *e, s8 src)
|
|||||||
RT_LOCKED(p->p.main_channel->table, tab)
|
RT_LOCKED(p->p.main_channel->table, tab)
|
||||||
{
|
{
|
||||||
|
|
||||||
/* Deleting all routes if flush is requested */
|
/* Deleting all routes if final flush is requested */
|
||||||
if (p->flush_routes)
|
if (p->flush_routes == 2)
|
||||||
goto delete;
|
goto delete;
|
||||||
|
|
||||||
/* We wait for the initial feed to have correct installed state */
|
/* We wait for the initial feed to have correct installed state */
|
||||||
@ -560,17 +564,6 @@ krt_prune(struct krt_proto *p)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
krt_flush_routes(struct krt_proto *p)
|
|
||||||
{
|
|
||||||
KRT_TRACE(p, D_EVENTS, "Flushing kernel routes");
|
|
||||||
p->flush_routes = 1;
|
|
||||||
krt_init_scan(p);
|
|
||||||
krt_do_scan(p);
|
|
||||||
/* No prune! */
|
|
||||||
p->flush_routes = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
krt_got_route_async(struct krt_proto *p, rte *e, int new, s8 src)
|
krt_got_route_async(struct krt_proto *p, rte *e, int new, s8 src)
|
||||||
{
|
{
|
||||||
@ -740,8 +733,11 @@ krt_rt_notify(struct proto *P, struct channel *ch UNUSED, const net_addr *net,
|
|||||||
{
|
{
|
||||||
struct krt_proto *p = (struct krt_proto *) P;
|
struct krt_proto *p = (struct krt_proto *) P;
|
||||||
|
|
||||||
if (config->shutdown)
|
if (p->flush_routes)
|
||||||
|
{
|
||||||
|
krt_replace_rte(p, net, NULL, old ?: new);
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_SINGLE_ROUTE
|
#ifdef CONFIG_SINGLE_ROUTE
|
||||||
/* Got the same route as we imported. Keep it, do nothing. */
|
/* Got the same route as we imported. Keep it, do nothing. */
|
||||||
@ -784,11 +780,23 @@ krt_reload_routes(struct channel *C)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void krt_cleanup(struct krt_proto *p);
|
||||||
|
|
||||||
static void
|
static void
|
||||||
krt_feed_end(struct channel *C)
|
krt_feed_end(struct channel *C)
|
||||||
{
|
{
|
||||||
struct krt_proto *p = (void *) C->proto;
|
struct krt_proto *p = (void *) C->proto;
|
||||||
|
|
||||||
|
if (p->flush_routes)
|
||||||
|
{
|
||||||
|
p->flush_routes = 2;
|
||||||
|
krt_init_scan(p);
|
||||||
|
krt_do_scan(p);
|
||||||
|
krt_cleanup(p);
|
||||||
|
proto_notify_state(&p->p, PS_DOWN);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
p->ready = 1;
|
p->ready = 1;
|
||||||
krt_scan_timer_kick(p);
|
krt_scan_timer_kick(p);
|
||||||
}
|
}
|
||||||
@ -907,21 +915,32 @@ krt_shutdown(struct proto *P)
|
|||||||
|
|
||||||
krt_scan_timer_stop(p);
|
krt_scan_timer_stop(p);
|
||||||
|
|
||||||
/* FIXME we should flush routes even when persist during reconfiguration */
|
|
||||||
if (p->initialized && !KRT_CF->persist && (P->down_code != PDC_CMD_GR_DOWN))
|
|
||||||
krt_flush_routes(p);
|
|
||||||
|
|
||||||
p->ready = 0;
|
|
||||||
p->initialized = 0;
|
|
||||||
|
|
||||||
if (p->p.proto_state == PS_START)
|
if (p->p.proto_state == PS_START)
|
||||||
return PS_DOWN;
|
return PS_DOWN;
|
||||||
|
|
||||||
|
/* FIXME we should flush routes even when persist during reconfiguration */
|
||||||
|
if (p->initialized && !KRT_CF->persist && (P->down_code != PDC_CMD_GR_DOWN))
|
||||||
|
{
|
||||||
|
p->flush_routes = 1;
|
||||||
|
channel_request_feeding(p->p.main_channel);
|
||||||
|
return PS_UP;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
krt_cleanup(p);
|
||||||
|
return PS_DOWN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
krt_cleanup(struct krt_proto *p)
|
||||||
|
{
|
||||||
|
p->ready = 0;
|
||||||
|
p->initialized = 0;
|
||||||
|
|
||||||
krt_sys_shutdown(p);
|
krt_sys_shutdown(p);
|
||||||
rem_node(&p->krt_node);
|
rem_node(&p->krt_node);
|
||||||
bmap_free(&p->sync_map);
|
bmap_free(&p->sync_map);
|
||||||
|
|
||||||
return PS_DOWN;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
Loading…
Reference in New Issue
Block a user