Annotation of embedaddon/bird2/nest/neighbor.c, revision 1.1
1.1 ! misho 1: /*
! 2: * BIRD -- Neighbor Cache
! 3: *
! 4: * (c) 1998--2000 Martin Mares <mj@ucw.cz>
! 5: * (c) 2008--2018 Ondrej Zajicek <santiago@crfreenet.org>
! 6: * (c) 2008--2018 CZ.NIC z.s.p.o.
! 7: *
! 8: * Can be freely distributed and used under the terms of the GNU GPL.
! 9: */
! 10:
! 11: /**
! 12: * DOC: Neighbor cache
! 13: *
! 14: * Most routing protocols need to associate their internal state data with
! 15: * neighboring routers, check whether an address given as the next hop attribute
! 16: * of a route is really an address of a directly connected host and which
! 17: * interface is it connected through. Also, they often need to be notified when
! 18: * a neighbor ceases to exist or when their long awaited neighbor becomes
! 19: * connected. The neighbor cache is there to solve all these problems.
! 20: *
! 21: * The neighbor cache maintains a collection of neighbor entries. Each entry
! 22: * represents one IP address corresponding to either our directly connected
! 23: * neighbor or our own end of the link (when the scope of the address is set to
! 24: * %SCOPE_HOST) together with per-neighbor data belonging to a single protocol.
! 25: * A neighbor entry may be bound to a specific interface, which is required for
! 26: * link-local IP addresses and optional for global IP addresses.
! 27: *
! 28: * Neighbor cache entries are stored in a hash table, which is indexed by triple
! 29: * (protocol, IP, requested-iface), so if both regular and iface-bound neighbors
! 30: * are requested, they are represented by two neighbor cache entries. Active
! 31: * entries are also linked in per-interface list (allowing quick processing of
! 32: * interface change events). Inactive entries exist only when the protocol has
! 33: * explicitly requested it via the %NEF_STICKY flag because it wishes to be
! 34: * notified when the node will again become a neighbor. Such entries are instead
! 35: * linked in a special list, which is walked whenever an interface changes its
! 36: * state to up. Neighbor entry VRF association is implied by respective
! 37: * protocol.
! 38: *
! 39: * Besides the already mentioned %NEF_STICKY flag, there is also %NEF_ONLINK,
! 40: * which specifies that neighbor should be considered reachable on given iface
! 41: * regardless of associated address ranges, and %NEF_IFACE, which represents
! 42: * pseudo-neighbor entry for whole interface (and uses %IPA_NONE IP address).
! 43: *
! 44: * When a neighbor event occurs (a neighbor gets disconnected or a sticky
! 45: * inactive neighbor becomes connected), the protocol hook neigh_notify() is
! 46: * called to advertise the change.
! 47: */
! 48:
! 49: #undef LOCAL_DEBUG
! 50:
! 51: #include "nest/bird.h"
! 52: #include "nest/iface.h"
! 53: #include "nest/protocol.h"
! 54: #include "lib/hash.h"
! 55: #include "lib/resource.h"
! 56:
! 57: #define NEIGH_HASH_SIZE 256
! 58: #define NEIGH_HASH_OFFSET 24
! 59:
! 60: static slab *neigh_slab;
! 61: static list neigh_hash_table[NEIGH_HASH_SIZE], sticky_neigh_list;
! 62:
! 63: static inline uint
! 64: neigh_hash(struct proto *p, ip_addr a, struct iface *i)
! 65: {
! 66: return (p->hash_key ^ ipa_hash(a) ^ ptr_hash(i)) >> NEIGH_HASH_OFFSET;
! 67: }
! 68:
! 69: static int
! 70: if_connected(ip_addr a, struct iface *i, struct ifa **ap, uint flags)
! 71: {
! 72: struct ifa *b;
! 73:
! 74: /* Handle iface pseudo-neighbors */
! 75: if (flags & NEF_IFACE)
! 76: return *ap = NULL, (i->flags & IF_UP) ? SCOPE_HOST : -1;
! 77:
! 78: /* Host addresses match even if iface is down */
! 79: WALK_LIST(b, i->addrs)
! 80: if (ipa_equal(a, b->ip))
! 81: return *ap = b, SCOPE_HOST;
! 82:
! 83: /* Rest do not match if iface is down */
! 84: if (!(i->flags & IF_UP))
! 85: return *ap = NULL, -1;
! 86:
! 87: /* Regular neighbors */
! 88: WALK_LIST(b, i->addrs)
! 89: {
! 90: if (b->flags & IA_PEER)
! 91: {
! 92: if (ipa_equal(a, b->opposite))
! 93: return *ap = b, b->scope;
! 94: }
! 95: else
! 96: {
! 97: if (ipa_in_netX(a, &b->prefix))
! 98: {
! 99: /* Do not allow IPv4 network and broadcast addresses */
! 100: if (ipa_is_ip4(a) &&
! 101: (net_pxlen(&b->prefix) < (IP4_MAX_PREFIX_LENGTH - 1)) &&
! 102: (ipa_equal(a, net_prefix(&b->prefix)) || /* Network address */
! 103: ipa_equal(a, b->brd))) /* Broadcast */
! 104: return *ap = NULL, -1;
! 105:
! 106: return *ap = b, b->scope;
! 107: }
! 108: }
! 109: }
! 110:
! 111: /* Handle ONLINK flag */
! 112: if (flags & NEF_ONLINK)
! 113: return *ap = NULL, ipa_classify(a) & IADDR_SCOPE_MASK;
! 114:
! 115: return *ap = NULL, -1;
! 116: }
! 117:
! 118: static inline int
! 119: if_connected_any(ip_addr a, struct iface *vrf, uint vrf_set, struct iface **iface, struct ifa **addr, uint flags)
! 120: {
! 121: struct iface *i;
! 122: struct ifa *b;
! 123: int s, scope = -1;
! 124:
! 125: *iface = NULL;
! 126: *addr = NULL;
! 127:
! 128: /* Get first match, but prefer SCOPE_HOST to other matches */
! 129: WALK_LIST(i, iface_list)
! 130: if ((!vrf_set || vrf == i->master) && ((s = if_connected(a, i, &b, flags)) >= 0))
! 131: if ((scope < 0) || ((scope > SCOPE_HOST) && (s == SCOPE_HOST)))
! 132: {
! 133: *iface = i;
! 134: *addr = b;
! 135: scope = s;
! 136: }
! 137:
! 138: return scope;
! 139: }
! 140:
! 141: /**
! 142: * neigh_find - find or create a neighbor entry
! 143: * @p: protocol which asks for the entry
! 144: * @a: IP address of the node to be searched for
! 145: * @iface: optionally bound neighbor to this iface (may be NULL)
! 146: * @flags: %NEF_STICKY for sticky entry, %NEF_ONLINK for onlink entry
! 147: *
! 148: * Search the neighbor cache for a node with given IP address. Iface can be
! 149: * specified for link-local addresses or for cases, where neighbor is expected
! 150: * on given interface. If it is found, a pointer to the neighbor entry is
! 151: * returned. If no such entry exists and the node is directly connected on one
! 152: * of our active interfaces, a new entry is created and returned to the caller
! 153: * with protocol-dependent fields initialized to zero. If the node is not
! 154: * connected directly or *@a is not a valid unicast IP address, neigh_find()
! 155: * returns %NULL.
! 156: */
! 157: neighbor *
! 158: neigh_find(struct proto *p, ip_addr a, struct iface *iface, uint flags)
! 159: {
! 160: neighbor *n;
! 161: int class, scope = -1;
! 162: uint h = neigh_hash(p, a, iface);
! 163: struct iface *ifreq = iface;
! 164: struct ifa *addr = NULL;
! 165:
! 166: WALK_LIST(n, neigh_hash_table[h]) /* Search the cache */
! 167: if ((n->proto == p) && ipa_equal(n->addr, a) && (n->ifreq == iface))
! 168: return n;
! 169:
! 170: if (flags & NEF_IFACE)
! 171: {
! 172: if (ipa_nonzero(a) || !iface)
! 173: return NULL;
! 174: }
! 175: else
! 176: {
! 177: class = ipa_classify(a);
! 178: if (class < 0) /* Invalid address */
! 179: return NULL;
! 180: if (((class & IADDR_SCOPE_MASK) == SCOPE_HOST) ||
! 181: (((class & IADDR_SCOPE_MASK) == SCOPE_LINK) && !iface) ||
! 182: !(class & IADDR_HOST))
! 183: return NULL; /* Bad scope or a somecast */
! 184: }
! 185:
! 186: if ((flags & NEF_ONLINK) && !iface)
! 187: return NULL;
! 188:
! 189: if (iface)
! 190: {
! 191: scope = if_connected(a, iface, &addr, flags);
! 192: iface = (scope < 0) ? NULL : iface;
! 193: }
! 194: else
! 195: scope = if_connected_any(a, p->vrf, p->vrf_set, &iface, &addr, flags);
! 196:
! 197: /* scope < 0 means i don't know neighbor */
! 198: /* scope >= 0 <=> iface != NULL */
! 199:
! 200: if ((scope < 0) && !(flags & NEF_STICKY))
! 201: return NULL;
! 202:
! 203: n = sl_alloc(neigh_slab);
! 204: memset(n, 0, sizeof(neighbor));
! 205:
! 206: add_tail(&neigh_hash_table[h], &n->n);
! 207: add_tail((scope >= 0) ? &iface->neighbors : &sticky_neigh_list, &n->if_n);
! 208: n->addr = a;
! 209: n->ifa = addr;
! 210: n->iface = iface;
! 211: n->ifreq = ifreq;
! 212: n->proto = p;
! 213: n->flags = flags;
! 214: n->scope = scope;
! 215:
! 216: return n;
! 217: }
! 218:
! 219: /**
! 220: * neigh_dump - dump specified neighbor entry.
! 221: * @n: the entry to dump
! 222: *
! 223: * This functions dumps the contents of a given neighbor entry to debug output.
! 224: */
! 225: void
! 226: neigh_dump(neighbor *n)
! 227: {
! 228: debug("%p %I %s %s ", n, n->addr,
! 229: n->iface ? n->iface->name : "[]",
! 230: n->ifreq ? n->ifreq->name : "[]");
! 231: debug("%s %p %08x scope %s", n->proto->name, n->data, n->aux, ip_scope_text(n->scope));
! 232: if (n->flags & NEF_STICKY)
! 233: debug(" STICKY");
! 234: if (n->flags & NEF_ONLINK)
! 235: debug(" ONLINK");
! 236: debug("\n");
! 237: }
! 238:
! 239: /**
! 240: * neigh_dump_all - dump all neighbor entries.
! 241: *
! 242: * This function dumps the contents of the neighbor cache to debug output.
! 243: */
! 244: void
! 245: neigh_dump_all(void)
! 246: {
! 247: neighbor *n;
! 248: int i;
! 249:
! 250: debug("Known neighbors:\n");
! 251: for(i=0; i<NEIGH_HASH_SIZE; i++)
! 252: WALK_LIST(n, neigh_hash_table[i])
! 253: neigh_dump(n);
! 254: debug("\n");
! 255: }
! 256:
! 257: static inline void
! 258: neigh_notify(neighbor *n)
! 259: {
! 260: if (n->proto->neigh_notify && (n->proto->proto_state != PS_STOP))
! 261: n->proto->neigh_notify(n);
! 262: }
! 263:
! 264: static void
! 265: neigh_up(neighbor *n, struct iface *i, struct ifa *a, int scope)
! 266: {
! 267: DBG("Waking up sticky neighbor %I\n", n->addr);
! 268: n->iface = i;
! 269: n->ifa = a;
! 270: n->scope = scope;
! 271:
! 272: rem_node(&n->if_n);
! 273: add_tail(&i->neighbors, &n->if_n);
! 274:
! 275: neigh_notify(n);
! 276: }
! 277:
! 278: static void
! 279: neigh_down(neighbor *n)
! 280: {
! 281: DBG("Flushing neighbor %I on %s\n", n->addr, n->iface->name);
! 282: n->iface = NULL;
! 283: n->ifa = NULL;
! 284: n->scope = -1;
! 285:
! 286: rem_node(&n->if_n);
! 287: add_tail(&sticky_neigh_list, &n->if_n);
! 288:
! 289: neigh_notify(n);
! 290: }
! 291:
! 292: static inline void
! 293: neigh_free(neighbor *n)
! 294: {
! 295: rem_node(&n->n);
! 296: rem_node(&n->if_n);
! 297: sl_free(neigh_slab, n);
! 298: }
! 299:
! 300: /**
! 301: * neigh_update: update neighbor entry w.r.t. change on specific iface
! 302: * @n: neighbor to update
! 303: * @iface: changed iface
! 304: *
! 305: * The function recalculates state of the neighbor entry @n assuming that only
! 306: * the interface @iface may changed its state or addresses. Then, appropriate
! 307: * actions are executed (the neighbor goes up, down, up-down, or just notified).
! 308: */
! 309: void
! 310: neigh_update(neighbor *n, struct iface *iface)
! 311: {
! 312: struct proto *p = n->proto;
! 313: struct ifa *ifa = NULL;
! 314: int scope = -1;
! 315:
! 316: /* Iface-bound neighbors ignore other ifaces */
! 317: if (n->ifreq && (n->ifreq != iface))
! 318: return;
! 319:
! 320: /* VRF-bound neighbors ignore changes in other VRFs */
! 321: if (p->vrf_set && (p->vrf != iface->master))
! 322: return;
! 323:
! 324: scope = if_connected(n->addr, iface, &ifa, n->flags);
! 325:
! 326: /* When neighbor is going down, try to respawn it on other ifaces */
! 327: if ((scope < 0) && (n->scope >= 0) && !n->ifreq && (n->flags & NEF_STICKY))
! 328: scope = if_connected_any(n->addr, p->vrf, p->vrf_set, &iface, &ifa, n->flags);
! 329:
! 330: /* No change or minor change - ignore or notify */
! 331: if ((scope == n->scope) && (iface == n->iface))
! 332: {
! 333: if (ifa != n->ifa)
! 334: {
! 335: n->ifa = ifa;
! 336: neigh_notify(n);
! 337: }
! 338:
! 339: return;
! 340: }
! 341:
! 342: /* Major change - going down and/or going up */
! 343:
! 344: if (n->scope >= 0)
! 345: neigh_down(n);
! 346:
! 347: if ((n->scope < 0) && !(n->flags & NEF_STICKY))
! 348: {
! 349: neigh_free(n);
! 350: return;
! 351: }
! 352:
! 353: if (scope >= 0)
! 354: neigh_up(n, iface, ifa, scope);
! 355: }
! 356:
! 357:
! 358: /**
! 359: * neigh_if_up: notify neighbor cache about interface up event
! 360: * @i: interface in question
! 361: *
! 362: * Tell the neighbor cache that a new interface became up.
! 363: *
! 364: * The neighbor cache wakes up all inactive sticky neighbors with
! 365: * addresses belonging to prefixes of the interface @i.
! 366: */
! 367: void
! 368: neigh_if_up(struct iface *i)
! 369: {
! 370: neighbor *n;
! 371: node *x, *y;
! 372:
! 373: WALK_LIST2_DELSAFE(n, x, y, sticky_neigh_list, if_n)
! 374: neigh_update(n, i);
! 375: }
! 376:
! 377: /**
! 378: * neigh_if_down - notify neighbor cache about interface down event
! 379: * @i: the interface in question
! 380: *
! 381: * Notify the neighbor cache that an interface has ceased to exist.
! 382: *
! 383: * It causes all neighbors connected to this interface to be updated or removed.
! 384: */
! 385: void
! 386: neigh_if_down(struct iface *i)
! 387: {
! 388: neighbor *n;
! 389: node *x, *y;
! 390:
! 391: WALK_LIST2_DELSAFE(n, x, y, i->neighbors, if_n)
! 392: neigh_update(n, i);
! 393: }
! 394:
! 395: /**
! 396: * neigh_if_link - notify neighbor cache about interface link change
! 397: * @i: the interface in question
! 398: *
! 399: * Notify the neighbor cache that an interface changed link state. All owners of
! 400: * neighbor entries connected to this interface are notified.
! 401: */
! 402: void
! 403: neigh_if_link(struct iface *i)
! 404: {
! 405: neighbor *n;
! 406: node *x, *y;
! 407:
! 408: WALK_LIST2_DELSAFE(n, x, y, i->neighbors, if_n)
! 409: neigh_notify(n);
! 410: }
! 411:
! 412: /**
! 413: * neigh_ifa_update: notify neighbor cache about interface address add or remove event
! 414: * @a: interface address in question
! 415: *
! 416: * Tell the neighbor cache that an address was added or removed.
! 417: *
! 418: * The neighbor cache wakes up all inactive sticky neighbors with
! 419: * addresses belonging to prefixes of the interface belonging to @ifa
! 420: * and causes all unreachable neighbors to be flushed.
! 421: */
! 422: void
! 423: neigh_ifa_update(struct ifa *a)
! 424: {
! 425: struct iface *i = a->iface;
! 426: neighbor *n;
! 427: node *x, *y;
! 428:
! 429: /* Update all neighbors whose scope has changed */
! 430: WALK_LIST2_DELSAFE(n, x, y, i->neighbors, if_n)
! 431: neigh_update(n, i);
! 432:
! 433: /* Wake up all sticky neighbors that are reachable now */
! 434: WALK_LIST2_DELSAFE(n, x, y, sticky_neigh_list, if_n)
! 435: neigh_update(n, i);
! 436: }
! 437:
! 438: static inline void
! 439: neigh_prune_one(neighbor *n)
! 440: {
! 441: if (n->proto->proto_state != PS_DOWN)
! 442: return;
! 443:
! 444: neigh_free(n);
! 445: }
! 446:
! 447: /**
! 448: * neigh_prune - prune neighbor cache
! 449: *
! 450: * neigh_prune() examines all neighbor entries cached and removes those
! 451: * corresponding to inactive protocols. It's called whenever a protocol
! 452: * is shut down to get rid of all its heritage.
! 453: */
! 454: void
! 455: neigh_prune(void)
! 456: {
! 457: neighbor *n;
! 458: node *m;
! 459: int i;
! 460:
! 461: DBG("Pruning neighbors\n");
! 462: for(i=0; i<NEIGH_HASH_SIZE; i++)
! 463: WALK_LIST_DELSAFE(n, m, neigh_hash_table[i])
! 464: neigh_prune_one(n);
! 465: }
! 466:
! 467: /**
! 468: * neigh_init - initialize the neighbor cache.
! 469: * @if_pool: resource pool to be used for neighbor entries.
! 470: *
! 471: * This function is called during BIRD startup to initialize
! 472: * the neighbor cache module.
! 473: */
! 474: void
! 475: neigh_init(pool *if_pool)
! 476: {
! 477: neigh_slab = sl_new(if_pool, sizeof(neighbor));
! 478:
! 479: for(int i = 0; i < NEIGH_HASH_SIZE; i++)
! 480: init_list(&neigh_hash_table[i]);
! 481:
! 482: init_list(&sticky_neigh_list);
! 483: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>