summaryrefslogtreecommitdiff
path: root/usr.sbin/bgpd/rde_rib.c
diff options
context:
space:
mode:
authorClaudio Jeker <claudio@cvs.openbsd.org>2018-10-31 14:50:08 +0000
committerClaudio Jeker <claudio@cvs.openbsd.org>2018-10-31 14:50:08 +0000
commitde24d4754e553c015daf25d58a0fc7fd9d20452e (patch)
tree14824473d453597774bc9a42a7f2eda7770df76d /usr.sbin/bgpd/rde_rib.c
parent3095a3c4e562ad7fa565df2321341a150af5bb1a (diff)
Remove tail queues which link peer, aspath and prefix together. These
lists are no longer needed and make it possible to share rde_aspath between peers & prefixes. Instead of the lists the rde_aspath is now reference counted. With this struct prefix is now the central place where everything is connected to making the RIB a bit easier to handle. With input and OK denis@
Diffstat (limited to 'usr.sbin/bgpd/rde_rib.c')
-rw-r--r--usr.sbin/bgpd/rde_rib.c111
1 files changed, 59 insertions, 52 deletions
diff --git a/usr.sbin/bgpd/rde_rib.c b/usr.sbin/bgpd/rde_rib.c
index b843d2c5b99..959e02e2fe6 100644
--- a/usr.sbin/bgpd/rde_rib.c
+++ b/usr.sbin/bgpd/rde_rib.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: rde_rib.c,v 1.183 2018/10/31 14:45:36 claudio Exp $ */
+/* $OpenBSD: rde_rib.c,v 1.184 2018/10/31 14:50:07 claudio Exp $ */
/*
* Copyright (c) 2003, 2004 Claudio Jeker <claudio@openbsd.org>
@@ -464,9 +464,10 @@ rib_dump_terminate(u_int16_t id, void *arg,
/* path specific functions */
-static struct rde_aspath *path_lookup(struct rde_aspath *, struct rde_peer *);
+static struct rde_aspath *path_lookup(struct rde_aspath *);
static u_int64_t path_hash(struct rde_aspath *);
-static void path_link(struct rde_aspath *, struct rde_peer *);
+static void path_link(struct rde_aspath *);
+static void path_unlink(struct rde_aspath *);
struct path_table {
struct aspath_head *path_hashtbl;
@@ -475,7 +476,27 @@ struct path_table {
SIPHASH_KEY pathtablekey;
-#define PATH_HASH(x) &pathtable.path_hashtbl[x & pathtable.path_hashmask]
+#define PATH_HASH(x) &pathtable.path_hashtbl[x & pathtable.path_hashmask]
+
+#define path_empty(asp) ((asp)->refcnt <= 0)
+
+static inline void
+path_ref(struct rde_aspath *asp)
+{
+ if ((asp->flags & F_ATTR_LINKED) == 0)
+ fatalx("%s: unlinked object", __func__);
+ asp->refcnt++;
+ rdemem.path_refs++;
+}
+
+static inline void
+path_unref(struct rde_aspath *asp)
+{
+ if ((asp->flags & F_ATTR_LINKED) == 0)
+ fatalx("%s: unlinked object", __func__);
+ asp->refcnt--;
+ rdemem.path_refs--;
+}
void
path_init(u_int32_t hashsize)
@@ -563,10 +584,10 @@ path_update(struct rib *rib, struct rde_peer *peer, struct filterstate *state,
* In both cases lookup the new aspath to make sure it is not
* already in the RIB.
*/
- if ((asp = path_lookup(nasp, peer)) == NULL) {
+ if ((asp = path_lookup(nasp)) == NULL) {
/* Path not available, create and link a new one. */
asp = path_copy(path_get(), nasp);
- path_link(asp, peer);
+ path_link(asp);
}
/* If the prefix was found move it else add it to the aspath. */
@@ -654,7 +675,7 @@ path_hash(struct rde_aspath *asp)
}
static struct rde_aspath *
-path_lookup(struct rde_aspath *aspath, struct rde_peer *peer)
+path_lookup(struct rde_aspath *aspath)
{
struct aspath_head *head;
struct rde_aspath *asp;
@@ -664,57 +685,43 @@ path_lookup(struct rde_aspath *aspath, struct rde_peer *peer)
head = PATH_HASH(hash);
LIST_FOREACH(asp, head, path_l) {
- if (asp->hash == hash && peer == asp->peer &&
- path_compare(aspath, asp) == 0)
+ if (asp->hash == hash && path_compare(aspath, asp) == 0)
return (asp);
}
return (NULL);
}
/*
- * This function can only called when all prefix have been removed first.
- * Normally this happens directly out of the prefix removal functions.
+ * Link this aspath into the global hash table.
+ * The asp had to be alloced with path_get.
*/
-void
-path_destroy(struct rde_aspath *asp)
+static void
+path_link(struct rde_aspath *asp)
{
- /* path_destroy can only unlink and free empty rde_aspath */
- if (!TAILQ_EMPTY(&asp->prefixes))
- log_warnx("path_destroy: still has prefixes, leaking");
-
- LIST_REMOVE(asp, path_l);
- TAILQ_REMOVE(&asp->peer->path_h, asp, peer_l);
- asp->peer = NULL;
- asp->flags &= ~F_ATTR_LINKED;
+ struct aspath_head *head;
- path_put(asp);
-}
+ asp->hash = path_hash(asp);
+ head = PATH_HASH(asp->hash);
-int
-path_empty(struct rde_aspath *asp)
-{
- return TAILQ_EMPTY(&asp->prefixes);
+ LIST_INSERT_HEAD(head, asp, path_l);
+ asp->flags |= F_ATTR_LINKED;
}
/*
- * the path object is linked into multiple lists for fast access.
- * These are peer_l, path_l and nexthop_l.
- * peer_l: list of all aspaths that belong to that peer
- * path_l: hash list to find paths quickly
+ * This function can only be called when all prefix have been removed first.
+ * Normally this happens directly out of the prefix removal functions.
*/
static void
-path_link(struct rde_aspath *asp, struct rde_peer *peer)
+path_unlink(struct rde_aspath *asp)
{
- struct aspath_head *head;
+ /* make sure no reference is hold for this rde_aspath */
+ if (!path_empty(asp))
+ fatalx("%s: still has prefixes", __func__);
- asp->peer = peer;
-
- asp->hash = path_hash(asp);
- head = PATH_HASH(asp->hash);
+ LIST_REMOVE(asp, path_l);
+ asp->flags &= ~F_ATTR_LINKED;
- LIST_INSERT_HEAD(head, asp, path_l);
- TAILQ_INSERT_HEAD(&peer->path_h, asp, peer_l);
- asp->flags |= F_ATTR_LINKED;
+ path_put(asp);
}
/*
@@ -739,6 +746,8 @@ path_copy(struct rde_aspath *dst, const struct rde_aspath *src)
dst->pftableid = pftable_ref(src->pftableid);
dst->flags = src->flags & ~F_ATTR_LINKED;
+ dst->refcnt = 0; /* not linked so no refcnt */
+
attr_copy(dst, src);
return (dst);
@@ -749,7 +758,6 @@ struct rde_aspath *
path_prep(struct rde_aspath *asp)
{
memset(asp, 0, sizeof(*asp));
- TAILQ_INIT(&asp->prefixes);
asp->origin = ORIGIN_INCOMPLETE;
asp->lpref = DEFAULT_LPREF;
@@ -770,11 +778,12 @@ path_get(void)
return (path_prep(asp));
}
+/* clean up an asp after use (frees all references to sub-objects) */
void
path_clean(struct rde_aspath *asp)
{
if (asp->flags & F_ATTR_LINKED)
- fatalx("path_clean: linked object");
+ fatalx("%s: linked object", __func__);
rtlabel_unref(asp->rtlabelid);
pftable_unref(asp->pftableid);
@@ -834,7 +843,7 @@ prefix_add(struct bgpd_addr *prefix, int prefixlen, struct rib *rib,
if (re == NULL)
re = rib_add(rib, prefix, prefixlen);
- p = prefix_bypeer(re, asp->peer);
+ p = prefix_bypeer(re, peer);
if (p == NULL) {
p = prefix_alloc();
prefix_link(p, re, peer, asp, state, vstate);
@@ -875,8 +884,8 @@ prefix_move(struct prefix *p, struct rde_peer *peer,
np->nhflags = state->nhflags;
np->validation_state = vstate;
- /* add to new as path */
- TAILQ_INSERT_HEAD(&asp->prefixes, np, path_l);
+ /* add reference to new as path */
+ path_ref(asp);
/*
* no need to update the peer prefix count because we are only moving
@@ -896,7 +905,7 @@ prefix_move(struct prefix *p, struct rde_peer *peer,
/* remove old prefix node */
oasp = prefix_aspath(p);
- TAILQ_REMOVE(&oasp->prefixes, p, path_l);
+ path_unref(oasp);
/* as before peer count needs no update because of move */
/* destroy all references to other objects and free the old prefix */
@@ -910,7 +919,7 @@ prefix_move(struct prefix *p, struct rde_peer *peer,
/* destroy old path if empty */
if (path_empty(oasp))
- path_destroy(oasp);
+ path_unlink(oasp);
return (0);
}
@@ -1091,7 +1100,7 @@ prefix_destroy(struct prefix *p)
prefix_free(p);
if (path_empty(asp))
- path_destroy(asp);
+ path_unlink(asp);
}
/*
@@ -1101,7 +1110,7 @@ static void
prefix_link(struct prefix *pref, struct rib_entry *re, struct rde_peer *peer,
struct rde_aspath *asp, struct filterstate *state, u_int8_t vstate)
{
- TAILQ_INSERT_HEAD(&asp->prefixes, pref, path_l);
+ path_ref(asp);
pref->aspath = asp;
pref->peer = peer;
@@ -1123,15 +1132,13 @@ static void
prefix_unlink(struct prefix *pref)
{
struct rib_entry *re = pref->re;
- struct prefix_queue *pq;
/* make route decision */
LIST_REMOVE(pref, rib_l);
prefix_evaluate(NULL, re);
- pq = &prefix_aspath(pref)->prefixes;
+ path_unref(prefix_aspath(pref));
- TAILQ_REMOVE(pq, pref, path_l);
if (rib_empty(re))
rib_remove(re);