File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / bird2 / proto / ospf / lsalib.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Mon Oct 21 16:03:56 2019 UTC (5 years, 5 months ago) by misho
Branches: bird2, MAIN
CVS tags: v2_0_7p0, HEAD
bird2 ver 2.0.7

    1: /*
    2:  *	BIRD -- OSPF
    3:  *
    4:  *	(c) 1999--2004 Ondrej Filip <feela@network.cz>
    5:  *	(c) 2009--2015 Ondrej Zajicek <santiago@crfreenet.org>
    6:  *	(c) 2009--2015 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: 
   13: #include "lib/fletcher16.h"
   14: 
   15: #define HDRLEN sizeof(struct ospf_lsa_header)
   16: 
   17: 
   18: #ifndef CPU_BIG_ENDIAN
   19: void
   20: lsa_hton_hdr(struct ospf_lsa_header *h, struct ospf_lsa_header *n)
   21: {
   22:   n->age = htons(h->age);
   23:   n->type_raw = htons(h->type_raw);
   24:   n->id = htonl(h->id);
   25:   n->rt = htonl(h->rt);
   26:   n->sn = htonl(h->sn);
   27:   n->checksum = htons(h->checksum);
   28:   n->length = htons(h->length);
   29: }
   30: 
   31: void
   32: lsa_ntoh_hdr(struct ospf_lsa_header *n, struct ospf_lsa_header *h)
   33: {
   34:   h->age = ntohs(n->age);
   35:   h->type_raw = ntohs(n->type_raw);
   36:   h->id = ntohl(n->id);
   37:   h->rt = ntohl(n->rt);
   38:   h->sn = ntohl(n->sn);
   39:   h->checksum = ntohs(n->checksum);
   40:   h->length = ntohs(n->length);
   41: }
   42: 
   43: void
   44: lsa_hton_body(void *h, void *n, u16 len)
   45: {
   46:   u32 *hid = h;
   47:   u32 *nid = n;
   48:   uint i;
   49: 
   50:   for (i = 0; i < (len / sizeof(u32)); i++)
   51:     nid[i] = htonl(hid[i]);
   52: }
   53: 
   54: void
   55: lsa_ntoh_body(void *n, void *h, u16 len)
   56: {
   57:   u32 *nid = n;
   58:   u32 *hid = h;
   59:   uint i;
   60: 
   61:   for (i = 0; i < (len / sizeof(u32)); i++)
   62:     hid[i] = ntohl(nid[i]);
   63: }
   64: #endif /* little endian */
   65: 
   66: 
   67: int
   68: lsa_flooding_allowed(u32 type, u32 domain, struct ospf_iface *ifa)
   69: {
   70:   /* Handle inactive vlinks */
   71:   if (ifa->state == OSPF_IS_DOWN)
   72:     return 0;
   73: 
   74:   /* 4.5.2 (Case 2) */
   75:   switch (LSA_SCOPE(type))
   76:   {
   77:   case LSA_SCOPE_LINK:
   78:     return ifa->iface_id == domain;
   79: 
   80:   case LSA_SCOPE_AREA:
   81:     return ifa->oa->areaid == domain;
   82: 
   83:   case LSA_SCOPE_AS:
   84:     if (ifa->type == OSPF_IT_VLINK)
   85:       return 0;
   86:     if (!oa_is_ext(ifa->oa))
   87:       return 0;
   88:     return 1;
   89: 
   90:   default:
   91:     log(L_ERR "OSPF: LSA with invalid scope");
   92:     return 0;
   93:   }
   94: }
   95: 
   96: int
   97: lsa_is_acceptable(u32 type, struct ospf_neighbor *n, struct ospf_proto *p)
   98: {
   99:   if (ospf_is_v2(p))
  100:   {
  101:     /* Do not check NSSA-LSA here, as OPT_N is only in HELLO packets */
  102: 
  103:     if (lsa_is_opaque(type))
  104:       return !!(n->options & OPT_O);
  105: 
  106:     return 1;
  107:   }
  108:   else
  109:   {
  110:     /*
  111:      * There should be check whether receiving router understands that type
  112:      * of LSA (for LSA types with U-bit == 0). But as we do not support any
  113:      * optional LSA types, this is not needed yet.
  114:      */
  115: 
  116:     return 1;
  117:   }
  118: }
  119: 
  120: static int
  121: unknown_lsa_type(u32 type)
  122: {
  123:   switch (type)
  124:   {
  125:   case LSA_T_RT:
  126:   case LSA_T_NET:
  127:   case LSA_T_SUM_NET:
  128:   case LSA_T_SUM_RT:
  129:   case LSA_T_EXT:
  130:   case LSA_T_NSSA:
  131:   case LSA_T_LINK:
  132:   case LSA_T_PREFIX:
  133:   case LSA_T_RI_LINK:
  134:   case LSA_T_RI_AREA:
  135:   case LSA_T_RI_AS:
  136:     return 0;
  137: 
  138:   default:
  139:     return 1;
  140:   }
  141: }
  142: 
  143: /* Maps OSPFv2 types to OSPFv3 types */
  144: static const u16 lsa_v2_types[] = {
  145:   0, LSA_T_RT, LSA_T_NET, LSA_T_SUM_NET, LSA_T_SUM_RT, LSA_T_EXT, 0, LSA_T_NSSA,
  146:   0, LSA_T_OPAQUE_LINK, LSA_T_OPAQUE_AREA, LSA_T_OPAQUE_AS
  147: };
  148: 
  149: /* Maps OSPFv2 opaque types to OSPFv3 function codes */
  150: static const u16 opaque_lsa_types[] = {
  151:   [LSA_OT_GR] = LSA_T_GR,
  152:   [LSA_OT_RI] = LSA_T_RI_,
  153: };
  154: 
  155: /* Maps (subset of) OSPFv3 function codes to OSPFv2 opaque types */
  156: static const u8 opaque_lsa_types_inv[] = {
  157:   [LSA_T_GR] = LSA_OT_GR,
  158:   [LSA_T_RI_] = LSA_OT_RI,
  159: };
  160: 
  161: #define LOOKUP(a, i) ({ uint _i = (i); (_i < ARRAY_SIZE(a)) ? a[_i] : 0; })
  162: 
  163: void
  164: lsa_get_type_domain_(u32 type, u32 id, struct ospf_iface *ifa, u32 *otype, u32 *domain)
  165: {
  166:   if (ospf_is_v2(ifa->oa->po))
  167:   {
  168:     type = type & LSA_T_V2_MASK;
  169:     type = LOOKUP(lsa_v2_types, type);
  170: 
  171:     uint code;
  172:     if (LSA_FUNCTION(type) == LSA_T_OPAQUE_)
  173:       if (code = LOOKUP(opaque_lsa_types, id >> 24))
  174:       {
  175: 	type = code | LSA_UBIT | LSA_SCOPE(type);
  176: 
  177: 	/* Hack for Grace-LSA: It does not use U-bit for link-scoped LSAs */
  178: 	if (type == (LSA_T_GR | LSA_UBIT))
  179: 	  type = LSA_T_GR;
  180:       }
  181:   }
  182:   else
  183:   {
  184:     /* For unkown LSAs without U-bit change scope to LSA_SCOPE_LINK */
  185:     if (unknown_lsa_type(type) && !(type & LSA_UBIT))
  186:       type = type & ~LSA_SCOPE_MASK;
  187:   }
  188: 
  189:   *otype = type;
  190: 
  191:   switch (LSA_SCOPE(type))
  192:   {
  193:   case LSA_SCOPE_LINK:
  194:     *domain = ifa->iface_id;
  195:     return;
  196: 
  197:   case LSA_SCOPE_AREA:
  198:     *domain = ifa->oa->areaid;
  199:     return;
  200: 
  201:   case LSA_SCOPE_AS:
  202:   default:
  203:     *domain = 0;
  204:     return;
  205:   }
  206: }
  207: 
  208: int
  209: lsa_is_opaque(u32 type)
  210: {
  211:   u32 fn = LSA_FUNCTION(type);
  212:   return LOOKUP(opaque_lsa_types_inv, fn) || (fn == LSA_T_OPAQUE_);
  213: }
  214: 
  215: u32
  216: lsa_get_opaque_type(u32 type)
  217: {
  218:   return LOOKUP(opaque_lsa_types_inv, LSA_FUNCTION(type));
  219: }
  220: 
  221: 
  222: void
  223: lsa_generate_checksum(struct ospf_lsa_header *lsa, const u8 *body)
  224: {
  225:   struct fletcher16_context ctx;
  226:   struct ospf_lsa_header hdr;
  227:   u16 len = lsa->length;
  228: 
  229:   /*
  230:    * lsa and body are in the host order, we need to compute Fletcher-16 checksum
  231:    * for data in the network order. We also skip the initial age field.
  232:    */
  233: 
  234:   lsa_hton_hdr(lsa, &hdr);
  235:   hdr.checksum = 0;
  236: 
  237:   fletcher16_init(&ctx);
  238:   fletcher16_update(&ctx, (u8 *) &hdr + 2, sizeof(struct ospf_lsa_header) - 2);
  239:   fletcher16_update_n32(&ctx, body, len - sizeof(struct ospf_lsa_header));
  240:   lsa->checksum = fletcher16_final(&ctx, len, OFFSETOF(struct ospf_lsa_header, checksum));
  241: }
  242: 
  243: u16
  244: lsa_verify_checksum(const void *lsa_n, int lsa_len)
  245: {
  246:   struct fletcher16_context ctx;
  247: 
  248:   /* The whole LSA is at lsa_n in net order, we just skip initial age field */
  249: 
  250:   fletcher16_init(&ctx);
  251:   fletcher16_update(&ctx, (u8 *) lsa_n + 2, lsa_len - 2);
  252: 
  253:   return fletcher16_compute(&ctx) == 0;
  254: }
  255: 
  256: 
  257: int
  258: lsa_comp(struct ospf_lsa_header *l1, struct ospf_lsa_header *l2)
  259: 			/* Return codes from point of view of l1 */
  260: {
  261:   u32 sn1, sn2;
  262: 
  263:   sn1 = l1->sn - LSA_INITSEQNO + 1;
  264:   sn2 = l2->sn - LSA_INITSEQNO + 1;
  265: 
  266:   if (sn1 > sn2)
  267:     return CMP_NEWER;
  268:   if (sn1 < sn2)
  269:     return CMP_OLDER;
  270: 
  271:   if (l1->checksum != l2->checksum)
  272:     return l1->checksum < l2->checksum ? CMP_OLDER : CMP_NEWER;
  273: 
  274:   if ((l1->age == LSA_MAXAGE) && (l2->age != LSA_MAXAGE))
  275:     return CMP_NEWER;
  276:   if ((l2->age == LSA_MAXAGE) && (l1->age != LSA_MAXAGE))
  277:     return CMP_OLDER;
  278: 
  279:   if (ABS(l1->age - l2->age) > LSA_MAXAGEDIFF)
  280:     return l1->age < l2->age ? CMP_NEWER : CMP_OLDER;
  281: 
  282:   return CMP_SAME;
  283: }
  284: 
  285: 
  286: #define LSA_TLV_LENGTH(tlv) \
  287:   (sizeof(struct ospf_tlv) + BIRD_ALIGN((tlv)->length, 4))
  288: 
  289: #define LSA_NEXT_TLV(tlv) \
  290:   ((struct ospf_tlv *) ((byte *) (tlv) + LSA_TLV_LENGTH(tlv)))
  291: 
  292: #define LSA_WALK_TLVS(tlv,buf,len)					\
  293:   for(struct ospf_tlv *tlv = (void *) (buf);				\
  294:       (byte *) tlv < (byte *) (buf) + (len);				\
  295:       tlv = LSA_NEXT_TLV(tlv))
  296: 
  297: struct ospf_tlv *
  298: lsa_get_tlv(struct top_hash_entry *en, uint type)
  299: {
  300:   LSA_WALK_TLVS(tlv, en->lsa_body, en->lsa.length - HDRLEN)
  301:     if (tlv->type == type)
  302:       return tlv;
  303: 
  304:   return NULL;
  305: }
  306: 
  307: int
  308: lsa_validate_tlvs(byte *buf, uint len)
  309: {
  310:   byte *pos = buf;
  311:   byte *end = buf + len;
  312: 
  313:   while (pos < end)
  314:   {
  315:     if ((pos + sizeof(struct ospf_tlv)) > end)
  316:       return 0;
  317: 
  318:     struct ospf_tlv *tlv = (void *) pos;
  319:     uint len = LSA_TLV_LENGTH(tlv);
  320: 
  321:     if ((pos + len) > end)
  322:       return 0;
  323: 
  324:     pos += len;
  325:   }
  326: 
  327:   return 1;
  328: }
  329: 
  330: 
  331: static inline int
  332: lsa_walk_rt2(struct ospf_lsa_rt_walk *rt)
  333: {
  334:   if (rt->buf >= rt->bufend)
  335:     return 0;
  336: 
  337:   struct ospf_lsa_rt2_link *l = rt->buf;
  338:   rt->buf += sizeof(struct ospf_lsa_rt2_link) + l->no_tos * sizeof(struct ospf_lsa_rt2_tos);
  339: 
  340:   rt->type = l->type;
  341:   rt->metric = l->metric;
  342:   rt->id = l->id;
  343:   rt->data = l->data;
  344:   return 1;
  345: }
  346: 
  347: static inline int
  348: lsa_walk_rt3(struct ospf_lsa_rt_walk *rt)
  349: {
  350:   while (rt->buf >= rt->bufend)
  351:   {
  352:     rt->en = ospf_hash_find_rt3_next(rt->en);
  353:     if (!rt->en)
  354:       return 0;
  355: 
  356:     rt->buf = rt->en->lsa_body;
  357:     rt->bufend = rt->buf + rt->en->lsa.length - sizeof(struct ospf_lsa_header);
  358:     rt->buf += sizeof(struct ospf_lsa_rt);
  359:   }
  360: 
  361:   struct ospf_lsa_rt3_link *l = rt->buf;
  362:   rt->buf += sizeof(struct ospf_lsa_rt3_link);
  363: 
  364:   rt->type = l->type;
  365:   rt->metric = l->metric;
  366:   rt->lif = l->lif;
  367:   rt->nif = l->nif;
  368:   rt->id = l->id;
  369:   return 1;
  370: }
  371: 
  372: void
  373: lsa_walk_rt_init(struct ospf_proto *p, struct top_hash_entry *act, struct ospf_lsa_rt_walk *rt)
  374: {
  375:   rt->ospf2 = ospf_is_v2(p);
  376:   rt->id = rt->data = rt->lif = rt->nif = 0;
  377: 
  378:   if (rt->ospf2)
  379:     rt->en = act;
  380:   else
  381:     rt->en = ospf_hash_find_rt3_first(p->gr, act->domain, act->lsa.rt);
  382: 
  383:   rt->buf = rt->en->lsa_body;
  384:   rt->bufend = rt->buf + rt->en->lsa.length - sizeof(struct ospf_lsa_header);
  385:   rt->buf += sizeof(struct ospf_lsa_rt);
  386: }
  387: 
  388: int
  389: lsa_walk_rt(struct ospf_lsa_rt_walk *rt)
  390: {
  391:   return rt->ospf2 ? lsa_walk_rt2(rt) : lsa_walk_rt3(rt);
  392: }
  393: 
  394: 
  395: void
  396: lsa_parse_sum_net(struct top_hash_entry *en, int ospf2, int af, net_addr *net, u8 *pxopts, u32 *metric)
  397: {
  398:   if (ospf2)
  399:   {
  400:     uint opts = lsa_get_options(&en->lsa);
  401:     struct ospf_lsa_sum2 *ls = en->lsa_body;
  402:     net_fill_ip4(net, ip4_from_u32(en->lsa.id & ls->netmask), u32_masklen(ls->netmask));
  403:     *pxopts = (opts & OPT_DN) ? OPT_PX_DN : 0;
  404:     *metric = ls->metric & LSA_METRIC_MASK;
  405:   }
  406:   else
  407:   {
  408:     struct ospf_lsa_sum3_net *ls = en->lsa_body;
  409:     ospf3_get_prefix(ls->prefix, af, net, pxopts, NULL);
  410:     *metric = ls->metric & LSA_METRIC_MASK;
  411:   }
  412: }
  413: 
  414: void
  415: lsa_parse_sum_rt(struct top_hash_entry *en, int ospf2, u32 *drid, u32 *metric, u32 *options)
  416: {
  417:   if (ospf2)
  418:   {
  419:     struct ospf_lsa_sum2 *ls = en->lsa_body;
  420:     *drid = en->lsa.id;
  421:     *metric = ls->metric & LSA_METRIC_MASK;
  422:     *options = 0;
  423:   }
  424:   else
  425:   {
  426:     struct ospf_lsa_sum3_rt *ls = en->lsa_body;
  427:     *drid = ls->drid;
  428:     *metric = ls->metric & LSA_METRIC_MASK;
  429:     *options = ls->options & LSA_OPTIONS_MASK;
  430:   }
  431: }
  432: 
  433: void
  434: lsa_parse_ext(struct top_hash_entry *en, int ospf2, int af, struct ospf_lsa_ext_local *rt)
  435: {
  436:   if (ospf2)
  437:   {
  438:     struct ospf_lsa_ext2 *ext = en->lsa_body;
  439:     net_fill_ip4(&rt->net,
  440: 		 ip4_from_u32(en->lsa.id & ext->netmask),
  441: 		 u32_masklen(ext->netmask));
  442:     rt->pxopts = 0;
  443:     rt->metric = ext->metric & LSA_METRIC_MASK;
  444:     rt->ebit = ext->metric & LSA_EXT2_EBIT;
  445: 
  446:     rt->fbit = ext->fwaddr;
  447:     rt->fwaddr = ipa_from_u32(ext->fwaddr);
  448: 
  449:     rt->tag = ext->tag;
  450:     rt->propagate = lsa_get_options(&en->lsa) & OPT_P;
  451:     rt->downwards = lsa_get_options(&en->lsa) & OPT_DN;
  452:   }
  453:   else
  454:   {
  455:     struct ospf_lsa_ext3 *ext = en->lsa_body;
  456:     u32 *buf = ospf3_get_prefix(ext->rest, af, &rt->net, &rt->pxopts, NULL);
  457:     rt->metric = ext->metric & LSA_METRIC_MASK;
  458:     rt->ebit = ext->metric & LSA_EXT3_EBIT;
  459: 
  460:     rt->fbit = ext->metric & LSA_EXT3_FBIT;
  461:     if (rt->fbit)
  462:       buf = ospf3_get_addr(buf, af, &rt->fwaddr);
  463:     else
  464:       rt->fwaddr = IPA_NONE;
  465: 
  466:     rt->tag = (ext->metric & LSA_EXT3_TBIT) ? *buf++ : 0;
  467:     rt->propagate = rt->pxopts & OPT_PX_P;
  468:     rt->downwards = rt->pxopts & OPT_PX_DN;
  469:   }
  470: }
  471: 
  472: 
  473: static int
  474: lsa_validate_rt2(struct ospf_lsa_header *lsa, struct ospf_lsa_rt *body)
  475: {
  476:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_rt)))
  477:     return 0;
  478: 
  479:   uint i = 0;
  480:   void *buf = body;
  481:   void *bufend = buf + lsa->length - HDRLEN;
  482:   buf += sizeof(struct ospf_lsa_rt);
  483: 
  484:   while (buf < bufend)
  485:   {
  486:     struct ospf_lsa_rt2_link *l = buf;
  487:     buf += sizeof(struct ospf_lsa_rt2_link) + l->no_tos * sizeof(struct ospf_lsa_rt2_tos);
  488:     i++;
  489: 
  490:     if (buf > bufend)
  491:       return 0;
  492: 
  493:     if (!((l->type == LSART_PTP) ||
  494: 	  (l->type == LSART_NET) ||
  495: 	  (l->type == LSART_STUB) ||
  496: 	  (l->type == LSART_VLNK)))
  497:       return 0;
  498:   }
  499: 
  500:   if ((body->options & LSA_RT2_LINKS) != i)
  501:     return 0;
  502: 
  503:   return 1;
  504: }
  505: 
  506: 
  507: static int
  508: lsa_validate_rt3(struct ospf_lsa_header *lsa, struct ospf_lsa_rt *body)
  509: {
  510:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_rt)))
  511:     return 0;
  512: 
  513:   void *buf = body;
  514:   void *bufend = buf + lsa->length - HDRLEN;
  515:   buf += sizeof(struct ospf_lsa_rt);
  516: 
  517:   while (buf < bufend)
  518:   {
  519:     struct ospf_lsa_rt3_link *l = buf;
  520:     buf += sizeof(struct ospf_lsa_rt3_link);
  521: 
  522:     if (buf > bufend)
  523:       return 0;
  524: 
  525:     if (!((l->type == LSART_PTP) ||
  526: 	  (l->type == LSART_NET) ||
  527: 	  (l->type == LSART_VLNK)))
  528:       return 0;
  529:   }
  530:   return 1;
  531: }
  532: 
  533: static int
  534: lsa_validate_net(struct ospf_lsa_header *lsa, struct ospf_lsa_net *body UNUSED)
  535: {
  536:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_net)))
  537:     return 0;
  538: 
  539:   return 1;
  540: }
  541: 
  542: static int
  543: lsa_validate_sum2(struct ospf_lsa_header *lsa, struct ospf_lsa_sum2 *body)
  544: {
  545:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_sum2)))
  546:     return 0;
  547: 
  548:   /* First field should have TOS = 0, we ignore other TOS fields */
  549:   if ((body->metric & LSA_SUM2_TOS) != 0)
  550:     return 0;
  551: 
  552:   return 1;
  553: }
  554: 
  555: static inline int
  556: pxlen(u32 *buf)
  557: {
  558:   return *buf >> 24;
  559: }
  560: 
  561: static int
  562: lsa_validate_sum3_net(struct ospf_lsa_header *lsa, struct ospf_lsa_sum3_net *body)
  563: {
  564:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_sum3_net) + 4))
  565:     return 0;
  566: 
  567:   u8 pxl = pxlen(body->prefix);
  568:   if (pxl > IP6_MAX_PREFIX_LENGTH)
  569:     return 0;
  570: 
  571:   if (lsa->length != (HDRLEN + sizeof(struct ospf_lsa_sum3_net) +
  572: 		      IPV6_PREFIX_SPACE(pxl)))
  573:     return 0;
  574: 
  575:   return 1;
  576: }
  577: 
  578: static int
  579: lsa_validate_sum3_rt(struct ospf_lsa_header *lsa, struct ospf_lsa_sum3_rt *body UNUSED)
  580: {
  581:   if (lsa->length != (HDRLEN + sizeof(struct ospf_lsa_sum3_rt)))
  582:     return 0;
  583: 
  584:   return 1;
  585: }
  586: 
  587: static int
  588: lsa_validate_ext2(struct ospf_lsa_header *lsa, struct ospf_lsa_ext2 *body)
  589: {
  590:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_ext2)))
  591:     return 0;
  592: 
  593:   /* First field should have TOS = 0, we ignore other TOS fields */
  594:   if ((body->metric & LSA_EXT2_TOS) != 0)
  595:     return 0;
  596: 
  597:   return 1;
  598: }
  599: 
  600: static int
  601: lsa_validate_ext3(struct ospf_lsa_header *lsa, struct ospf_lsa_ext3 *body)
  602: {
  603:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_ext3) + 4))
  604:     return 0;
  605: 
  606:   u8 pxl = pxlen(body->rest);
  607:   if (pxl > IP6_MAX_PREFIX_LENGTH)
  608:     return 0;
  609: 
  610:   int len = IPV6_PREFIX_SPACE(pxl);
  611:   if (body->metric & LSA_EXT3_FBIT) // forwarding address
  612:     len += 16;
  613:   if (body->metric & LSA_EXT3_TBIT) // route tag
  614:     len += 4;
  615:   if (*body->rest & 0xFFFF) // referenced LS type field
  616:     len += 4;
  617: 
  618:   if (lsa->length != (HDRLEN + sizeof(struct ospf_lsa_ext3) + len))
  619:     return 0;
  620: 
  621:   return 1;
  622: }
  623: 
  624: static int
  625: lsa_validate_pxlist(struct ospf_lsa_header *lsa, u32 pxcount, uint offset, u8 *pbuf)
  626: {
  627:   uint bound = lsa->length - HDRLEN - 4;
  628:   u32 i;
  629: 
  630:   for (i = 0; i < pxcount; i++)
  631:     {
  632:       if (offset > bound)
  633: 	return 0;
  634: 
  635:       u8 pxl = pxlen((u32 *) (pbuf + offset));
  636:       if (pxl > IP6_MAX_PREFIX_LENGTH)
  637: 	return 0;
  638: 
  639:       offset += IPV6_PREFIX_SPACE(pxl);
  640:     }
  641: 
  642:   if (lsa->length != (HDRLEN + offset))
  643:     return 0;
  644: 
  645:   return 1;
  646: }
  647: 
  648: static int
  649: lsa_validate_link(struct ospf_lsa_header *lsa, struct ospf_lsa_link *body)
  650: {
  651:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_link)))
  652:     return 0;
  653: 
  654:   return lsa_validate_pxlist(lsa, body->pxcount, sizeof(struct ospf_lsa_link), (u8 *) body);
  655: }
  656: 
  657: static int
  658: lsa_validate_prefix(struct ospf_lsa_header *lsa, struct ospf_lsa_prefix *body)
  659: {
  660:   if (lsa->length < (HDRLEN + sizeof(struct ospf_lsa_prefix)))
  661:     return 0;
  662: 
  663:   return lsa_validate_pxlist(lsa, body->pxcount, sizeof(struct ospf_lsa_prefix), (u8 *) body);
  664: }
  665: 
  666: static int
  667: lsa_validate_gr(struct ospf_lsa_header *lsa, void *body)
  668: {
  669:   return lsa_validate_tlvs(body, lsa->length - HDRLEN);
  670: }
  671: 
  672: static int
  673: lsa_validate_ri(struct ospf_lsa_header *lsa UNUSED, struct ospf_lsa_net *body UNUSED)
  674: {
  675:   /*
  676:    * There should be proper validation. But we do not really process RI LSAs, so
  677:    * we can just accept them like another unknown opaque LSAs.
  678:    */
  679: 
  680:   return 1;
  681: }
  682: 
  683: 
  684: /**
  685:  * lsa_validate - check whether given LSA is valid
  686:  * @lsa: LSA header
  687:  * @lsa_type: internal LSA type (%LSA_T_xxx)
  688:  * @ospf2: %true for OSPFv2, %false for OSPFv3
  689:  * @body: pointer to LSA body
  690:  *
  691:  * Checks internal structure of given LSA body (minimal length,
  692:  * consistency). Returns true if valid.
  693:  */
  694: int
  695: lsa_validate(struct ospf_lsa_header *lsa, u32 lsa_type, int ospf2, void *body)
  696: {
  697:   if (ospf2)
  698:   {
  699:     switch (lsa_type)
  700:     {
  701:     case LSA_T_RT:
  702:       return lsa_validate_rt2(lsa, body);
  703:     case LSA_T_NET:
  704:       return lsa_validate_net(lsa, body);
  705:     case LSA_T_SUM_NET:
  706:       return lsa_validate_sum2(lsa, body);
  707:     case LSA_T_SUM_RT:
  708:       return lsa_validate_sum2(lsa, body);
  709:     case LSA_T_EXT:
  710:     case LSA_T_NSSA:
  711:       return lsa_validate_ext2(lsa, body);
  712:     case LSA_T_GR:
  713:       return lsa_validate_gr(lsa, body);
  714:     case LSA_T_RI_LINK:
  715:     case LSA_T_RI_AREA:
  716:     case LSA_T_RI_AS:
  717:       return lsa_validate_ri(lsa, body);
  718:     case LSA_T_OPAQUE_LINK:
  719:     case LSA_T_OPAQUE_AREA:
  720:     case LSA_T_OPAQUE_AS:
  721:       return 1;	/* Unknown Opaque LSAs */
  722:     default:
  723:       return 0;	/* Should not happen, unknown LSAs are already rejected */
  724:     }
  725:   }
  726:   else
  727:   {
  728:     switch (lsa_type)
  729:     {
  730:     case LSA_T_RT:
  731:       return lsa_validate_rt3(lsa, body);
  732:     case LSA_T_NET:
  733:       return lsa_validate_net(lsa, body);
  734:     case LSA_T_SUM_NET:
  735:       return lsa_validate_sum3_net(lsa, body);
  736:     case LSA_T_SUM_RT:
  737:       return lsa_validate_sum3_rt(lsa, body);
  738:     case LSA_T_EXT:
  739:     case LSA_T_NSSA:
  740:       return lsa_validate_ext3(lsa, body);
  741:     case LSA_T_LINK:
  742:       return lsa_validate_link(lsa, body);
  743:     case LSA_T_PREFIX:
  744:       return lsa_validate_prefix(lsa, body);
  745:     case LSA_T_GR:
  746:       return lsa_validate_gr(lsa, body);
  747:     case LSA_T_RI_LINK:
  748:     case LSA_T_RI_AREA:
  749:     case LSA_T_RI_AS:
  750:       return lsa_validate_ri(lsa, body);
  751:     default:
  752:       return 1;	/* Unknown LSAs are OK in OSPFv3 */
  753:     }
  754:   }
  755: }

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