Annotation of embedaddon/bird2/proto/ospf/iface.c, revision 1.1.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>