mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-03 07:31:54 +00:00
BFD: fixed a request pickup race condition
When several BGPs requested a BFD session in short time, chances were that the second BGP would file a request while the pickup routine was still running and it would get enqueued into the waiting list instead of being picked up. Fixed this by enforcing pickup loop restart when new requests got added, and also by atomically moving the unpicked requests to a temporary list to announce admin down before actually being added into the wait list.
This commit is contained in:
parent
4a69a64745
commit
22a0900ec2
@ -121,6 +121,7 @@ static struct {
|
|||||||
list wait_list;
|
list wait_list;
|
||||||
list pickup_list;
|
list pickup_list;
|
||||||
list proto_list;
|
list proto_list;
|
||||||
|
uint pickup_reload;
|
||||||
} bfd_global;
|
} bfd_global;
|
||||||
|
|
||||||
const char *bfd_state_names[] = { "AdminDown", "Down", "Init", "Up" };
|
const char *bfd_state_names[] = { "AdminDown", "Down", "Init", "Up" };
|
||||||
@ -670,18 +671,29 @@ bfd_add_request(struct bfd_proto *p, struct bfd_request *req)
|
|||||||
struct bfd_config *cf = (struct bfd_config *) (p->p.cf);
|
struct bfd_config *cf = (struct bfd_config *) (p->p.cf);
|
||||||
|
|
||||||
if (p->p.vrf && (p->p.vrf != req->vrf))
|
if (p->p.vrf && (p->p.vrf != req->vrf))
|
||||||
|
{
|
||||||
|
TRACE(D_EVENTS, "Not accepting request to %I with different VRF", req->addr);
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (ipa_is_ip4(req->addr) ? !cf->accept_ipv4 : !cf->accept_ipv6)
|
if (ipa_is_ip4(req->addr) ? !cf->accept_ipv4 : !cf->accept_ipv6)
|
||||||
|
{
|
||||||
|
TRACE(D_EVENTS, "Not accepting request to %I (AF limit)", req->addr);
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (req->iface ? !cf->accept_direct : !cf->accept_multihop)
|
if (req->iface ? !cf->accept_direct : !cf->accept_multihop)
|
||||||
|
{
|
||||||
|
TRACE(D_EVENTS, "Not accepting %s request to %I", req->iface ? "direct" : "multihop", req->addr);
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
uint ifindex = req->iface ? req->iface->index : 0;
|
uint ifindex = req->iface ? req->iface->index : 0;
|
||||||
struct bfd_session *s = bfd_find_session_by_addr(p, req->addr, ifindex);
|
struct bfd_session *s = bfd_find_session_by_addr(p, req->addr, ifindex);
|
||||||
|
|
||||||
if (!s)
|
if (s)
|
||||||
|
TRACE(D_EVENTS, "Session to %I reused", s->addr);
|
||||||
|
else
|
||||||
s = bfd_add_session(p, req->addr, req->local, req->iface, &req->opts);
|
s = bfd_add_session(p, req->addr, req->local, req->iface, &req->opts);
|
||||||
|
|
||||||
rem_node(&req->n);
|
rem_node(&req->n);
|
||||||
@ -725,33 +737,57 @@ bfd_pickup_requests(void *_data UNUSED)
|
|||||||
* Thank you, my future self, for understanding. Have a nice day!
|
* Thank you, my future self, for understanding. Have a nice day!
|
||||||
*/
|
*/
|
||||||
|
|
||||||
node *n;
|
DBG("BFD pickup loop starting");
|
||||||
WALK_LIST(n, bfd_global.proto_list)
|
|
||||||
{
|
|
||||||
struct bfd_proto *p = SKIP_BACK(struct bfd_proto, bfd_node, n);
|
|
||||||
birdloop_enter(p->p.loop);
|
|
||||||
BFD_LOCK;
|
|
||||||
|
|
||||||
node *rn, *rnxt;
|
|
||||||
WALK_LIST_DELSAFE(rn, rnxt, bfd_global.pickup_list)
|
|
||||||
bfd_add_request(p, SKIP_BACK(struct bfd_request, n, rn));
|
|
||||||
|
|
||||||
BFD_UNLOCK;
|
|
||||||
birdloop_leave(p->p.loop);
|
|
||||||
}
|
|
||||||
|
|
||||||
BFD_LOCK;
|
BFD_LOCK;
|
||||||
while (!EMPTY_LIST(bfd_global.pickup_list))
|
do {
|
||||||
{
|
bfd_global.pickup_reload = 0;
|
||||||
struct bfd_request *req = SKIP_BACK(struct bfd_request, n, HEAD(bfd_global.pickup_list));
|
|
||||||
rem_node(&req->n);
|
|
||||||
BFD_UNLOCK;
|
BFD_UNLOCK;
|
||||||
|
|
||||||
bfd_request_notify(req, BFD_STATE_ADMIN_DOWN, 0);
|
node *n;
|
||||||
|
WALK_LIST(n, bfd_global.proto_list)
|
||||||
|
{
|
||||||
|
struct bfd_proto *p = SKIP_BACK(struct bfd_proto, bfd_node, n);
|
||||||
|
birdloop_enter(p->p.loop);
|
||||||
|
BFD_LOCK;
|
||||||
|
|
||||||
|
TRACE(D_EVENTS, "Picking up new requests (%d available)", list_length(&bfd_global.pickup_list));
|
||||||
|
|
||||||
|
node *rn, *rnxt;
|
||||||
|
WALK_LIST_DELSAFE(rn, rnxt, bfd_global.pickup_list)
|
||||||
|
bfd_add_request(p, SKIP_BACK(struct bfd_request, n, rn));
|
||||||
|
|
||||||
|
BFD_UNLOCK;
|
||||||
|
|
||||||
|
/* Remove sessions with no requests */
|
||||||
|
HASH_WALK_DELSAFE(p->session_hash_id, next_id, s)
|
||||||
|
{
|
||||||
|
if (EMPTY_LIST(s->request_list))
|
||||||
|
bfd_remove_session_locked(p, s);
|
||||||
|
}
|
||||||
|
HASH_WALK_END;
|
||||||
|
|
||||||
|
birdloop_leave(p->p.loop);
|
||||||
|
}
|
||||||
|
|
||||||
BFD_LOCK;
|
BFD_LOCK;
|
||||||
add_tail(&bfd_global.wait_list, &req->n);
|
} while (bfd_global.pickup_reload);
|
||||||
}
|
|
||||||
|
list tmp_list;
|
||||||
|
init_list(&tmp_list);
|
||||||
|
add_tail_list(&tmp_list, &bfd_global.pickup_list);
|
||||||
|
|
||||||
|
init_list(&bfd_global.pickup_list);
|
||||||
|
BFD_UNLOCK;
|
||||||
|
|
||||||
|
log(L_TRACE "No protocol for %d BFD requests", list_length(&tmp_list));
|
||||||
|
|
||||||
|
node *n;
|
||||||
|
WALK_LIST(n, tmp_list)
|
||||||
|
bfd_request_notify(SKIP_BACK(struct bfd_request, n, n), BFD_STATE_ADMIN_DOWN, 0);
|
||||||
|
|
||||||
|
BFD_LOCK;
|
||||||
|
add_tail_list(&bfd_global.wait_list, &tmp_list);
|
||||||
BFD_UNLOCK;
|
BFD_UNLOCK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -817,8 +853,10 @@ bfd_request_session(pool *p, ip_addr addr, ip_addr local,
|
|||||||
req->session = NULL;
|
req->session = NULL;
|
||||||
|
|
||||||
BFD_LOCK;
|
BFD_LOCK;
|
||||||
|
bfd_global.pickup_reload++;
|
||||||
add_tail(&bfd_global.pickup_list, &req->n);
|
add_tail(&bfd_global.pickup_list, &req->n);
|
||||||
ev_send(&global_event_list, &bfd_pickup_event);
|
ev_send(&global_event_list, &bfd_pickup_event);
|
||||||
|
DBG("New BFD request enlisted.\n");
|
||||||
BFD_UNLOCK;
|
BFD_UNLOCK;
|
||||||
|
|
||||||
return req;
|
return req;
|
||||||
@ -842,15 +880,12 @@ static void
|
|||||||
bfd_request_free(resource *r)
|
bfd_request_free(resource *r)
|
||||||
{
|
{
|
||||||
struct bfd_request *req = (struct bfd_request *) r;
|
struct bfd_request *req = (struct bfd_request *) r;
|
||||||
struct bfd_session *s = req->session;
|
|
||||||
|
|
||||||
|
BFD_LOCK;
|
||||||
rem_node(&req->n);
|
rem_node(&req->n);
|
||||||
|
BFD_UNLOCK;
|
||||||
|
|
||||||
/* Remove the session if there is no request for it. Skip that if
|
ev_send(&global_event_list, &bfd_pickup_event);
|
||||||
inside notify hooks, will be handled by bfd_notify_hook() itself */
|
|
||||||
|
|
||||||
if (s && EMPTY_LIST(s->request_list) && !s->notify_running)
|
|
||||||
bfd_remove_session(s->ifa->bfd, s);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -1007,10 +1042,8 @@ bfd_notify_hook(void *data)
|
|||||||
diag = s->loc_diag;
|
diag = s->loc_diag;
|
||||||
bfd_unlock_sessions(p);
|
bfd_unlock_sessions(p);
|
||||||
|
|
||||||
s->notify_running = 1;
|
|
||||||
WALK_LIST_DELSAFE(n, nn, s->request_list)
|
WALK_LIST_DELSAFE(n, nn, s->request_list)
|
||||||
bfd_request_notify(SKIP_BACK(struct bfd_request, n, n), state, diag);
|
bfd_request_notify(SKIP_BACK(struct bfd_request, n, n), state, diag);
|
||||||
s->notify_running = 0;
|
|
||||||
|
|
||||||
/* Remove the session if all requests were removed in notify hooks */
|
/* Remove the session if all requests were removed in notify hooks */
|
||||||
if (EMPTY_LIST(s->request_list))
|
if (EMPTY_LIST(s->request_list))
|
||||||
|
@ -165,7 +165,6 @@ struct bfd_session
|
|||||||
|
|
||||||
list request_list; /* List of client requests (struct bfd_request) */
|
list request_list; /* List of client requests (struct bfd_request) */
|
||||||
btime last_state_change; /* Time of last state change */
|
btime last_state_change; /* Time of last state change */
|
||||||
u8 notify_running; /* 1 if notify hooks are running */
|
|
||||||
|
|
||||||
u8 rx_csn_known; /* Received crypto sequence number is known */
|
u8 rx_csn_known; /* Received crypto sequence number is known */
|
||||||
u32 rx_csn; /* Last received crypto sequence number */
|
u32 rx_csn; /* Last received crypto sequence number */
|
||||||
|
@ -1524,15 +1524,22 @@ static void
|
|||||||
bgp_update_bfd(struct bgp_proto *p, const struct bfd_options *bfd)
|
bgp_update_bfd(struct bgp_proto *p, const struct bfd_options *bfd)
|
||||||
{
|
{
|
||||||
if (bfd && p->bfd_req)
|
if (bfd && p->bfd_req)
|
||||||
|
{
|
||||||
|
BGP_TRACE(D_EVENTS, "Updating existing BFD request");
|
||||||
bfd_update_request(p->bfd_req, bfd);
|
bfd_update_request(p->bfd_req, bfd);
|
||||||
|
}
|
||||||
|
|
||||||
if (bfd && !p->bfd_req && !bgp_is_dynamic(p))
|
if (bfd && !p->bfd_req && !bgp_is_dynamic(p))
|
||||||
|
{
|
||||||
p->bfd_req = bfd_request_session(p->p.pool, p->remote_ip, p->local_ip,
|
p->bfd_req = bfd_request_session(p->p.pool, p->remote_ip, p->local_ip,
|
||||||
p->cf->multihop ? NULL : p->neigh->iface,
|
p->cf->multihop ? NULL : p->neigh->iface,
|
||||||
p->p.vrf, bgp_bfd_notify, p, p->p.loop, bfd);
|
p->p.vrf, bgp_bfd_notify, p, p->p.loop, bfd);
|
||||||
|
BGP_TRACE(D_EVENTS, "Requesting a new BFD session");
|
||||||
|
}
|
||||||
|
|
||||||
if (!bfd && p->bfd_req)
|
if (!bfd && p->bfd_req)
|
||||||
{
|
{
|
||||||
|
BGP_TRACE(D_EVENTS, "Retracting the BFD request");
|
||||||
rfree(p->bfd_req);
|
rfree(p->bfd_req);
|
||||||
p->bfd_req = NULL;
|
p->bfd_req = NULL;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user