Annotation of embedaddon/bird2/proto/ospf/iface.c, revision 1.1

1.1     ! misho       1: /*
        !             2:  *     BIRD -- OSPF
        !             3:  *
        !             4:  *     (c) 1999--2005 Ondrej Filip <feela@network.cz>
        !             5:  *     (c) 2009--2014 Ondrej Zajicek <santiago@crfreenet.org>
        !             6:  *     (c) 2009--2014 CZ.NIC z.s.p.o.
        !             7:  *
        !             8:  *     Can be freely distributed and used under the terms of the GNU GPL.
        !             9:  */
        !            10: 
        !            11: #include "ospf.h"
        !            12: #include "nest/password.h"
        !            13: 
        !            14: 
        !            15: const char *ospf_is_names[] = {
        !            16:   "Down", "Loopback", "Waiting", "PtP", "DROther", "Backup", "DR"
        !            17: };
        !            18: 
        !            19: const char *ospf_ism_names[] = {
        !            20:   "InterfaceUp", "WaitTimer", "BackupSeen", "NeighborChange",
        !            21:   "LoopInd", "UnloopInd", "InterfaceDown"
        !            22: };
        !            23: 
        !            24: const char *ospf_it[] = { "broadcast", "nbma", "ptp", "ptmp", "virtual link" };
        !            25: 
        !            26: 
        !            27: static void
        !            28: poll_timer_hook(timer * timer)
        !            29: {
        !            30:   ospf_send_hello(timer->data, OHS_POLL, NULL);
        !            31: }
        !            32: 
        !            33: static void
        !            34: hello_timer_hook(timer * timer)
        !            35: {
        !            36:   ospf_send_hello(timer->data, OHS_HELLO, NULL);
        !            37: }
        !            38: 
        !            39: static void
        !            40: wait_timer_hook(timer * timer)
        !            41: {
        !            42:   struct ospf_iface *ifa = (struct ospf_iface *) timer->data;
        !            43:   struct ospf_proto *p = ifa->oa->po;
        !            44: 
        !            45:   OSPF_TRACE(D_EVENTS, "Wait timer fired on %s", ifa->ifname);
        !            46:   ospf_iface_sm(ifa, ISM_WAITF);
        !            47: }
        !            48: 
        !            49: static inline uint
        !            50: ifa_tx_length(struct ospf_iface *ifa)
        !            51: {
        !            52:   return ifa->cf->tx_length ?: ifa->iface->mtu;
        !            53: }
        !            54: 
        !            55: static inline uint
        !            56: ifa_tx_hdrlen(struct ospf_iface *ifa)
        !            57: {
        !            58:   struct ospf_proto *p = ifa->oa->po;
        !            59: 
        !            60:   uint hlen = ospf_is_v2(p) ? IP4_HEADER_LENGTH : IP6_HEADER_LENGTH;
        !            61: 
        !            62:   /* Relevant just for OSPFv2 */
        !            63:   if (ifa->autype == OSPF_AUTH_CRYPT)
        !            64:   {
        !            65:     hlen += ospf_is_v2(p) ? 0 : sizeof(struct ospf_auth3);
        !            66:     hlen += max_mac_length(ifa->passwords);
        !            67:   }
        !            68: 
        !            69:   return hlen;
        !            70: }
        !            71: 
        !            72: static inline uint
        !            73: ifa_bufsize(struct ospf_iface *ifa)
        !            74: {
        !            75:   uint bsize = ifa->cf->rx_buffer ?: ifa->iface->mtu;
        !            76:   return MAX(bsize, ifa->tx_length);
        !            77: }
        !            78: 
        !            79: static inline uint
        !            80: ifa_flood_queue_size(struct ospf_iface *ifa)
        !            81: {
        !            82:   return ifa->tx_length / 24;
        !            83: }
        !            84: 
        !            85: int
        !            86: ospf_iface_assure_bufsize(struct ospf_iface *ifa, uint plen)
        !            87: {
        !            88:   plen += ifa->tx_hdrlen;
        !            89: 
        !            90:   if (plen <= ifa->sk->tbsize)
        !            91:     return 0;
        !            92: 
        !            93:   if (ifa->cf->rx_buffer || (plen > 0xffff))
        !            94:     return -1;
        !            95: 
        !            96:   plen = BIRD_ALIGN(plen, 1024);
        !            97:   plen = MIN(plen, 0xffff);
        !            98:   sk_set_tbsize(ifa->sk, plen);
        !            99:   return 1;
        !           100: }
        !           101: 
        !           102: 
        !           103: struct nbma_node *
        !           104: find_nbma_node_(list *nnl, ip_addr ip)
        !           105: {
        !           106:   struct nbma_node *nn;
        !           107: 
        !           108:   WALK_LIST(nn, *nnl)
        !           109:     if (ipa_equal(nn->ip, ip))
        !           110:       return nn;
        !           111: 
        !           112:   return NULL;
        !           113: }
        !           114: 
        !           115: 
        !           116: static int
        !           117: ospf_sk_open(struct ospf_iface *ifa)
        !           118: {
        !           119:   struct ospf_proto *p = ifa->oa->po;
        !           120: 
        !           121:   sock *sk = sk_new(ifa->pool);
        !           122:   sk->type = SK_IP;
        !           123:   sk->subtype = ospf_is_v2(p) ? SK_IPV4 : SK_IPV6;
        !           124:   sk->dport = OSPF_PROTO;
        !           125:   sk->saddr = ifa->addr->ip;
        !           126:   sk->iface = ifa->iface;
        !           127:   sk->vrf = p->p.vrf;
        !           128: 
        !           129:   sk->tos = ifa->cf->tx_tos;
        !           130:   sk->priority = ifa->cf->tx_priority;
        !           131:   sk->rx_hook = ospf_rx_hook;
        !           132:   // sk->tx_hook = ospf_tx_hook;
        !           133:   sk->err_hook = ospf_err_hook;
        !           134:   sk->rbsize = sk->tbsize = ifa_bufsize(ifa);
        !           135:   sk->data = (void *) ifa;
        !           136:   sk->flags = SKF_LADDR_RX | (ifa->check_ttl ? SKF_TTL_RX : 0);
        !           137:   sk->ttl = ifa->cf->ttl_security ? 255 : 1;
        !           138: 
        !           139:   if (sk_open(sk) < 0)
        !           140:     goto err;
        !           141: 
        !           142:   /* 12 is an offset of the checksum in an OSPFv3 packet */
        !           143:   if (ospf_is_v3(p) && !ifa->autype)
        !           144:     if (sk_set_ipv6_checksum(sk, 12) < 0)
        !           145:       goto err;
        !           146: 
        !           147:   if ((ifa->type == OSPF_IT_BCAST) || (ifa->type == OSPF_IT_PTP))
        !           148:   {
        !           149:     if (ifa->cf->real_bcast)
        !           150:     {
        !           151:       ifa->all_routers = ifa->addr->brd;
        !           152:       ifa->des_routers = IPA_NONE;
        !           153: 
        !           154:       if (sk_setup_broadcast(sk) < 0)
        !           155:        goto err;
        !           156:     }
        !           157:     else
        !           158:     {
        !           159:       ifa->all_routers = ospf_is_v2(p) ? IP4_OSPF_ALL_ROUTERS : IP6_OSPF_ALL_ROUTERS;
        !           160:       ifa->des_routers = ospf_is_v2(p) ? IP4_OSPF_DES_ROUTERS : IP6_OSPF_DES_ROUTERS;
        !           161: 
        !           162:       if (sk_setup_multicast(sk) < 0)
        !           163:        goto err;
        !           164: 
        !           165:       if (sk_join_group(sk, ifa->all_routers) < 0)
        !           166:        goto err;
        !           167:     }
        !           168:   }
        !           169: 
        !           170:   ifa->sk = sk;
        !           171:   ifa->sk_dr = 0;
        !           172:   return 1;
        !           173: 
        !           174:  err:
        !           175:   sk_log_error(sk, p->p.name);
        !           176:   rfree(sk);
        !           177:   return 0;
        !           178: }
        !           179: 
        !           180: static inline void
        !           181: ospf_sk_join_dr(struct ospf_iface *ifa)
        !           182: {
        !           183:   if (ifa->sk_dr)
        !           184:     return;
        !           185: 
        !           186:   if (sk_join_group(ifa->sk, ifa->des_routers) < 0)
        !           187:     sk_log_error(ifa->sk, ifa->oa->po->p.name);
        !           188: 
        !           189:   ifa->sk_dr = 1;
        !           190: }
        !           191: 
        !           192: static inline void
        !           193: ospf_sk_leave_dr(struct ospf_iface *ifa)
        !           194: {
        !           195:   if (!ifa->sk_dr)
        !           196:     return;
        !           197: 
        !           198:   if (sk_leave_group(ifa->sk, ifa->des_routers) < 0)
        !           199:     sk_log_error(ifa->sk, ifa->oa->po->p.name);
        !           200: 
        !           201:   ifa->sk_dr = 0;
        !           202: }
        !           203: 
        !           204: void
        !           205: ospf_open_vlink_sk(struct ospf_proto *p)
        !           206: {
        !           207:   sock *sk = sk_new(p->p.pool);
        !           208:   sk->type = SK_IP;
        !           209:   sk->subtype = ospf_is_v2(p) ? SK_IPV4 : SK_IPV6;
        !           210:   sk->dport = OSPF_PROTO;
        !           211:   sk->vrf = p->p.vrf;
        !           212: 
        !           213:   /* FIXME: configurable tos/priority ? */
        !           214:   sk->tos = IP_PREC_INTERNET_CONTROL;
        !           215:   sk->priority = sk_priority_control;
        !           216:   sk->err_hook = ospf_verr_hook;
        !           217: 
        !           218:   sk->rbsize = 0;
        !           219:   sk->tbsize = ospf_is_v2(p) ? IP4_MIN_MTU : IP6_MIN_MTU;
        !           220:   sk->data = (void *) p;
        !           221:   sk->flags = 0;
        !           222: 
        !           223:   if (sk_open(sk) < 0)
        !           224:     goto err;
        !           225: 
        !           226:   /* 12 is an offset of the checksum in an OSPFv3 packet */
        !           227:   if (ospf_is_v3(p))
        !           228:     if (sk_set_ipv6_checksum(sk, 12) < 0)
        !           229:       goto err;
        !           230: 
        !           231:   p->vlink_sk = sk;
        !           232:   return;
        !           233: 
        !           234:  err:
        !           235:   sk_log_error(sk, p->p.name);
        !           236:   log(L_ERR "%s: Cannot open virtual link socket", p->p.name);
        !           237:   rfree(sk);
        !           238: }
        !           239: 
        !           240: static void
        !           241: ospf_iface_down(struct ospf_iface *ifa)
        !           242: {
        !           243:   struct ospf_proto *p = ifa->oa->po;
        !           244:   struct ospf_neighbor *n, *nx;
        !           245:   struct ospf_iface *iff;
        !           246: 
        !           247:   if (ifa->type != OSPF_IT_VLINK)
        !           248:   {
        !           249:     if (ospf_is_v3(ifa->oa->po))
        !           250:       OSPF_TRACE(D_EVENTS, "Removing interface %s (IID %d) from area %R",
        !           251:                 ifa->ifname, ifa->instance_id, ifa->oa->areaid);
        !           252:     else if (ifa->addr->flags & IA_PEER)
        !           253:       OSPF_TRACE(D_EVENTS, "Removing interface %s (peer %I) from area %R",
        !           254:                 ifa->ifname, ifa->addr->opposite, ifa->oa->areaid);
        !           255:     else
        !           256:       OSPF_TRACE(D_EVENTS, "Removing interface %s (%N) from area %R",
        !           257:                 ifa->ifname, &ifa->addr->prefix, ifa->oa->areaid);
        !           258: 
        !           259:     /* First of all kill all the related vlinks */
        !           260:     WALK_LIST(iff, p->iface_list)
        !           261:     {
        !           262:       if ((iff->type == OSPF_IT_VLINK) && (iff->vifa == ifa))
        !           263:        ospf_iface_sm(iff, ISM_DOWN);
        !           264:     }
        !           265:   }
        !           266: 
        !           267:   WALK_LIST_DELSAFE(n, nx, ifa->neigh_list)
        !           268:     ospf_neigh_sm(n, INM_KILLNBR);
        !           269: 
        !           270:   if (ifa->hello_timer)
        !           271:     tm_stop(ifa->hello_timer);
        !           272: 
        !           273:   if (ifa->poll_timer)
        !           274:     tm_stop(ifa->poll_timer);
        !           275: 
        !           276:   if (ifa->wait_timer)
        !           277:     tm_stop(ifa->wait_timer);
        !           278: 
        !           279:   ospf_flush2_lsa(p, &ifa->link_lsa);
        !           280:   ospf_flush2_lsa(p, &ifa->net_lsa);
        !           281:   ospf_flush2_lsa(p, &ifa->pxn_lsa);
        !           282: 
        !           283:   if (ifa->type == OSPF_IT_VLINK)
        !           284:   {
        !           285:     ifa->vifa = NULL;
        !           286:     ifa->addr = NULL;
        !           287:     ifa->cost = 0;
        !           288:     ifa->vip = IPA_NONE;
        !           289:   }
        !           290: 
        !           291:   ifa->rt_pos_beg = 0;
        !           292:   ifa->rt_pos_end = 0;
        !           293:   ifa->px_pos_beg = 0;
        !           294:   ifa->px_pos_end = 0;
        !           295: }
        !           296: 
        !           297: 
        !           298: void
        !           299: ospf_iface_remove(struct ospf_iface *ifa)
        !           300: {
        !           301:   struct ospf_proto *p = ifa->oa->po;
        !           302:   int i;
        !           303: 
        !           304:   if (ifa->type == OSPF_IT_VLINK)
        !           305:     OSPF_TRACE(D_EVENTS, "Removing vlink to %R via area %R", ifa->vid, ifa->voa->areaid);
        !           306: 
        !           307:   /* Release LSAs from flood queue */
        !           308:   if (!ifa->stub)
        !           309:     for (i = 0; i < ifa->flood_queue_used; i++)
        !           310:       ifa->flood_queue[i]->ret_count--;
        !           311: 
        !           312:   ospf_iface_sm(ifa, ISM_DOWN);
        !           313:   rem_node(NODE ifa);
        !           314:   rfree(ifa->pool);
        !           315: }
        !           316: 
        !           317: void
        !           318: ospf_iface_shutdown(struct ospf_iface *ifa)
        !           319: {
        !           320:   if (ifa->state > OSPF_IS_DOWN)
        !           321:     ospf_send_hello(ifa, OHS_SHUTDOWN, NULL);
        !           322: }
        !           323: 
        !           324: /**
        !           325:  * ospf_iface_chstate - handle changes of interface state
        !           326:  * @ifa: OSPF interface
        !           327:  * @state: new state
        !           328:  *
        !           329:  * Many actions must be taken according to interface state changes. New network
        !           330:  * LSAs must be originated, flushed, new multicast sockets to listen for messages for
        !           331:  * %ALLDROUTERS have to be opened, etc.
        !           332:  */
        !           333: void
        !           334: ospf_iface_chstate(struct ospf_iface *ifa, u8 state)
        !           335: {
        !           336:   struct ospf_proto *p = ifa->oa->po;
        !           337:   u8 oldstate = ifa->state;
        !           338: 
        !           339:   if (state == oldstate)
        !           340:     return;
        !           341: 
        !           342:   OSPF_TRACE(D_EVENTS, "Interface %s changed state from %s to %s",
        !           343:             ifa->ifname, ospf_is_names[oldstate], ospf_is_names[state]);
        !           344: 
        !           345:   ifa->state = state;
        !           346: 
        !           347:   if ((ifa->type == OSPF_IT_BCAST) && ipa_nonzero(ifa->des_routers) && ifa->sk)
        !           348:   {
        !           349:     if ((state == OSPF_IS_BACKUP) || (state == OSPF_IS_DR))
        !           350:       ospf_sk_join_dr(ifa);
        !           351:     else
        !           352:       ospf_sk_leave_dr(ifa);
        !           353:   }
        !           354: 
        !           355:   if ((oldstate > OSPF_IS_LOOP) && (state <= OSPF_IS_LOOP))
        !           356:     ospf_iface_down(ifa);
        !           357: 
        !           358:   /* RFC 2328 12.4 Event 2 - iface state change */
        !           359:   ospf_notify_rt_lsa(ifa->oa);
        !           360: 
        !           361:   /* RFC 5340 4.4.3 Event 1 - iface state change */
        !           362:   ospf_notify_link_lsa(ifa);
        !           363: 
        !           364:   /* RFC 2328 12.4 Event 3 - iface enters/leaves DR state */
        !           365:   ospf_notify_net_lsa(ifa);
        !           366: }
        !           367: 
        !           368: /**
        !           369:  * ospf_iface_sm - OSPF interface state machine
        !           370:  * @ifa: OSPF interface
        !           371:  * @event: event comming to state machine
        !           372:  *
        !           373:  * This fully respects 9.3 of RFC 2328 except we have slightly
        !           374:  * different handling of %DOWN and %LOOP state. We remove intefaces
        !           375:  * that are %DOWN. %DOWN state is used when an interface is waiting
        !           376:  * for a lock. %LOOP state is used when an interface does not have a
        !           377:  * link.
        !           378:  */
        !           379: void
        !           380: ospf_iface_sm(struct ospf_iface *ifa, int event)
        !           381: {
        !           382:   DBG("SM on %s. Event is '%s'\n", ifa->ifname, ospf_ism_names[event]);
        !           383: 
        !           384:   switch (event)
        !           385:   {
        !           386:   case ISM_UP:
        !           387:     if (ifa->state <= OSPF_IS_LOOP)
        !           388:     {
        !           389:       /* Now, nothing should be adjacent */
        !           390:       if ((ifa->type == OSPF_IT_PTP) ||
        !           391:          (ifa->type == OSPF_IT_PTMP) ||
        !           392:          (ifa->type == OSPF_IT_VLINK))
        !           393:       {
        !           394:        ospf_iface_chstate(ifa, OSPF_IS_PTP);
        !           395:       }
        !           396:       else
        !           397:       {
        !           398:        if (ifa->priority == 0)
        !           399:          ospf_iface_chstate(ifa, OSPF_IS_DROTHER);
        !           400:        else
        !           401:        {
        !           402:          ospf_iface_chstate(ifa, OSPF_IS_WAITING);
        !           403:          if (ifa->wait_timer)
        !           404:            tm_start(ifa->wait_timer, ifa->waitint S);
        !           405:        }
        !           406:       }
        !           407: 
        !           408:       if (ifa->hello_timer)
        !           409:        tm_start(ifa->hello_timer, ifa->helloint S);
        !           410: 
        !           411:       if (ifa->poll_timer)
        !           412:        tm_start(ifa->poll_timer, ifa->pollint S);
        !           413: 
        !           414:       ospf_send_hello(ifa, OHS_HELLO, NULL);
        !           415:     }
        !           416:     break;
        !           417: 
        !           418:   case ISM_BACKS:
        !           419:   case ISM_WAITF:
        !           420:     if (ifa->state == OSPF_IS_WAITING)
        !           421:       ospf_dr_election(ifa);
        !           422:     break;
        !           423: 
        !           424:   case ISM_NEICH:
        !           425:     if (ifa->state >= OSPF_IS_DROTHER)
        !           426:       ospf_dr_election(ifa);
        !           427:     break;
        !           428: 
        !           429:   case ISM_LOOP:
        !           430:     if ((ifa->state > OSPF_IS_LOOP) && ifa->check_link)
        !           431:       ospf_iface_chstate(ifa, OSPF_IS_LOOP);
        !           432:     break;
        !           433: 
        !           434:   case ISM_UNLOOP:
        !           435:     /* Immediate go UP */
        !           436:     if (ifa->state == OSPF_IS_LOOP)
        !           437:       ospf_iface_sm(ifa, ISM_UP);
        !           438:     break;
        !           439: 
        !           440:   case ISM_DOWN:
        !           441:     ospf_iface_chstate(ifa, OSPF_IS_DOWN);
        !           442:     break;
        !           443: 
        !           444:   default:
        !           445:     bug("OSPF_I_SM - Unknown event?");
        !           446:     break;
        !           447:   }
        !           448: 
        !           449: }
        !           450: 
        !           451: static u8
        !           452: ospf_iface_classify_(struct iface *ifa, struct ifa *addr)
        !           453: {
        !           454:   if (ipa_nonzero(addr->opposite))
        !           455:     return (ifa->flags & IF_MULTICAST) ? OSPF_IT_PTP :  OSPF_IT_PTMP;
        !           456: 
        !           457:   if ((ifa->flags & (IF_MULTIACCESS | IF_MULTICAST)) ==
        !           458:       (IF_MULTIACCESS | IF_MULTICAST))
        !           459:     return OSPF_IT_BCAST;
        !           460: 
        !           461:   if ((ifa->flags & (IF_MULTIACCESS | IF_MULTICAST)) == IF_MULTIACCESS)
        !           462:     return OSPF_IT_NBMA;
        !           463: 
        !           464:   return OSPF_IT_PTP;
        !           465: }
        !           466: 
        !           467: static inline u8
        !           468: ospf_iface_classify(u8 type, struct ifa *addr)
        !           469: {
        !           470:   return (type != OSPF_IT_UNDEF) ? type : ospf_iface_classify_(addr->iface, addr);
        !           471: }
        !           472: 
        !           473: 
        !           474: struct ospf_iface *
        !           475: ospf_iface_find(struct ospf_proto *p, struct iface *what)
        !           476: {
        !           477:   struct ospf_iface *ifa;
        !           478: 
        !           479:   WALK_LIST(ifa, p->iface_list)
        !           480:     if ((ifa->iface == what) && (ifa->type != OSPF_IT_VLINK))
        !           481:       return ifa;
        !           482: 
        !           483:   return NULL;
        !           484: }
        !           485: 
        !           486: static void
        !           487: ospf_iface_add(struct object_lock *lock)
        !           488: {
        !           489:   struct ospf_iface *ifa = lock->data;
        !           490:   struct ospf_proto *p = ifa->oa->po;
        !           491: 
        !           492:   /* Open socket if interface is not stub */
        !           493:   if (! ifa->stub && ! ospf_sk_open(ifa))
        !           494:   {
        !           495:     log(L_ERR "%s: Cannot open socket for %s, declaring as stub", p->p.name, ifa->ifname);
        !           496:     ifa->ioprob = OSPF_I_SK;
        !           497:     ifa->stub = 1;
        !           498:   }
        !           499: 
        !           500:   if (! ifa->stub)
        !           501:   {
        !           502:     ifa->hello_timer = tm_new_init(ifa->pool, hello_timer_hook, ifa, ifa->helloint S, 0);
        !           503: 
        !           504:     if (ifa->type == OSPF_IT_NBMA)
        !           505:       ifa->poll_timer = tm_new_init(ifa->pool, poll_timer_hook, ifa, ifa->pollint S, 0);
        !           506: 
        !           507:     if ((ifa->type == OSPF_IT_BCAST) || (ifa->type == OSPF_IT_NBMA))
        !           508:       ifa->wait_timer = tm_new_init(ifa->pool, wait_timer_hook, ifa, 0, 0);
        !           509: 
        !           510:     ifa->flood_queue_size = ifa_flood_queue_size(ifa);
        !           511:     ifa->flood_queue = mb_allocz(ifa->pool, ifa->flood_queue_size * sizeof(void *));
        !           512:   }
        !           513: 
        !           514:   /* Do iface UP, unless there is no link (then wait in LOOP state) */
        !           515:   if (!ifa->check_link || (ifa->iface->flags & IF_LINK_UP))
        !           516:     ospf_iface_sm(ifa, ISM_UP);
        !           517:   else
        !           518:     ospf_iface_chstate(ifa, OSPF_IS_LOOP);
        !           519: }
        !           520: 
        !           521: static inline void
        !           522: add_nbma_node(struct ospf_iface *ifa, struct nbma_node *src, int found)
        !           523: {
        !           524:   struct nbma_node *n = mb_alloc(ifa->pool, sizeof(struct nbma_node));
        !           525:   add_tail(&ifa->nbma_list, NODE n);
        !           526:   n->ip = src->ip;
        !           527:   n->eligible = src->eligible;
        !           528:   n->found = found;
        !           529: }
        !           530: 
        !           531: static int
        !           532: ospf_iface_stubby(struct ospf_iface_patt *ip, struct ifa *addr)
        !           533: {
        !           534:   /* vlink cannot be stub */
        !           535:   if (ip->type == OSPF_IT_VLINK)
        !           536:     return 0;
        !           537: 
        !           538:   /* a host address */
        !           539:   if (addr->flags & IA_HOST)
        !           540:     return 1;
        !           541: 
        !           542:   /* a loopback iface */
        !           543:   if (addr->iface->flags & IF_LOOPBACK)
        !           544:     return 1;
        !           545: 
        !           546:   return ip->stub;
        !           547: }
        !           548: 
        !           549: void
        !           550: ospf_iface_new(struct ospf_area *oa, struct ifa *addr, struct ospf_iface_patt *ip)
        !           551: {
        !           552:   struct ospf_proto *p = oa->po;
        !           553:   struct iface *iface = addr->iface;
        !           554:   struct ospf_iface *ifa;
        !           555:   struct pool *pool;
        !           556: 
        !           557:   if (ospf_is_v3(p))
        !           558:     OSPF_TRACE(D_EVENTS, "Adding interface %s (IID %d) to area %R",
        !           559:               iface->name, ip->instance_id, oa->areaid);
        !           560:   else if (addr->flags & IA_PEER)
        !           561:     OSPF_TRACE(D_EVENTS, "Adding interface %s (peer %I) to area %R",
        !           562:               iface->name, addr->opposite, oa->areaid);
        !           563:   else
        !           564:     OSPF_TRACE(D_EVENTS, "Adding interface %s (%N) to area %R",
        !           565:               iface->name, &addr->prefix, oa->areaid);
        !           566: 
        !           567:   pool = rp_new(p->p.pool, "OSPF Interface");
        !           568:   ifa = mb_allocz(pool, sizeof(struct ospf_iface));
        !           569:   ifa->iface = iface;
        !           570:   ifa->addr = addr;
        !           571:   ifa->oa = oa;
        !           572:   ifa->cf = ip;
        !           573:   ifa->pool = pool;
        !           574: 
        !           575:   ifa->iface_id = iface->index;
        !           576:   ifa->ifname = iface->name;
        !           577: 
        !           578:   ifa->cost = ip->cost;
        !           579:   ifa->rxmtint = ip->rxmtint;
        !           580:   ifa->inftransdelay = ip->inftransdelay;
        !           581:   ifa->priority = ip->priority;
        !           582:   ifa->helloint = ip->helloint;
        !           583:   ifa->pollint = ip->pollint;
        !           584:   ifa->strictnbma = ip->strictnbma;
        !           585:   ifa->waitint = ip->waitint;
        !           586:   ifa->deadint = ip->deadint;
        !           587:   ifa->stub = ospf_iface_stubby(ip, addr);
        !           588:   ifa->ioprob = OSPF_I_OK;
        !           589:   ifa->tx_length = ifa_tx_length(ifa);
        !           590:   ifa->tx_hdrlen = ifa_tx_hdrlen(ifa);
        !           591:   ifa->check_link = ip->check_link;
        !           592:   ifa->ecmp_weight = ip->ecmp_weight;
        !           593:   ifa->check_ttl = (ip->ttl_security == 1);
        !           594:   ifa->bfd = ip->bfd;
        !           595:   ifa->autype = ip->autype;
        !           596:   ifa->passwords = ip->passwords;
        !           597:   ifa->instance_id = ip->instance_id;
        !           598: 
        !           599:   ifa->ptp_netmask = !(addr->flags & IA_PEER);
        !           600:   if (ip->ptp_netmask < 2)
        !           601:     ifa->ptp_netmask = ip->ptp_netmask;
        !           602: 
        !           603:   ifa->drip = ifa->bdrip = ospf_is_v2(p) ? IPA_NONE4 : IPA_NONE6;
        !           604: 
        !           605:   ifa->type = ospf_iface_classify(ip->type, addr);
        !           606: 
        !           607:   /* Check validity of interface type */
        !           608:   int old_type = ifa->type;
        !           609:   u32 if_multi_flag = ip->real_bcast ? IF_BROADCAST : IF_MULTICAST;
        !           610: 
        !           611:   if (ospf_is_v2(p) && (ifa->type == OSPF_IT_BCAST) && (addr->flags & IA_PEER))
        !           612:     ifa->type = OSPF_IT_PTP;
        !           613: 
        !           614:   if (ospf_is_v2(p) && (ifa->type == OSPF_IT_NBMA) && (addr->flags & IA_PEER))
        !           615:     ifa->type = OSPF_IT_PTMP;
        !           616: 
        !           617:   if ((ifa->type == OSPF_IT_BCAST) && !(iface->flags & if_multi_flag) && !ifa->stub)
        !           618:     ifa->type = OSPF_IT_NBMA;
        !           619: 
        !           620:   if ((ifa->type == OSPF_IT_PTP) && !(iface->flags & if_multi_flag) && !ifa->stub)
        !           621:     ifa->type = OSPF_IT_PTMP;
        !           622: 
        !           623:   if (ifa->type != old_type)
        !           624:     log(L_WARN "%s: Cannot use interface %s as %s, forcing %s",
        !           625:        p->p.name, iface->name, ospf_it[old_type], ospf_it[ifa->type]);
        !           626: 
        !           627: 
        !           628:   if ((ifa->type == OSPF_IT_PTP) || (ifa->type == OSPF_IT_PTMP))
        !           629:     ifa->link_lsa_suppression = ip->link_lsa_suppression;
        !           630: 
        !           631:   ifa->state = OSPF_IS_DOWN;
        !           632:   init_list(&ifa->neigh_list);
        !           633:   init_list(&ifa->nbma_list);
        !           634: 
        !           635:   struct nbma_node *nb;
        !           636:   WALK_LIST(nb, ip->nbma_list)
        !           637:   {
        !           638:     /* In OSPFv3, addr is link-local while configured neighbors could
        !           639:        have global IP (although RFC 5340 C.5 says link-local addresses
        !           640:        should be used). Because OSPFv3 iface is not subnet-specific,
        !           641:        there is no need for ipa_in_net() check */
        !           642: 
        !           643:     if (ospf_is_v2(p) && !ipa_in_netX(nb->ip, &addr->prefix))
        !           644:       continue;
        !           645: 
        !           646:     if (ospf_is_v3(p) && !ipa_is_link_local(nb->ip))
        !           647:       log(L_WARN "%s: Configured neighbor address (%I) should be link-local",
        !           648:          p->p.name, nb->ip);
        !           649: 
        !           650:     add_nbma_node(ifa, nb, 0);
        !           651:   }
        !           652: 
        !           653:   add_tail(&oa->po->iface_list, NODE ifa);
        !           654: 
        !           655:   struct object_lock *lock = olock_new(pool);
        !           656:   lock->addr = ospf_is_v2(p) ? ipa_from_ip4(net4_prefix(&ifa->addr->prefix)) : IPA_NONE;
        !           657:   lock->type = OBJLOCK_IP;
        !           658:   lock->port = OSPF_PROTO;
        !           659:   lock->inst = ifa->instance_id;
        !           660:   lock->iface = iface;
        !           661:   lock->data = ifa;
        !           662:   lock->hook = ospf_iface_add;
        !           663: 
        !           664:   olock_acquire(lock);
        !           665: }
        !           666: 
        !           667: void
        !           668: ospf_iface_new_vlink(struct ospf_proto *p, struct ospf_iface_patt *ip)
        !           669: {
        !           670:   struct ospf_iface *ifa;
        !           671:   struct pool *pool;
        !           672: 
        !           673:   if (!p->vlink_sk)
        !           674:     return;
        !           675: 
        !           676:   OSPF_TRACE(D_EVENTS, "Adding vlink to %R via area %R", ip->vid, ip->voa);
        !           677: 
        !           678:   /* Vlink ifname is stored just after the ospf_iface structure */
        !           679: 
        !           680:   pool = rp_new(p->p.pool, "OSPF Vlink");
        !           681:   ifa = mb_allocz(pool, sizeof(struct ospf_iface) + 16);
        !           682:   ifa->oa = p->backbone;
        !           683:   ifa->cf = ip;
        !           684:   ifa->pool = pool;
        !           685: 
        !           686:   /* Assign iface ID, for vlinks, this is ugly hack */
        !           687:   u32 vlink_id = p->last_vlink_id++;
        !           688:   ifa->iface_id = vlink_id + OSPF_VLINK_ID_OFFSET;
        !           689:   ifa->ifname = (void *) (ifa + 1);
        !           690:   bsprintf(ifa->ifname, "vlink%d", vlink_id);
        !           691: 
        !           692:   ifa->voa = ospf_find_area(p, ip->voa);
        !           693:   ifa->vid = ip->vid;
        !           694:   ifa->sk = p->vlink_sk;
        !           695: 
        !           696:   ifa->helloint = ip->helloint;
        !           697:   ifa->rxmtint = ip->rxmtint;
        !           698:   ifa->waitint = ip->waitint;
        !           699:   ifa->deadint = ip->deadint;
        !           700:   ifa->inftransdelay = ip->inftransdelay;
        !           701:   ifa->tx_length = ospf_is_v2(p) ? IP4_MIN_MTU : IP6_MIN_MTU;
        !           702:   ifa->tx_hdrlen = ifa_tx_hdrlen(ifa);
        !           703:   ifa->autype = ip->autype;
        !           704:   ifa->passwords = ip->passwords;
        !           705:   ifa->instance_id = ip->instance_id;
        !           706: 
        !           707:   ifa->type = OSPF_IT_VLINK;
        !           708: 
        !           709:   ifa->state = OSPF_IS_DOWN;
        !           710:   init_list(&ifa->neigh_list);
        !           711:   init_list(&ifa->nbma_list);
        !           712: 
        !           713:   add_tail(&p->iface_list, NODE ifa);
        !           714: 
        !           715:   ifa->hello_timer = tm_new_init(ifa->pool, hello_timer_hook, ifa, ifa->helloint S, 0);
        !           716: 
        !           717:   ifa->flood_queue_size = ifa_flood_queue_size(ifa);
        !           718:   ifa->flood_queue = mb_allocz(ifa->pool, ifa->flood_queue_size * sizeof(void *));
        !           719: }
        !           720: 
        !           721: static void
        !           722: ospf_iface_change_timer(timer *tm, uint val)
        !           723: {
        !           724:   if (!tm)
        !           725:     return;
        !           726: 
        !           727:   tm->recurrent = val S;
        !           728: 
        !           729:   if (tm_active(tm))
        !           730:     tm_start(tm, val S);
        !           731: }
        !           732: 
        !           733: static inline void
        !           734: ospf_iface_update_flood_queue_size(struct ospf_iface *ifa)
        !           735: {
        !           736:   uint old_size = ifa->flood_queue_size;
        !           737:   uint new_size = ifa_flood_queue_size(ifa);
        !           738: 
        !           739:   if (new_size <= old_size)
        !           740:     return;
        !           741: 
        !           742:   ifa->flood_queue_size = new_size;
        !           743:   ifa->flood_queue = mb_realloc(ifa->flood_queue, new_size * sizeof(void *));
        !           744:   bzero(ifa->flood_queue + old_size, (new_size - old_size) * sizeof(void *));
        !           745: }
        !           746: 
        !           747: int
        !           748: ospf_iface_reconfigure(struct ospf_iface *ifa, struct ospf_iface_patt *new)
        !           749: {
        !           750:   struct ospf_proto *p = ifa->oa->po;
        !           751:   struct ospf_iface_patt *old = ifa->cf;
        !           752:   char *ifname = ifa->ifname;
        !           753: 
        !           754:   /* Type could be changed in ospf_iface_new(),
        !           755:      but if config values are same then also results are same */
        !           756:   int old_type = ospf_iface_classify(old->type, ifa->addr);
        !           757:   int new_type = ospf_iface_classify(new->type, ifa->addr);
        !           758:   if (old_type != new_type)
        !           759:     return 0;
        !           760: 
        !           761:   int new_stub = ospf_iface_stubby(new, ifa->addr);
        !           762:   if (ifa->stub != new_stub)
        !           763:     return 0;
        !           764: 
        !           765:   /* Change of these options would require to reset the iface socket */
        !           766:   if ((new->real_bcast != old->real_bcast) ||
        !           767:       (new->tx_tos != old->tx_tos) ||
        !           768:       (new->tx_priority != old->tx_priority) ||
        !           769:       (new->ttl_security != old->ttl_security))
        !           770:     return 0;
        !           771: 
        !           772:   ifa->cf = new;
        !           773:   ifa->marked = 0;
        !           774: 
        !           775:   /* Cancel GR peers if GR is disabled */
        !           776:   if (!p->gr_mode && p->gr_count)
        !           777:   {
        !           778:     struct ospf_neighbor *n, *nx;
        !           779:     WALK_LIST_DELSAFE(n, nx, ifa->neigh_list)
        !           780:       if (n->gr_active)
        !           781:        ospf_neigh_cancel_graceful_restart(n);
        !           782:   }
        !           783: 
        !           784:   /* HELLO TIMER */
        !           785:   if (ifa->helloint != new->helloint)
        !           786:   {
        !           787:     OSPF_TRACE(D_EVENTS, "Changing hello interval of %s from %d to %d",
        !           788:               ifname, ifa->helloint, new->helloint);
        !           789: 
        !           790:     ifa->helloint = new->helloint;
        !           791:     ospf_iface_change_timer(ifa->hello_timer, ifa->helloint);
        !           792:   }
        !           793: 
        !           794:   /* RXMT TIMER */
        !           795:   if (ifa->rxmtint != new->rxmtint)
        !           796:   {
        !           797:     OSPF_TRACE(D_EVENTS, "Changing retransmit interval of %s from %d to %d",
        !           798:               ifname, ifa->rxmtint, new->rxmtint);
        !           799: 
        !           800:     ifa->rxmtint = new->rxmtint;
        !           801:     /* FIXME: Update neighbors' timers */
        !           802:   }
        !           803: 
        !           804:   /* POLL TIMER */
        !           805:   if (ifa->pollint != new->pollint)
        !           806:   {
        !           807:     OSPF_TRACE(D_EVENTS, "Changing poll interval of %s from %d to %d",
        !           808:               ifname, ifa->pollint, new->pollint);
        !           809: 
        !           810:     ifa->pollint = new->pollint;
        !           811:     ospf_iface_change_timer(ifa->poll_timer, ifa->pollint);
        !           812:   }
        !           813: 
        !           814:   /* WAIT TIMER */
        !           815:   if (ifa->waitint != new->waitint)
        !           816:   {
        !           817:     OSPF_TRACE(D_EVENTS, "Changing wait interval of %s from %d to %d",
        !           818:               ifname, ifa->waitint, new->waitint);
        !           819: 
        !           820:     ifa->waitint = new->waitint;
        !           821:     if (ifa->wait_timer && tm_active(ifa->wait_timer))
        !           822:       tm_start(ifa->wait_timer, ifa->waitint S);
        !           823:   }
        !           824: 
        !           825:   /* DEAD TIMER */
        !           826:   if (ifa->deadint != new->deadint)
        !           827:   {
        !           828:     OSPF_TRACE(D_EVENTS, "Changing dead interval of %s from %d to %d",
        !           829:               ifname, ifa->deadint, new->deadint);
        !           830:     ifa->deadint = new->deadint;
        !           831:   }
        !           832: 
        !           833:   /* INFTRANS */
        !           834:   if (ifa->inftransdelay != new->inftransdelay)
        !           835:   {
        !           836:     OSPF_TRACE(D_EVENTS, "Changing transmit delay of %s from %d to %d",
        !           837:                     ifname, ifa->inftransdelay, new->inftransdelay);
        !           838:     ifa->inftransdelay = new->inftransdelay;
        !           839:   }
        !           840: 
        !           841:   /* AUTHENTICATION */
        !           842:   if (ifa->autype != new->autype)
        !           843:   {
        !           844:     OSPF_TRACE(D_EVENTS, "Changing authentication type of %s", ifname);
        !           845:     ifa->autype = new->autype;
        !           846: 
        !           847:     /* For OSPFv3, we need to update checksum calculation by OS */
        !           848:     if (ospf_is_v3(p) && ifa->sk)
        !           849:       if (sk_set_ipv6_checksum(ifa->sk, ifa->autype ? -1 : 12) < 0)
        !           850:       {
        !           851:        sk_log_error(ifa->sk, p->p.name);
        !           852:        return 0;
        !           853:       }
        !           854:   }
        !           855: 
        !           856:   /* Update passwords */
        !           857:   ifa->passwords = new->passwords;
        !           858: 
        !           859:   /* Update header length */
        !           860:   ifa->tx_hdrlen = ifa_tx_hdrlen(ifa);
        !           861: 
        !           862:   /* Remaining options are just for proper interfaces */
        !           863:   if (ifa->type == OSPF_IT_VLINK)
        !           864:     return 1;
        !           865: 
        !           866: 
        !           867:   /* COST */
        !           868:   if (ifa->cost != new->cost)
        !           869:   {
        !           870:     OSPF_TRACE(D_EVENTS, "Changing cost of %s from %d to %d",
        !           871:               ifname, ifa->cost, new->cost);
        !           872: 
        !           873:     ifa->cost = new->cost;
        !           874:   }
        !           875: 
        !           876:   /* PRIORITY */
        !           877:   if (ifa->priority != new->priority)
        !           878:   {
        !           879:     OSPF_TRACE(D_EVENTS, "Changing priority of %s from %d to %d",
        !           880:               ifname, ifa->priority, new->priority);
        !           881: 
        !           882:     ifa->priority = new->priority;
        !           883:     ospf_iface_sm(ifa, ISM_NEICH);
        !           884:     ospf_notify_link_lsa(ifa);
        !           885:   }
        !           886: 
        !           887:   /* STRICT NBMA */
        !           888:   if (ifa->strictnbma != new->strictnbma)
        !           889:   {
        !           890:     OSPF_TRACE(D_EVENTS, "Changing NBMA strictness of %s from %d to %d",
        !           891:               ifname, ifa->strictnbma, new->strictnbma);
        !           892:     ifa->strictnbma = new->strictnbma;
        !           893:   }
        !           894: 
        !           895:   struct nbma_node *nb, *nbx;
        !           896: 
        !           897:   /* NBMA LIST - remove or update old */
        !           898:   WALK_LIST_DELSAFE(nb, nbx, ifa->nbma_list)
        !           899:   {
        !           900:     struct nbma_node *nb2 = find_nbma_node_(&new->nbma_list, nb->ip);
        !           901:     if (nb2)
        !           902:     {
        !           903:       if (nb->eligible != nb2->eligible)
        !           904:       {
        !           905:        OSPF_TRACE(D_EVENTS, "Changing eligibility of NBMA neighbor %I on %s",
        !           906:                   nb->ip, ifname);
        !           907:        nb->eligible = nb2->eligible;
        !           908:       }
        !           909:     }
        !           910:     else
        !           911:     {
        !           912:       OSPF_TRACE(D_EVENTS, "Removing NBMA neighbor %I on %s",
        !           913:                       nb->ip, ifname);
        !           914:       rem_node(NODE nb);
        !           915:       mb_free(nb);
        !           916:     }
        !           917:   }
        !           918: 
        !           919:   /* NBMA LIST - add new */
        !           920:   WALK_LIST(nb, new->nbma_list)
        !           921:   {
        !           922:     /* See related note in ospf_iface_new() */
        !           923:     if (ospf_is_v2(p) && !ipa_in_netX(nb->ip, &ifa->addr->prefix))
        !           924:       continue;
        !           925: 
        !           926:     if (ospf_is_v3(p) && !ipa_is_link_local(nb->ip))
        !           927:       log(L_WARN "%s: Configured neighbor address (%I) should be link-local",
        !           928:          p->p.name, nb->ip);
        !           929: 
        !           930:     if (! find_nbma_node(ifa, nb->ip))
        !           931:     {
        !           932:       OSPF_TRACE(D_EVENTS, "Adding NBMA neighbor %I on %s",
        !           933:                 nb->ip, ifname);
        !           934:       add_nbma_node(ifa, nb, !!find_neigh_by_ip(ifa, nb->ip));
        !           935:     }
        !           936:   }
        !           937: 
        !           938:   int update_buffers = 0;
        !           939: 
        !           940:   /* TX LENGTH */
        !           941:   if (old->tx_length != new->tx_length)
        !           942:   {
        !           943:     OSPF_TRACE(D_EVENTS, "Changing TX length of %s from %d to %d",
        !           944:               ifname, old->tx_length, new->tx_length);
        !           945: 
        !           946:     /* ifa cannot be vlink */
        !           947:     ifa->tx_length = ifa_tx_length(ifa);
        !           948:     update_buffers = 1;
        !           949: 
        !           950:     if (!ifa->stub)
        !           951:       ospf_iface_update_flood_queue_size(ifa);
        !           952:   }
        !           953: 
        !           954:   /* RX BUFFER */
        !           955:   if (old->rx_buffer != new->rx_buffer)
        !           956:   {
        !           957:     OSPF_TRACE(D_EVENTS, "Changing buffer size of %s from %d to %d",
        !           958:               ifname, old->rx_buffer, new->rx_buffer);
        !           959: 
        !           960:     /* ifa cannot be vlink */
        !           961:     update_buffers = 1;
        !           962:   }
        !           963: 
        !           964:   /* Buffer size depends on both tx_length and rx_buffer options */
        !           965:   if (update_buffers && ifa->sk)
        !           966:   {
        !           967:     uint bsize = ifa_bufsize(ifa);
        !           968:     sk_set_rbsize(ifa->sk, bsize);
        !           969:     sk_set_tbsize(ifa->sk, bsize);
        !           970:   }
        !           971: 
        !           972:   /* LINK */
        !           973:   if (ifa->check_link != new->check_link)
        !           974:   {
        !           975:     OSPF_TRACE(D_EVENTS, "%s link check for %s",
        !           976:               new->check_link ? "Enabling" : "Disabling", ifname);
        !           977:     ifa->check_link = new->check_link;
        !           978: 
        !           979:     /* ifa cannot be vlink */
        !           980:     if (!(ifa->iface->flags & IF_LINK_UP))
        !           981:       ospf_iface_sm(ifa, ifa->check_link ? ISM_LOOP : ISM_UNLOOP);
        !           982:   }
        !           983: 
        !           984:   /* ECMP weight */
        !           985:   if (ifa->ecmp_weight != new->ecmp_weight)
        !           986:   {
        !           987:     OSPF_TRACE(D_EVENTS, "Changing ECMP weight of %s from %d to %d",
        !           988:               ifname, (int)ifa->ecmp_weight + 1, (int)new->ecmp_weight + 1);
        !           989:     ifa->ecmp_weight = new->ecmp_weight;
        !           990:   }
        !           991: 
        !           992:   /* Link LSA suppression */
        !           993:   if (((ifa->type == OSPF_IT_PTP) || (ifa->type == OSPF_IT_PTMP)) &&
        !           994:       (ifa->link_lsa_suppression != new->link_lsa_suppression))
        !           995:   {
        !           996:     OSPF_TRACE(D_EVENTS, "Changing link LSA suppression of %s from %d to %d",
        !           997:               ifname, ifa->link_lsa_suppression, new->link_lsa_suppression);
        !           998: 
        !           999:     ifa->link_lsa_suppression = new->link_lsa_suppression;
        !          1000:     ospf_notify_link_lsa(ifa);
        !          1001:   }
        !          1002: 
        !          1003:   /* BFD */
        !          1004:   if (ifa->bfd != new->bfd)
        !          1005:   {
        !          1006:     OSPF_TRACE(D_EVENTS, "%s BFD for %s",
        !          1007:               new->bfd ? "Enabling" : "Disabling", ifname);
        !          1008:     ifa->bfd = new->bfd;
        !          1009: 
        !          1010:     struct ospf_neighbor *n;
        !          1011:     WALK_LIST(n, ifa->neigh_list)
        !          1012:       ospf_neigh_update_bfd(n, ifa->bfd);
        !          1013:   }
        !          1014: 
        !          1015: 
        !          1016:   /* instance_id is not updated - it is part of key */
        !          1017: 
        !          1018:   return 1;
        !          1019: }
        !          1020: 
        !          1021: 
        !          1022: /*
        !          1023:  * State for matching iface pattterns walk
        !          1024:  *
        !          1025:  * This is significantly different in OSPFv2 and OSPFv3.
        !          1026:  * In OSPFv2, OSPF ifaces are created for each IP prefix (struct ifa)
        !          1027:  * In OSPFv3, OSPF ifaces are created based on real iface (struct iface)
        !          1028:  * We support instance_id for both OSPFv2 (RFC 6549) and OSPFv3.
        !          1029:  *
        !          1030:  * We process one ifa/iface and match it for all configured instance IDs. We
        !          1031:  * maintain bitfields to track whether given instance ID was already matched.
        !          1032:  * We have two bitfields, one global (active) and one per area (ignore), to
        !          1033:  * detect misconfigured cases where one iface with one instance ID matches in
        !          1034:  * multiple areas.
        !          1035:  */
        !          1036: 
        !          1037: struct ospf_mip_walk {
        !          1038:   u32 active[8];               /* Bitfield of active instance IDs */
        !          1039:   u32 ignore[8];               /* Bitfield of instance IDs matched in current area */
        !          1040:   struct ospf_area *oa;                /* Current area */
        !          1041:   struct ospf_iface_patt *ip;  /* Current iface pattern */
        !          1042:   struct iface *iface;         /* Specified iface (input) */
        !          1043:   struct ifa *a;               /* Specified ifa (input) */
        !          1044:   int warn;                    /* Whether iface matched in multiple areas */
        !          1045: };
        !          1046: 
        !          1047: static int
        !          1048: ospf_walk_matching_iface_patts(struct ospf_proto *p, struct ospf_mip_walk *s)
        !          1049: {
        !          1050:   int id;
        !          1051: 
        !          1052:   if (s->ip)
        !          1053:     goto step;
        !          1054: 
        !          1055:   WALK_LIST(s->oa, p->area_list)
        !          1056:   {
        !          1057:     if (s->oa->marked)
        !          1058:       continue;
        !          1059: 
        !          1060:     WALK_LIST(s->ip, s->oa->ac->patt_list)
        !          1061:     {
        !          1062:       id = s->ip->instance_id;
        !          1063:       if (BIT32_TEST(s->ignore, id))
        !          1064:        continue;
        !          1065: 
        !          1066:       if (iface_patt_match(&s->ip->i, s->iface, s->a))
        !          1067:       {
        !          1068:        /* Now we matched ifa/iface/instance_id for the first time in current area */
        !          1069:        BIT32_SET(s->ignore, id);
        !          1070: 
        !          1071:        /* If we already found it in previous areas, ignore it and add warning */
        !          1072:        if (BIT32_TEST(s->active, id))
        !          1073:          { s->warn = 1; continue; }
        !          1074: 
        !          1075:        BIT32_SET(s->active, id);
        !          1076:        return 1;
        !          1077:       step:
        !          1078:        ;
        !          1079:       }
        !          1080:     }
        !          1081:     BIT32_ZERO(s->ignore, 256);
        !          1082:   }
        !          1083: 
        !          1084:   if (s->warn)
        !          1085:     log(L_WARN "%s: Interface %s matches for multiple areas", p->p.name, s->iface->name);
        !          1086: 
        !          1087:   return 0;
        !          1088: }
        !          1089: 
        !          1090: 
        !          1091: static struct ospf_iface *
        !          1092: ospf_iface_find_by_key(struct ospf_proto *p, struct ifa *a, int instance_id)
        !          1093: {
        !          1094:   struct ospf_iface *ifa;
        !          1095: 
        !          1096:   WALK_LIST(ifa, p->iface_list)
        !          1097:     if ((ifa->addr == a) && (ifa->instance_id == instance_id) &&
        !          1098:        (ifa->type != OSPF_IT_VLINK))
        !          1099:       return ifa;
        !          1100: 
        !          1101:   return NULL;
        !          1102: }
        !          1103: 
        !          1104: 
        !          1105: void
        !          1106: ospf_ifa_notify2(struct proto *P, uint flags, struct ifa *a)
        !          1107: {
        !          1108:   struct ospf_proto *p = (struct ospf_proto *) P;
        !          1109: 
        !          1110:   if (a->prefix.type != NET_IP4)
        !          1111:     return;
        !          1112: 
        !          1113:   if (a->flags & IA_SECONDARY)
        !          1114:     return;
        !          1115: 
        !          1116:   if (a->scope <= SCOPE_LINK)
        !          1117:     return;
        !          1118: 
        !          1119:   /* In OSPFv2, we create OSPF iface for each address. */
        !          1120:   if (flags & IF_CHANGE_UP)
        !          1121:   {
        !          1122:     struct ospf_mip_walk s = { .iface = a->iface, .a = a };
        !          1123:     while (ospf_walk_matching_iface_patts(p, &s))
        !          1124:       ospf_iface_new(s.oa, a, s.ip);
        !          1125:   }
        !          1126: 
        !          1127:   if (flags & IF_CHANGE_DOWN)
        !          1128:   {
        !          1129:     struct ospf_iface *ifa, *ifx;
        !          1130:     WALK_LIST_DELSAFE(ifa, ifx, p->iface_list)
        !          1131:       if ((ifa->type != OSPF_IT_VLINK) && (ifa->addr == a))
        !          1132:        ospf_iface_remove(ifa);
        !          1133:     /* See a note in ospf_iface_notify() */
        !          1134:   }
        !          1135: }
        !          1136: 
        !          1137: void
        !          1138: ospf_ifa_notify3(struct proto *P, uint flags, struct ifa *a)
        !          1139: {
        !          1140:   struct ospf_proto *p = (struct ospf_proto *) P;
        !          1141: 
        !          1142:   if (a->flags & IA_SECONDARY)
        !          1143:     return;
        !          1144: 
        !          1145:   if (a->scope < SCOPE_LINK)
        !          1146:     return;
        !          1147: 
        !          1148:   /* In OSPFv3, we create OSPF iface for link-local address,
        !          1149:      other addresses are used for link-LSA. */
        !          1150:   if (a->scope == SCOPE_LINK)
        !          1151:   {
        !          1152:     if (a->prefix.type != NET_IP6)
        !          1153:       return;
        !          1154: 
        !          1155:     if (flags & IF_CHANGE_UP)
        !          1156:     {
        !          1157:       struct ospf_mip_walk s = { .iface = a->iface };
        !          1158:       while (ospf_walk_matching_iface_patts(p, &s))
        !          1159:        ospf_iface_new(s.oa, a, s.ip);
        !          1160:     }
        !          1161: 
        !          1162:     if (flags & IF_CHANGE_DOWN)
        !          1163:     {
        !          1164:       struct ospf_iface *ifa, *ifx;
        !          1165:       WALK_LIST_DELSAFE(ifa, ifx, p->iface_list)
        !          1166:        if ((ifa->addr == a) && (ifa->type != OSPF_IT_VLINK))
        !          1167:          ospf_iface_remove(ifa);
        !          1168:     }
        !          1169:   }
        !          1170:   else
        !          1171:   {
        !          1172:     if (a->prefix.type != ospf_get_af(p))
        !          1173:       return;
        !          1174: 
        !          1175:     struct ospf_iface *ifa;
        !          1176:     WALK_LIST(ifa, p->iface_list)
        !          1177:       if (ifa->iface == a->iface)
        !          1178:       {
        !          1179:        /* RFC 5340 4.4.3 Event 5 - prefix added/deleted */
        !          1180:        ospf_notify_link_lsa(ifa);
        !          1181:        ospf_notify_rt_lsa(ifa->oa);
        !          1182:       }
        !          1183:   }
        !          1184: }
        !          1185: 
        !          1186: 
        !          1187: static void
        !          1188: ospf_reconfigure_ifaces2(struct ospf_proto *p)
        !          1189: {
        !          1190:   struct iface *iface;
        !          1191:   struct ifa *a;
        !          1192: 
        !          1193:   WALK_LIST(iface, iface_list)
        !          1194:   {
        !          1195:     if (! (iface->flags & IF_UP))
        !          1196:       continue;
        !          1197: 
        !          1198:     WALK_LIST(a, iface->addrs)
        !          1199:     {
        !          1200:       if (a->prefix.type != NET_IP4)
        !          1201:        continue;
        !          1202: 
        !          1203:       if (a->flags & IA_SECONDARY)
        !          1204:        continue;
        !          1205: 
        !          1206:       if (a->scope <= SCOPE_LINK)
        !          1207:        continue;
        !          1208: 
        !          1209:       struct ospf_mip_walk s = { .iface = iface, .a = a };
        !          1210:       while (ospf_walk_matching_iface_patts(p, &s))
        !          1211:       {
        !          1212:        /* Main inner loop */
        !          1213:        struct ospf_iface *ifa = ospf_iface_find_by_key(p, a, s.ip->instance_id);
        !          1214:        if (ifa)
        !          1215:        {
        !          1216:          if ((ifa->oa == s.oa) && (ifa->marked < 2) &&
        !          1217:              ospf_iface_reconfigure(ifa, s.ip))
        !          1218:            continue;
        !          1219: 
        !          1220:          /* Hard restart */
        !          1221:          log(L_INFO "%s: Restarting interface %s (%N) in area %R",
        !          1222:              p->p.name, ifa->ifname, &a->prefix, s.oa->areaid);
        !          1223:          ospf_iface_shutdown(ifa);
        !          1224:          ospf_iface_remove(ifa);
        !          1225:        }
        !          1226: 
        !          1227:        ospf_iface_new(s.oa, a, s.ip);
        !          1228:       }
        !          1229:     }
        !          1230:   }
        !          1231: }
        !          1232: 
        !          1233: static void
        !          1234: ospf_reconfigure_ifaces3(struct ospf_proto *p)
        !          1235: {
        !          1236:   struct iface *iface;
        !          1237:   struct ifa *a;
        !          1238: 
        !          1239:   WALK_LIST(iface, iface_list)
        !          1240:   {
        !          1241:     if (! (iface->flags & IF_UP))
        !          1242:       continue;
        !          1243: 
        !          1244:     WALK_LIST(a, iface->addrs)
        !          1245:     {
        !          1246:       if (a->prefix.type != NET_IP6)
        !          1247:        continue;
        !          1248: 
        !          1249:       if (a->flags & IA_SECONDARY)
        !          1250:        continue;
        !          1251: 
        !          1252:       if (a->scope != SCOPE_LINK)
        !          1253:        continue;
        !          1254: 
        !          1255:       struct ospf_mip_walk s = { .iface = iface };
        !          1256:       while (ospf_walk_matching_iface_patts(p, &s))
        !          1257:       {
        !          1258:        /* Main inner loop */
        !          1259:        struct ospf_iface *ifa = ospf_iface_find_by_key(p, a, s.ip->instance_id);
        !          1260:        if (ifa)
        !          1261:        {
        !          1262:          if ((ifa->oa == s.oa) && (ifa->marked < 2) &&
        !          1263:              ospf_iface_reconfigure(ifa, s.ip))
        !          1264:            continue;
        !          1265: 
        !          1266:          /* Hard restart */
        !          1267:          log(L_INFO "%s: Restarting interface %s (IID %d) in area %R",
        !          1268:              p->p.name, ifa->ifname, ifa->instance_id, s.oa->areaid);
        !          1269:          ospf_iface_shutdown(ifa);
        !          1270:          ospf_iface_remove(ifa);
        !          1271:        }
        !          1272: 
        !          1273:        ospf_iface_new(s.oa, a, s.ip);
        !          1274:       }
        !          1275:     }
        !          1276:   }
        !          1277: }
        !          1278: 
        !          1279: void
        !          1280: ospf_reconfigure_ifaces(struct ospf_proto *p)
        !          1281: {
        !          1282:   if (ospf_is_v2(p))
        !          1283:     ospf_reconfigure_ifaces2(p);
        !          1284:   else
        !          1285:     ospf_reconfigure_ifaces3(p);
        !          1286: }
        !          1287: 
        !          1288: 
        !          1289: static void
        !          1290: ospf_iface_change_mtu(struct ospf_proto *p, struct ospf_iface *ifa)
        !          1291: {
        !          1292:   /* ifa is not vlink */
        !          1293: 
        !          1294:   OSPF_TRACE(D_EVENTS, "Interface %s changed MTU to %d",
        !          1295:             ifa->ifname, ifa->iface->mtu);
        !          1296: 
        !          1297:   ifa->tx_length = ifa_tx_length(ifa);
        !          1298: 
        !          1299:   if (!ifa->sk)
        !          1300:     return;
        !          1301: 
        !          1302:   /* We do not shrink dynamic buffers */
        !          1303:   uint bsize = ifa_bufsize(ifa);
        !          1304:   if (bsize > ifa->sk->rbsize)
        !          1305:     sk_set_rbsize(ifa->sk, bsize);
        !          1306:   if (bsize > ifa->sk->tbsize)
        !          1307:     sk_set_tbsize(ifa->sk, bsize);
        !          1308: 
        !          1309:   if (!ifa->stub)
        !          1310:     ospf_iface_update_flood_queue_size(ifa);
        !          1311: }
        !          1312: 
        !          1313: static void
        !          1314: ospf_iface_notify(struct ospf_proto *p, uint flags, struct ospf_iface *ifa)
        !          1315: {
        !          1316:   /* ifa is not vlink */
        !          1317: 
        !          1318:   if (flags & IF_CHANGE_DOWN)
        !          1319:   {
        !          1320:     ospf_iface_remove(ifa);
        !          1321:     return;
        !          1322:   }
        !          1323: 
        !          1324:   if (flags & IF_CHANGE_LINK)
        !          1325:     ospf_iface_sm(ifa, (ifa->iface->flags & IF_LINK_UP) ? ISM_UNLOOP : ISM_LOOP);
        !          1326: 
        !          1327:   if (flags & IF_CHANGE_MTU)
        !          1328:     ospf_iface_change_mtu(p, ifa);
        !          1329: }
        !          1330: 
        !          1331: void
        !          1332: ospf_if_notify(struct proto *P, uint flags, struct iface *iface)
        !          1333: {
        !          1334:   struct ospf_proto *p = (struct ospf_proto *) P;
        !          1335: 
        !          1336:   /*
        !          1337:   if (iface->flags & IF_IGNORE)
        !          1338:     return;
        !          1339:   */
        !          1340: 
        !          1341:   /* Going up means that there are no such ifaces yet */
        !          1342:   if (flags & IF_CHANGE_UP)
        !          1343:     return;
        !          1344: 
        !          1345:   struct ospf_iface *ifa, *ifx;
        !          1346:   WALK_LIST_DELSAFE(ifa, ifx, p->iface_list)
        !          1347:     if (ifa->iface == iface)
        !          1348:       ospf_iface_notify(p, flags, ifa);
        !          1349: 
        !          1350:   /* We use here that even shutting down iface also shuts down
        !          1351:      the vlinks, but vlinks are not freed and stays in the
        !          1352:      iface_list even when down */
        !          1353: }
        !          1354: 
        !          1355: void
        !          1356: ospf_iface_info(struct ospf_iface *ifa)
        !          1357: {
        !          1358:   char *more = "";
        !          1359: 
        !          1360:   if (ifa->strictnbma &&
        !          1361:       ((ifa->type == OSPF_IT_NBMA) || (ifa->type == OSPF_IT_PTMP)))
        !          1362:     more = " (strict)";
        !          1363: 
        !          1364:   if (ifa->cf->real_bcast &&
        !          1365:       ((ifa->type == OSPF_IT_BCAST) || (ifa->type == OSPF_IT_PTP)))
        !          1366:     more = " (real)";
        !          1367: 
        !          1368:   if (ifa->type == OSPF_IT_VLINK)
        !          1369:   {
        !          1370:     cli_msg(-1015, "Virtual link %s to %R", ifa->ifname, ifa->vid);
        !          1371:     cli_msg(-1015, "\tPeer IP: %I", ifa->vip);
        !          1372:     cli_msg(-1015, "\tTransit area: %R (%u)", ifa->voa->areaid, ifa->voa->areaid);
        !          1373:   }
        !          1374:   else
        !          1375:   {
        !          1376:     if (ospf_is_v3(ifa->oa->po))
        !          1377:       cli_msg(-1015, "Interface %s (IID %d)", ifa->ifname, ifa->instance_id);
        !          1378:     else if (ifa->addr->flags & IA_PEER)
        !          1379:       cli_msg(-1015, "Interface %s (peer %I)", ifa->ifname, ifa->addr->opposite);
        !          1380:     else
        !          1381:       cli_msg(-1015, "Interface %s (%N)", ifa->ifname, &ifa->addr->prefix);
        !          1382: 
        !          1383:     cli_msg(-1015, "\tType: %s%s", ospf_it[ifa->type], more);
        !          1384:     cli_msg(-1015, "\tArea: %R (%u)", ifa->oa->areaid, ifa->oa->areaid);
        !          1385:   }
        !          1386:   cli_msg(-1015, "\tState: %s%s", ospf_is_names[ifa->state], ifa->stub ? " (stub)" : "");
        !          1387:   cli_msg(-1015, "\tPriority: %u", ifa->priority);
        !          1388:   cli_msg(-1015, "\tCost: %u", ifa->cost);
        !          1389:   if (ifa->oa->po->ecmp)
        !          1390:     cli_msg(-1015, "\tECMP weight: %d", ((int) ifa->ecmp_weight) + 1);
        !          1391:   cli_msg(-1015, "\tHello timer: %u", ifa->helloint);
        !          1392: 
        !          1393:   if (ifa->type == OSPF_IT_NBMA)
        !          1394:   {
        !          1395:     cli_msg(-1015, "\tPoll timer: %u", ifa->pollint);
        !          1396:   }
        !          1397:   cli_msg(-1015, "\tWait timer: %u", ifa->waitint);
        !          1398:   cli_msg(-1015, "\tDead timer: %u", ifa->deadint);
        !          1399:   cli_msg(-1015, "\tRetransmit timer: %u", ifa->rxmtint);
        !          1400:   if ((ifa->type == OSPF_IT_BCAST) || (ifa->type == OSPF_IT_NBMA))
        !          1401:   {
        !          1402:     cli_msg(-1015, "\tDesignated router (ID): %R", ifa->drid);
        !          1403:     cli_msg(-1015, "\tDesignated router (IP): %I", ifa->drip);
        !          1404:     cli_msg(-1015, "\tBackup designated router (ID): %R", ifa->bdrid);
        !          1405:     cli_msg(-1015, "\tBackup designated router (IP): %I", ifa->bdrip);
        !          1406:   }
        !          1407: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>