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>