Annotation of embedaddon/bird2/proto/static/static.c, revision 1.1
1.1 ! misho 1: /*
! 2: * BIRD -- Static Route Generator
! 3: *
! 4: * (c) 1998--2000 Martin Mares <mj@ucw.cz>
! 5: *
! 6: * Can be freely distributed and used under the terms of the GNU GPL.
! 7: */
! 8:
! 9: /**
! 10: * DOC: Static
! 11: *
! 12: * The Static protocol is implemented in a straightforward way. It keeps a list
! 13: * of static routes. Routes of dest RTD_UNICAST have associated sticky node in
! 14: * the neighbor cache to be notified about gaining or losing the neighbor and
! 15: * about interface-related events (e.g. link down). They may also have a BFD
! 16: * request if associated with a BFD session. When a route is notified,
! 17: * static_decide() is used to see whether the route activeness is changed. In
! 18: * such case, the route is marked as dirty and scheduled to be announced or
! 19: * withdrawn, which is done asynchronously from event hook. Routes of other
! 20: * types (e.g. black holes) are announced all the time.
! 21: *
! 22: * Multipath routes are a bit tricky. To represent additional next hops, dummy
! 23: * static_route nodes are used, which are chained using @mp_next field and link
! 24: * to the master node by @mp_head field. Each next hop has a separate neighbor
! 25: * entry and an activeness state, but the master node is used for most purposes.
! 26: * Note that most functions DO NOT accept dummy nodes as arguments.
! 27: *
! 28: * The only other thing worth mentioning is that when asked for reconfiguration,
! 29: * Static not only compares the two configurations, but it also calculates
! 30: * difference between the lists of static routes and it just inserts the newly
! 31: * added routes, removes the obsolete ones and reannounces changed ones.
! 32: */
! 33:
! 34: #undef LOCAL_DEBUG
! 35:
! 36: #include <stdlib.h>
! 37:
! 38: #include "nest/bird.h"
! 39: #include "nest/iface.h"
! 40: #include "nest/protocol.h"
! 41: #include "nest/route.h"
! 42: #include "nest/cli.h"
! 43: #include "conf/conf.h"
! 44: #include "filter/filter.h"
! 45: #include "lib/string.h"
! 46: #include "lib/alloca.h"
! 47:
! 48: #include "static.h"
! 49:
! 50: static linpool *static_lp;
! 51:
! 52: static void
! 53: static_announce_rte(struct static_proto *p, struct static_route *r)
! 54: {
! 55: rta *a = allocz(RTA_MAX_SIZE);
! 56: a->src = p->p.main_source;
! 57: a->source = RTS_STATIC;
! 58: a->scope = SCOPE_UNIVERSE;
! 59: a->dest = r->dest;
! 60:
! 61: if (r->dest == RTD_UNICAST)
! 62: {
! 63: struct static_route *r2;
! 64: struct nexthop *nhs = NULL;
! 65:
! 66: for (r2 = r; r2; r2 = r2->mp_next)
! 67: {
! 68: if (!r2->active)
! 69: continue;
! 70:
! 71: struct nexthop *nh = allocz(NEXTHOP_MAX_SIZE);
! 72: nh->gw = r2->via;
! 73: nh->iface = r2->neigh->iface;
! 74: nh->flags = r2->onlink ? RNF_ONLINK : 0;
! 75: nh->weight = r2->weight;
! 76: if (r2->mls)
! 77: {
! 78: nh->labels = r2->mls->len;
! 79: memcpy(nh->label, r2->mls->stack, r2->mls->len * sizeof(u32));
! 80: }
! 81:
! 82: nexthop_insert(&nhs, nh);
! 83: }
! 84:
! 85: if (!nhs)
! 86: goto withdraw;
! 87:
! 88: nexthop_link(a, nhs);
! 89: }
! 90:
! 91: if (r->dest == RTDX_RECURSIVE)
! 92: {
! 93: rtable *tab = ipa_is_ip4(r->via) ? p->igp_table_ip4 : p->igp_table_ip6;
! 94: rta_set_recursive_next_hop(p->p.main_channel->table, a, tab, r->via, IPA_NONE, r->mls);
! 95: }
! 96:
! 97: /* Already announced */
! 98: if (r->state == SRS_CLEAN)
! 99: return;
! 100:
! 101: /* We skip rta_lookup() here */
! 102: rte *e = rte_get_temp(a);
! 103: e->pflags = 0;
! 104:
! 105: if (r->cmds)
! 106: f_eval_rte(r->cmds, &e, static_lp);
! 107:
! 108: rte_update(&p->p, r->net, e);
! 109: r->state = SRS_CLEAN;
! 110:
! 111: if (r->cmds)
! 112: lp_flush(static_lp);
! 113:
! 114: return;
! 115:
! 116: withdraw:
! 117: if (r->state == SRS_DOWN)
! 118: return;
! 119:
! 120: rte_update(&p->p, r->net, NULL);
! 121: r->state = SRS_DOWN;
! 122: }
! 123:
! 124: static void
! 125: static_mark_rte(struct static_proto *p, struct static_route *r)
! 126: {
! 127: if (r->state == SRS_DIRTY)
! 128: return;
! 129:
! 130: r->state = SRS_DIRTY;
! 131: BUFFER_PUSH(p->marked) = r;
! 132:
! 133: if (!ev_active(p->event))
! 134: ev_schedule(p->event);
! 135: }
! 136:
! 137: static void
! 138: static_announce_marked(void *P)
! 139: {
! 140: struct static_proto *p = P;
! 141:
! 142: BUFFER_WALK(p->marked, r)
! 143: static_announce_rte(P, r);
! 144:
! 145: BUFFER_FLUSH(p->marked);
! 146: }
! 147:
! 148: static void
! 149: static_bfd_notify(struct bfd_request *req);
! 150:
! 151: static void
! 152: static_update_bfd(struct static_proto *p, struct static_route *r)
! 153: {
! 154: /* The @r is a RTD_UNICAST next hop, may be a dummy node */
! 155:
! 156: struct neighbor *nb = r->neigh;
! 157: int bfd_up = (nb->scope > 0) && r->use_bfd;
! 158:
! 159: if (bfd_up && !r->bfd_req)
! 160: {
! 161: // ip_addr local = ipa_nonzero(r->local) ? r->local : nb->ifa->ip;
! 162: r->bfd_req = bfd_request_session(p->p.pool, r->via, nb->ifa->ip,
! 163: nb->iface, p->p.vrf,
! 164: static_bfd_notify, r);
! 165: }
! 166:
! 167: if (!bfd_up && r->bfd_req)
! 168: {
! 169: rfree(r->bfd_req);
! 170: r->bfd_req = NULL;
! 171: }
! 172: }
! 173:
! 174: static int
! 175: static_decide(struct static_proto *p, struct static_route *r)
! 176: {
! 177: /* The @r is a RTD_UNICAST next hop, may be a dummy node */
! 178:
! 179: struct static_config *cf = (void *) p->p.cf;
! 180: uint old_active = r->active;
! 181:
! 182: if (r->neigh->scope < 0)
! 183: goto fail;
! 184:
! 185: if (cf->check_link && !(r->neigh->iface->flags & IF_LINK_UP))
! 186: goto fail;
! 187:
! 188: if (r->bfd_req && (r->bfd_req->state != BFD_STATE_UP))
! 189: goto fail;
! 190:
! 191: r->active = 1;
! 192: return !old_active;
! 193:
! 194: fail:
! 195: r->active = 0;
! 196: return old_active;
! 197: }
! 198:
! 199: static void
! 200: static_add_rte(struct static_proto *p, struct static_route *r)
! 201: {
! 202: if (r->dest == RTD_UNICAST)
! 203: {
! 204: struct static_route *r2;
! 205: struct neighbor *n;
! 206:
! 207: for (r2 = r; r2; r2 = r2->mp_next)
! 208: {
! 209: n = neigh_find(&p->p, r2->via, r2->iface, NEF_STICKY |
! 210: (r2->onlink ? NEF_ONLINK : 0) |
! 211: (ipa_zero(r2->via) ? NEF_IFACE : 0));
! 212:
! 213: if (!n)
! 214: {
! 215: log(L_WARN "Invalid next hop %I of static route %N", r2->via, r2->net);
! 216: continue;
! 217: }
! 218:
! 219: r2->neigh = n;
! 220: r2->chain = n->data;
! 221: n->data = r2;
! 222:
! 223: static_update_bfd(p, r2);
! 224: static_decide(p, r2);
! 225: }
! 226: }
! 227:
! 228: static_announce_rte(p, r);
! 229: }
! 230:
! 231: static void
! 232: static_reset_rte(struct static_proto *p UNUSED, struct static_route *r)
! 233: {
! 234: struct static_route *r2;
! 235:
! 236: for (r2 = r; r2; r2 = r2->mp_next)
! 237: {
! 238: r2->neigh = NULL;
! 239: r2->chain = NULL;
! 240:
! 241: r2->state = 0;
! 242: r2->active = 0;
! 243:
! 244: rfree(r2->bfd_req);
! 245: r2->bfd_req = NULL;
! 246: }
! 247: }
! 248:
! 249: static void
! 250: static_remove_rte(struct static_proto *p, struct static_route *r)
! 251: {
! 252: if (r->state)
! 253: rte_update(&p->p, r->net, NULL);
! 254:
! 255: static_reset_rte(p, r);
! 256: }
! 257:
! 258:
! 259: static inline int
! 260: static_same_dest(struct static_route *x, struct static_route *y)
! 261: {
! 262: if (x->dest != y->dest)
! 263: return 0;
! 264:
! 265: switch (x->dest)
! 266: {
! 267: case RTD_UNICAST:
! 268: for (; x && y; x = x->mp_next, y = y->mp_next)
! 269: {
! 270: if (!ipa_equal(x->via, y->via) ||
! 271: (x->iface != y->iface) ||
! 272: (x->onlink != y->onlink) ||
! 273: (x->weight != y->weight) ||
! 274: (x->use_bfd != y->use_bfd) ||
! 275: (!x->mls != !y->mls) ||
! 276: ((x->mls) && (y->mls) && (x->mls->len != y->mls->len)))
! 277: return 0;
! 278:
! 279: if (!x->mls)
! 280: continue;
! 281:
! 282: for (uint i = 0; i < x->mls->len; i++)
! 283: if (x->mls->stack[i] != y->mls->stack[i])
! 284: return 0;
! 285: }
! 286: return !x && !y;
! 287:
! 288: case RTDX_RECURSIVE:
! 289: if (!ipa_equal(x->via, y->via) ||
! 290: (!x->mls != !y->mls) ||
! 291: ((x->mls) && (y->mls) && (x->mls->len != y->mls->len)))
! 292: return 0;
! 293:
! 294: if (!x->mls)
! 295: return 1;
! 296:
! 297: for (uint i = 0; i < x->mls->len; i++)
! 298: if (x->mls->stack[i] != y->mls->stack[i])
! 299: return 0;
! 300:
! 301: return 1;
! 302:
! 303: default:
! 304: return 1;
! 305: }
! 306: }
! 307:
! 308: static inline int
! 309: static_same_rte(struct static_route *or, struct static_route *nr)
! 310: {
! 311: /* Note that i_same() requires arguments in (new, old) order */
! 312: return static_same_dest(or, nr) && f_same(nr->cmds, or->cmds);
! 313: }
! 314:
! 315: static void
! 316: static_reconfigure_rte(struct static_proto *p, struct static_route *or, struct static_route *nr)
! 317: {
! 318: if ((or->state == SRS_CLEAN) && !static_same_rte(or, nr))
! 319: nr->state = SRS_DIRTY;
! 320: else
! 321: nr->state = or->state;
! 322:
! 323: static_add_rte(p, nr);
! 324: static_reset_rte(p, or);
! 325: }
! 326:
! 327:
! 328: static void
! 329: static_neigh_notify(struct neighbor *n)
! 330: {
! 331: struct static_proto *p = (void *) n->proto;
! 332: struct static_route *r;
! 333:
! 334: DBG("Static: neighbor notify for %I: iface %p\n", n->addr, n->iface);
! 335: for (r = n->data; r; r = r->chain)
! 336: {
! 337: static_update_bfd(p, r);
! 338:
! 339: if (static_decide(p, r))
! 340: static_mark_rte(p, r->mp_head);
! 341: }
! 342: }
! 343:
! 344: static void
! 345: static_bfd_notify(struct bfd_request *req)
! 346: {
! 347: struct static_route *r = req->data;
! 348: struct static_proto *p = (void *) r->neigh->proto;
! 349:
! 350: // if (req->down) TRACE(D_EVENTS, "BFD session down for nbr %I on %s", XXXX);
! 351:
! 352: if (static_decide(p, r))
! 353: static_mark_rte(p, r->mp_head);
! 354: }
! 355:
! 356: static int
! 357: static_rte_mergable(rte *pri UNUSED, rte *sec UNUSED)
! 358: {
! 359: return 1;
! 360: }
! 361:
! 362:
! 363: static void
! 364: static_postconfig(struct proto_config *CF)
! 365: {
! 366: struct static_config *cf = (void *) CF;
! 367: struct static_route *r;
! 368:
! 369: if (EMPTY_LIST(CF->channels))
! 370: cf_error("Channel not specified");
! 371:
! 372: struct channel_config *cc = proto_cf_main_channel(CF);
! 373:
! 374: if (!cf->igp_table_ip4)
! 375: cf->igp_table_ip4 = (cc->table->addr_type == NET_IP4) ?
! 376: cc->table : cf->c.global->def_tables[NET_IP4];
! 377:
! 378: if (!cf->igp_table_ip6)
! 379: cf->igp_table_ip6 = (cc->table->addr_type == NET_IP6) ?
! 380: cc->table : cf->c.global->def_tables[NET_IP6];
! 381:
! 382: WALK_LIST(r, cf->routes)
! 383: if (r->net && (r->net->type != CF->net_type))
! 384: cf_error("Route %N incompatible with channel type", r->net);
! 385: }
! 386:
! 387: static struct proto *
! 388: static_init(struct proto_config *CF)
! 389: {
! 390: struct proto *P = proto_new(CF);
! 391: struct static_proto *p = (void *) P;
! 392: struct static_config *cf = (void *) CF;
! 393:
! 394: P->main_channel = proto_add_channel(P, proto_cf_main_channel(CF));
! 395:
! 396: P->neigh_notify = static_neigh_notify;
! 397: P->rte_mergable = static_rte_mergable;
! 398:
! 399: if (cf->igp_table_ip4)
! 400: p->igp_table_ip4 = cf->igp_table_ip4->table;
! 401:
! 402: if (cf->igp_table_ip6)
! 403: p->igp_table_ip6 = cf->igp_table_ip6->table;
! 404:
! 405: return P;
! 406: }
! 407:
! 408: static int
! 409: static_start(struct proto *P)
! 410: {
! 411: struct static_proto *p = (void *) P;
! 412: struct static_config *cf = (void *) P->cf;
! 413: struct static_route *r;
! 414:
! 415: if (!static_lp)
! 416: static_lp = lp_new(&root_pool, LP_GOOD_SIZE(1024));
! 417:
! 418: if (p->igp_table_ip4)
! 419: rt_lock_table(p->igp_table_ip4);
! 420:
! 421: if (p->igp_table_ip6)
! 422: rt_lock_table(p->igp_table_ip6);
! 423:
! 424: p->event = ev_new_init(p->p.pool, static_announce_marked, p);
! 425:
! 426: BUFFER_INIT(p->marked, p->p.pool, 4);
! 427:
! 428: /* We have to go UP before routes could be installed */
! 429: proto_notify_state(P, PS_UP);
! 430:
! 431: WALK_LIST(r, cf->routes)
! 432: static_add_rte(p, r);
! 433:
! 434: return PS_UP;
! 435: }
! 436:
! 437: static int
! 438: static_shutdown(struct proto *P)
! 439: {
! 440: struct static_proto *p = (void *) P;
! 441: struct static_config *cf = (void *) P->cf;
! 442: struct static_route *r;
! 443:
! 444: /* Just reset the flag, the routes will be flushed by the nest */
! 445: WALK_LIST(r, cf->routes)
! 446: static_reset_rte(p, r);
! 447:
! 448: return PS_DOWN;
! 449: }
! 450:
! 451: static void
! 452: static_cleanup(struct proto *P)
! 453: {
! 454: struct static_proto *p = (void *) P;
! 455:
! 456: if (p->igp_table_ip4)
! 457: rt_unlock_table(p->igp_table_ip4);
! 458:
! 459: if (p->igp_table_ip6)
! 460: rt_unlock_table(p->igp_table_ip6);
! 461: }
! 462:
! 463: static void
! 464: static_dump_rte(struct static_route *r)
! 465: {
! 466: debug("%-1N: ", r->net);
! 467: if (r->dest == RTD_UNICAST)
! 468: if (r->iface && ipa_zero(r->via))
! 469: debug("dev %s\n", r->iface->name);
! 470: else
! 471: debug("via %I%J\n", r->via, r->iface);
! 472: else
! 473: debug("rtd %d\n", r->dest);
! 474: }
! 475:
! 476: static void
! 477: static_dump(struct proto *P)
! 478: {
! 479: struct static_config *c = (void *) P->cf;
! 480: struct static_route *r;
! 481:
! 482: debug("Static routes:\n");
! 483: WALK_LIST(r, c->routes)
! 484: static_dump_rte(r);
! 485: }
! 486:
! 487: #define IGP_TABLE(cf, sym) ((cf)->igp_table_##sym ? (cf)->igp_table_##sym ->table : NULL )
! 488:
! 489: static inline int
! 490: static_cmp_rte(const void *X, const void *Y)
! 491: {
! 492: struct static_route *x = *(void **)X, *y = *(void **)Y;
! 493: return net_compare(x->net, y->net);
! 494: }
! 495:
! 496: static int
! 497: static_reconfigure(struct proto *P, struct proto_config *CF)
! 498: {
! 499: struct static_proto *p = (void *) P;
! 500: struct static_config *o = (void *) P->cf;
! 501: struct static_config *n = (void *) CF;
! 502: struct static_route *r, *r2, *or, *nr;
! 503:
! 504: /* Check change in IGP tables */
! 505: if ((IGP_TABLE(o, ip4) != IGP_TABLE(n, ip4)) ||
! 506: (IGP_TABLE(o, ip6) != IGP_TABLE(n, ip6)))
! 507: return 0;
! 508:
! 509: if (!proto_configure_channel(P, &P->main_channel, proto_cf_main_channel(CF)))
! 510: return 0;
! 511:
! 512: p->p.cf = CF;
! 513:
! 514: /* Reset route lists in neighbor entries */
! 515: WALK_LIST(r, o->routes)
! 516: for (r2 = r; r2; r2 = r2->mp_next)
! 517: if (r2->neigh)
! 518: r2->neigh->data = NULL;
! 519:
! 520: /* Reconfigure initial matching sequence */
! 521: for (or = HEAD(o->routes), nr = HEAD(n->routes);
! 522: NODE_VALID(or) && NODE_VALID(nr) && net_equal(or->net, nr->net);
! 523: or = NODE_NEXT(or), nr = NODE_NEXT(nr))
! 524: static_reconfigure_rte(p, or, nr);
! 525:
! 526: if (!NODE_VALID(or) && !NODE_VALID(nr))
! 527: return 1;
! 528:
! 529: /* Reconfigure remaining routes, sort them to find matching pairs */
! 530: struct static_route *or2, *nr2, **orbuf, **nrbuf;
! 531: uint ornum = 0, nrnum = 0, orpos = 0, nrpos = 0, i;
! 532:
! 533: for (or2 = or; NODE_VALID(or2); or2 = NODE_NEXT(or2))
! 534: ornum++;
! 535:
! 536: for (nr2 = nr; NODE_VALID(nr2); nr2 = NODE_NEXT(nr2))
! 537: nrnum++;
! 538:
! 539: orbuf = xmalloc(ornum * sizeof(void *));
! 540: nrbuf = xmalloc(nrnum * sizeof(void *));
! 541:
! 542: for (i = 0, or2 = or; i < ornum; i++, or2 = NODE_NEXT(or2))
! 543: orbuf[i] = or2;
! 544:
! 545: for (i = 0, nr2 = nr; i < nrnum; i++, nr2 = NODE_NEXT(nr2))
! 546: nrbuf[i] = nr2;
! 547:
! 548: qsort(orbuf, ornum, sizeof(struct static_route *), static_cmp_rte);
! 549: qsort(nrbuf, nrnum, sizeof(struct static_route *), static_cmp_rte);
! 550:
! 551: while ((orpos < ornum) && (nrpos < nrnum))
! 552: {
! 553: int x = net_compare(orbuf[orpos]->net, nrbuf[nrpos]->net);
! 554: if (x < 0)
! 555: static_remove_rte(p, orbuf[orpos++]);
! 556: else if (x > 0)
! 557: static_add_rte(p, nrbuf[nrpos++]);
! 558: else
! 559: static_reconfigure_rte(p, orbuf[orpos++], nrbuf[nrpos++]);
! 560: }
! 561:
! 562: while (orpos < ornum)
! 563: static_remove_rte(p, orbuf[orpos++]);
! 564:
! 565: while (nrpos < nrnum)
! 566: static_add_rte(p, nrbuf[nrpos++]);
! 567:
! 568: xfree(orbuf);
! 569: xfree(nrbuf);
! 570:
! 571: return 1;
! 572: }
! 573:
! 574: static void
! 575: static_copy_config(struct proto_config *dest, struct proto_config *src)
! 576: {
! 577: struct static_config *d = (struct static_config *) dest;
! 578: struct static_config *s = (struct static_config *) src;
! 579:
! 580: struct static_route *srt, *snh;
! 581:
! 582: /* Copy route list */
! 583: init_list(&d->routes);
! 584: WALK_LIST(srt, s->routes)
! 585: {
! 586: struct static_route *drt = NULL, *dnh = NULL, **dnp = &drt;
! 587:
! 588: for (snh = srt; snh; snh = snh->mp_next)
! 589: {
! 590: dnh = cfg_alloc(sizeof(struct static_route));
! 591: memcpy(dnh, snh, sizeof(struct static_route));
! 592:
! 593: if (!drt)
! 594: add_tail(&d->routes, &(dnh->n));
! 595:
! 596: *dnp = dnh;
! 597: dnp = &(dnh->mp_next);
! 598:
! 599: if (snh->mp_head)
! 600: dnh->mp_head = drt;
! 601: }
! 602: }
! 603: }
! 604:
! 605: static void
! 606: static_show_rt(struct static_route *r)
! 607: {
! 608: switch (r->dest)
! 609: {
! 610: case RTD_UNICAST:
! 611: {
! 612: struct static_route *r2;
! 613:
! 614: cli_msg(-1009, "%N", r->net);
! 615: for (r2 = r; r2; r2 = r2->mp_next)
! 616: {
! 617: if (r2->iface && ipa_zero(r2->via))
! 618: cli_msg(-1009, "\tdev %s%s", r2->iface->name,
! 619: r2->active ? "" : " (dormant)");
! 620: else
! 621: cli_msg(-1009, "\tvia %I%J%s%s%s", r2->via, r2->iface,
! 622: r2->onlink ? " onlink" : "",
! 623: r2->bfd_req ? " (bfd)" : "",
! 624: r2->active ? "" : " (dormant)");
! 625: }
! 626: break;
! 627: }
! 628:
! 629: case RTD_NONE:
! 630: case RTD_BLACKHOLE:
! 631: case RTD_UNREACHABLE:
! 632: case RTD_PROHIBIT:
! 633: cli_msg(-1009, "%N\t%s", r->net, rta_dest_names[r->dest]);
! 634: break;
! 635:
! 636: case RTDX_RECURSIVE:
! 637: cli_msg(-1009, "%N\trecursive %I", r->net, r->via);
! 638: break;
! 639: }
! 640: }
! 641:
! 642: void
! 643: static_show(struct proto *P)
! 644: {
! 645: struct static_config *c = (void *) P->cf;
! 646: struct static_route *r;
! 647:
! 648: WALK_LIST(r, c->routes)
! 649: static_show_rt(r);
! 650: cli_msg(0, "");
! 651: }
! 652:
! 653:
! 654: struct protocol proto_static = {
! 655: .name = "Static",
! 656: .template = "static%d",
! 657: .class = PROTOCOL_STATIC,
! 658: .preference = DEF_PREF_STATIC,
! 659: .channel_mask = NB_ANY,
! 660: .proto_size = sizeof(struct static_proto),
! 661: .config_size = sizeof(struct static_config),
! 662: .postconfig = static_postconfig,
! 663: .init = static_init,
! 664: .dump = static_dump,
! 665: .start = static_start,
! 666: .shutdown = static_shutdown,
! 667: .cleanup = static_cleanup,
! 668: .reconfigure = static_reconfigure,
! 669: .copy_config = static_copy_config
! 670: };
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>