From 5fe9f9631d3be324202667f26e93cddead762e8c Mon Sep 17 00:00:00 2001 From: vivek Date: Tue, 22 Mar 2016 17:46:30 +0000 Subject: Quagga: Make routemap updates or deletes work for VRFs Updates to routemaps and delete of the routemap were not working properly for VRFs. This was because while routemaps are global, the routemap update processing timer and the processing were at the per-instance level. This approach was unable to handle processing for multiple instances as the routemap has no tracking of which instances are still pending processing. This lead to the processing happening correctly only for the first instance - which could be the default instance or some other instance. It could also result in reference to freed memory for an instance. The fix done is to make the update/delete processing also global and not per instance. This means that the route-map delay timer will be global and a global thread will handle the change (or delete) for all instances instead of spawning a separate thread for each instance. To support this, a global BGP command "bgp route-map delay-timer " has been implemented. The existing command per-instance is not deleted but will update the global timer. Signed-off-by: Vivek Venkatraman Reviewed-by: Donald Sharp Ticket: CM-6970, CM-9918 Reviewed By: CCR-4320 Testing Done: Manual, bgpsmoke --- bgpd/bgp_routemap.c | 63 ++++++++++++++++++++++++++--------------------------- 1 file changed, 31 insertions(+), 32 deletions(-) (limited to 'bgpd/bgp_routemap.c') diff --git a/bgpd/bgp_routemap.c b/bgpd/bgp_routemap.c index 38e5bd729..311ebb363 100644 --- a/bgpd/bgp_routemap.c +++ b/bgpd/bgp_routemap.c @@ -2804,8 +2804,8 @@ bgp_route_map_update_peer_group(const char *rmap_name, struct route_map *map, * called for each route-map and within this function we walk the list of peers, * network statements, etc looking to see if they use this route-map. */ -static int -bgp_route_map_process_update (void *arg, const char *rmap_name, int route_update) +static void +bgp_route_map_process_update (struct bgp *bgp, const char *rmap_name, int route_update) { int i; afi_t afi; @@ -2813,14 +2813,10 @@ bgp_route_map_process_update (void *arg, const char *rmap_name, int route_update struct peer *peer; struct bgp_node *bn; struct bgp_static *bgp_static; - struct bgp *bgp = (struct bgp *)arg; struct listnode *node, *nnode; struct route_map *map; char buf[INET6_ADDRSTRLEN]; - if (!bgp) - return (-1); - map = route_map_lookup_by_name (rmap_name); for (ALL_LIST_ELEMENTS (bgp->peer, node, nnode, peer)) @@ -2918,24 +2914,26 @@ bgp_route_map_process_update (void *arg, const char *rmap_name, int route_update } } } - - return (0); } static int -bgp_route_map_process_update_cb (void *arg, char *rmap_name) +bgp_route_map_process_update_cb (char *rmap_name) { - return bgp_route_map_process_update (arg, rmap_name, 1); + struct listnode *node, *nnode; + struct bgp *bgp; + + for (ALL_LIST_ELEMENTS (bm->bgp, node, nnode, bgp)) + bgp_route_map_process_update(bgp, rmap_name, 1); + + return 0; } int bgp_route_map_update_timer(struct thread *thread) { - struct bgp *bgp = THREAD_ARG(thread); + bm->t_rmap_update = NULL; - bgp->t_rmap_update = NULL; - - route_map_walk_update_list((void *)bgp, bgp_route_map_process_update_cb); + route_map_walk_update_list(bgp_route_map_process_update_cb); return (0); } @@ -2943,26 +2941,27 @@ bgp_route_map_update_timer(struct thread *thread) static void bgp_route_map_mark_update (const char *rmap_name) { - struct listnode *node, *nnode; - struct bgp *bgp; - - for (ALL_LIST_ELEMENTS (bm->bgp, node, nnode, bgp)) + if (bm->t_rmap_update == NULL) { + struct listnode *node, *nnode; + struct bgp *bgp; - if (bgp->t_rmap_update == NULL) - { - /* rmap_update_timer of 0 means don't do route updates */ - if (bgp->rmap_update_timer) - { - bgp->t_rmap_update = - thread_add_timer(bm->master, bgp_route_map_update_timer, bgp, - bgp->rmap_update_timer); - /* Signal the groups that a route-map update event has started */ - update_group_policy_update(bgp, BGP_POLICY_ROUTE_MAP, rmap_name, 1, 1); - } - else - bgp_route_map_process_update((void *)bgp, rmap_name, 0); - } + /* rmap_update_timer of 0 means don't do route updates */ + if (bm->rmap_update_timer) + { + bm->t_rmap_update = + thread_add_timer(bm->master, bgp_route_map_update_timer, NULL, + bm->rmap_update_timer); + + /* Signal the groups that a route-map update event has started */ + for (ALL_LIST_ELEMENTS (bm->bgp, node, nnode, bgp)) + update_group_policy_update(bgp, BGP_POLICY_ROUTE_MAP, rmap_name, 1, 1); + } + else + { + for (ALL_LIST_ELEMENTS (bm->bgp, node, nnode, bgp)) + bgp_route_map_process_update(bgp, rmap_name, 0); + } } } -- cgit v1.2.3