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>