Annotation of embedaddon/quagga/ospfd/ospf_opaque.c, revision 1.1

1.1     ! misho       1: /*
        !             2:  * This is an implementation of rfc2370.
        !             3:  * Copyright (C) 2001 KDD R&D Laboratories, Inc.
        !             4:  * http://www.kddlabs.co.jp/
        !             5:  *
        !             6:  * This file is part of GNU Zebra.
        !             7:  *
        !             8:  * GNU Zebra is free software; you can redistribute it and/or modify it
        !             9:  * under the terms of the GNU General Public License as published by the
        !            10:  * Free Software Foundation; either version 2, or (at your option) any
        !            11:  * later version.
        !            12:  * 
        !            13:  * GNU Zebra is distributed in the hope that it will be useful, but
        !            14:  * WITHOUT ANY WARRANTY; without even the implied warranty of
        !            15:  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
        !            16:  * General Public License for more details.
        !            17:  *
        !            18:  * You should have received a copy of the GNU General Public License
        !            19:  * along with GNU Zebra; see the file COPYING.  If not, write to the Free
        !            20:  * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
        !            21:  * 02111-1307, USA.
        !            22:  */
        !            23: 
        !            24: /***** MTYPE definitions are not reflected to "memory.h" yet. *****/
        !            25: #define MTYPE_OSPF_OPAQUE_FUNCTAB      0
        !            26: #define MTYPE_OPAQUE_INFO_PER_TYPE     0
        !            27: #define MTYPE_OPAQUE_INFO_PER_ID       0
        !            28: 
        !            29: #include <zebra.h>
        !            30: #ifdef HAVE_OPAQUE_LSA
        !            31: 
        !            32: #include "linklist.h"
        !            33: #include "prefix.h"
        !            34: #include "if.h"
        !            35: #include "table.h"
        !            36: #include "memory.h"
        !            37: #include "command.h"
        !            38: #include "vty.h"
        !            39: #include "stream.h"
        !            40: #include "log.h"
        !            41: #include "thread.h"
        !            42: #include "hash.h"
        !            43: #include "sockunion.h"         /* for inet_aton() */
        !            44: 
        !            45: #include "ospfd/ospfd.h"
        !            46: #include "ospfd/ospf_interface.h"
        !            47: #include "ospfd/ospf_ism.h"
        !            48: #include "ospfd/ospf_asbr.h"
        !            49: #include "ospfd/ospf_lsa.h"
        !            50: #include "ospfd/ospf_lsdb.h"
        !            51: #include "ospfd/ospf_neighbor.h"
        !            52: #include "ospfd/ospf_nsm.h"
        !            53: #include "ospfd/ospf_flood.h"
        !            54: #include "ospfd/ospf_packet.h"
        !            55: #include "ospfd/ospf_spf.h"
        !            56: #include "ospfd/ospf_dump.h"
        !            57: #include "ospfd/ospf_route.h"
        !            58: #include "ospfd/ospf_ase.h"
        !            59: #include "ospfd/ospf_zebra.h"
        !            60: 
        !            61: /*------------------------------------------------------------------------*
        !            62:  * Followings are initialize/terminate functions for Opaque-LSAs handling.
        !            63:  *------------------------------------------------------------------------*/
        !            64: 
        !            65: #ifdef HAVE_OSPF_TE
        !            66: #include "ospfd/ospf_te.h"
        !            67: #endif /* HAVE_OSPF_TE */
        !            68: 
        !            69: #ifdef SUPPORT_OSPF_API
        !            70: int ospf_apiserver_init (void);
        !            71: void ospf_apiserver_term (void); 
        !            72: /* Init apiserver? It's disabled by default. */
        !            73: int ospf_apiserver_enable;
        !            74: #endif /* SUPPORT_OSPF_API */
        !            75: 
        !            76: static void ospf_opaque_register_vty (void);
        !            77: static void ospf_opaque_funclist_init (void);
        !            78: static void ospf_opaque_funclist_term (void);
        !            79: static void free_opaque_info_per_type (void *val);
        !            80: static void free_opaque_info_per_id (void *val);
        !            81: static int ospf_opaque_lsa_install_hook (struct ospf_lsa *lsa);
        !            82: static int ospf_opaque_lsa_delete_hook (struct ospf_lsa *lsa);
        !            83: 
        !            84: void
        !            85: ospf_opaque_init (void)
        !            86: {
        !            87:   ospf_opaque_register_vty ();
        !            88:   ospf_opaque_funclist_init ();
        !            89: 
        !            90: #ifdef HAVE_OSPF_TE
        !            91:   if (ospf_mpls_te_init () != 0)
        !            92:     exit (1);
        !            93: #endif /* HAVE_OSPF_TE */
        !            94: 
        !            95: #ifdef SUPPORT_OSPF_API
        !            96:   if ((ospf_apiserver_enable) && (ospf_apiserver_init () != 0))
        !            97:     exit (1);
        !            98: #endif /* SUPPORT_OSPF_API */
        !            99: 
        !           100:   return;
        !           101: }
        !           102: 
        !           103: void
        !           104: ospf_opaque_term (void)
        !           105: {
        !           106: #ifdef HAVE_OSPF_TE
        !           107:   ospf_mpls_te_term ();
        !           108: #endif /* HAVE_OSPF_TE */
        !           109: 
        !           110: #ifdef SUPPORT_OSPF_API
        !           111:   ospf_apiserver_term ();
        !           112: #endif /* SUPPORT_OSPF_API */
        !           113: 
        !           114:   ospf_opaque_funclist_term ();
        !           115:   return;
        !           116: }
        !           117: 
        !           118: int
        !           119: ospf_opaque_type9_lsa_init (struct ospf_interface *oi)
        !           120: {
        !           121:   if (oi->opaque_lsa_self != NULL)
        !           122:     list_delete (oi->opaque_lsa_self);
        !           123: 
        !           124:   oi->opaque_lsa_self = list_new ();
        !           125:   oi->opaque_lsa_self->del = free_opaque_info_per_type;
        !           126:   oi->t_opaque_lsa_self = NULL;
        !           127:   return 0;
        !           128: }
        !           129: 
        !           130: void
        !           131: ospf_opaque_type9_lsa_term (struct ospf_interface *oi)
        !           132: {
        !           133:   OSPF_TIMER_OFF (oi->t_opaque_lsa_self);
        !           134:   if (oi->opaque_lsa_self != NULL)
        !           135:     list_delete (oi->opaque_lsa_self);
        !           136:   oi->opaque_lsa_self = NULL;
        !           137:   return;
        !           138: }
        !           139: 
        !           140: int
        !           141: ospf_opaque_type10_lsa_init (struct ospf_area *area)
        !           142: {
        !           143:   if (area->opaque_lsa_self != NULL)
        !           144:     list_delete (area->opaque_lsa_self);
        !           145: 
        !           146:   area->opaque_lsa_self = list_new ();
        !           147:   area->opaque_lsa_self->del = free_opaque_info_per_type;
        !           148:   area->t_opaque_lsa_self = NULL;
        !           149: 
        !           150: #ifdef MONITOR_LSDB_CHANGE
        !           151:   area->lsdb->new_lsa_hook = ospf_opaque_lsa_install_hook;
        !           152:   area->lsdb->del_lsa_hook = ospf_opaque_lsa_delete_hook;
        !           153: #endif /* MONITOR_LSDB_CHANGE */
        !           154:   return 0;
        !           155: }
        !           156: 
        !           157: void
        !           158: ospf_opaque_type10_lsa_term (struct ospf_area *area)
        !           159: {
        !           160: #ifdef MONITOR_LSDB_CHANGE
        !           161:   area->lsdb->new_lsa_hook = 
        !           162:   area->lsdb->del_lsa_hook = NULL;
        !           163: #endif /* MONITOR_LSDB_CHANGE */
        !           164: 
        !           165:   OSPF_TIMER_OFF (area->t_opaque_lsa_self);
        !           166:   if (area->opaque_lsa_self != NULL)
        !           167:     list_delete (area->opaque_lsa_self);
        !           168:   area->opaque_lsa_self = NULL;
        !           169:   return;
        !           170: }
        !           171: 
        !           172: int
        !           173: ospf_opaque_type11_lsa_init (struct ospf *top)
        !           174: {
        !           175:   if (top->opaque_lsa_self != NULL)
        !           176:     list_delete (top->opaque_lsa_self);
        !           177: 
        !           178:   top->opaque_lsa_self = list_new ();
        !           179:   top->opaque_lsa_self->del = free_opaque_info_per_type;
        !           180:   top->t_opaque_lsa_self = NULL;
        !           181: 
        !           182: #ifdef MONITOR_LSDB_CHANGE
        !           183:   top->lsdb->new_lsa_hook = ospf_opaque_lsa_install_hook;
        !           184:   top->lsdb->del_lsa_hook = ospf_opaque_lsa_delete_hook;
        !           185: #endif /* MONITOR_LSDB_CHANGE */
        !           186:   return 0;
        !           187: }
        !           188: 
        !           189: void
        !           190: ospf_opaque_type11_lsa_term (struct ospf *top)
        !           191: {
        !           192: #ifdef MONITOR_LSDB_CHANGE
        !           193:   top->lsdb->new_lsa_hook = 
        !           194:   top->lsdb->del_lsa_hook = NULL;
        !           195: #endif /* MONITOR_LSDB_CHANGE */
        !           196: 
        !           197:   OSPF_TIMER_OFF (top->t_opaque_lsa_self);
        !           198:   if (top->opaque_lsa_self != NULL)
        !           199:     list_delete (top->opaque_lsa_self);
        !           200:   top->opaque_lsa_self = NULL;
        !           201:   return;
        !           202: }
        !           203: 
        !           204: static const char *
        !           205: ospf_opaque_type_name (u_char opaque_type)
        !           206: {
        !           207:   const char *name = "Unknown";
        !           208: 
        !           209:   switch (opaque_type)
        !           210:     {
        !           211:     case OPAQUE_TYPE_WILDCARD: /* This is a special assignment! */
        !           212:       name = "Wildcard";
        !           213:       break;
        !           214:     case OPAQUE_TYPE_TRAFFIC_ENGINEERING_LSA:
        !           215:       name = "Traffic Engineering LSA";
        !           216:       break;
        !           217:     case OPAQUE_TYPE_SYCAMORE_OPTICAL_TOPOLOGY_DESC:
        !           218:       name = "Sycamore optical topology description";
        !           219:       break;
        !           220:     case OPAQUE_TYPE_GRACE_LSA:
        !           221:       name = "Grace-LSA";
        !           222:       break;
        !           223:     default:
        !           224:       if (OPAQUE_TYPE_RANGE_UNASSIGNED (opaque_type))
        !           225:         name = "Unassigned";
        !           226:       /* XXX warning: comparison is always true due to limited range of data type */
        !           227:       else if (OPAQUE_TYPE_RANGE_RESERVED (opaque_type))
        !           228:         name = "Private/Experimental";
        !           229:       break;
        !           230:     }
        !           231:   return name;
        !           232: }
        !           233: 
        !           234: /*------------------------------------------------------------------------*
        !           235:  * Followings are management functions to store user specified callbacks.
        !           236:  *------------------------------------------------------------------------*/
        !           237: 
        !           238: struct opaque_info_per_type; /* Forward declaration. */
        !           239: 
        !           240: struct ospf_opaque_functab
        !           241: {
        !           242:   u_char opaque_type;
        !           243:   struct opaque_info_per_type *oipt;
        !           244: 
        !           245:   int (* new_if_hook)(struct interface *ifp);
        !           246:   int (* del_if_hook)(struct interface *ifp);
        !           247:   void (* ism_change_hook)(struct ospf_interface *oi, int old_status);
        !           248:   void (* nsm_change_hook)(struct ospf_neighbor *nbr, int old_status);
        !           249:   void (* config_write_router)(struct vty *vty);
        !           250:   void (* config_write_if    )(struct vty *vty, struct interface *ifp);
        !           251:   void (* config_write_debug )(struct vty *vty);
        !           252:   void (* show_opaque_info   )(struct vty *vty, struct ospf_lsa *lsa);
        !           253:   int  (* lsa_originator)(void *arg);
        !           254:   struct ospf_lsa *(* lsa_refresher )(struct ospf_lsa *lsa);
        !           255:   int (* new_lsa_hook)(struct ospf_lsa *lsa);
        !           256:   int (* del_lsa_hook)(struct ospf_lsa *lsa);
        !           257: };
        !           258: 
        !           259: /* Handle LSA-9/10/11 altogether. */
        !           260: static struct list *ospf_opaque_wildcard_funclist;
        !           261: static struct list *ospf_opaque_type9_funclist;
        !           262: static struct list *ospf_opaque_type10_funclist;
        !           263: static struct list *ospf_opaque_type11_funclist;
        !           264: 
        !           265: static void
        !           266: ospf_opaque_del_functab (void *val)
        !           267: {
        !           268:   XFREE (MTYPE_OSPF_OPAQUE_FUNCTAB, val);
        !           269:   return;
        !           270: }
        !           271: 
        !           272: static void
        !           273: ospf_opaque_funclist_init (void)
        !           274: {
        !           275:   struct list *funclist;
        !           276: 
        !           277:   funclist = ospf_opaque_wildcard_funclist = list_new ();
        !           278:   funclist->del = ospf_opaque_del_functab;
        !           279: 
        !           280:   funclist = ospf_opaque_type9_funclist  = list_new ();
        !           281:   funclist->del = ospf_opaque_del_functab;
        !           282: 
        !           283:   funclist = ospf_opaque_type10_funclist = list_new ();
        !           284:   funclist->del = ospf_opaque_del_functab;
        !           285: 
        !           286:   funclist = ospf_opaque_type11_funclist = list_new ();
        !           287:   funclist->del = ospf_opaque_del_functab;
        !           288:   return;
        !           289: }
        !           290: 
        !           291: static void
        !           292: ospf_opaque_funclist_term (void)
        !           293: {
        !           294:   struct list *funclist;
        !           295: 
        !           296:   funclist = ospf_opaque_wildcard_funclist;
        !           297:   list_delete (funclist);
        !           298: 
        !           299:   funclist = ospf_opaque_type9_funclist;
        !           300:   list_delete (funclist);
        !           301: 
        !           302:   funclist = ospf_opaque_type10_funclist;
        !           303:   list_delete (funclist);
        !           304: 
        !           305:   funclist = ospf_opaque_type11_funclist;
        !           306:   list_delete (funclist);
        !           307:   return;
        !           308: }
        !           309: 
        !           310: static struct list *
        !           311: ospf_get_opaque_funclist (u_char lsa_type)
        !           312: {
        !           313:   struct list *funclist = NULL;
        !           314: 
        !           315:   switch (lsa_type)
        !           316:     {
        !           317:     case OPAQUE_TYPE_WILDCARD:
        !           318:       /* XXX
        !           319:        * This is an ugly trick to handle type-9/10/11 LSA altogether.
        !           320:        * Yes, "OPAQUE_TYPE_WILDCARD (value 0)" is not an LSA-type, nor
        !           321:        * an officially assigned opaque-type.
        !           322:        * Though it is possible that the value might be officially used
        !           323:        * in the future, we use it internally as a special label, for now.
        !           324:        */
        !           325:       funclist = ospf_opaque_wildcard_funclist;
        !           326:       break;
        !           327:     case OSPF_OPAQUE_LINK_LSA:
        !           328:       funclist = ospf_opaque_type9_funclist;
        !           329:       break;
        !           330:     case OSPF_OPAQUE_AREA_LSA:
        !           331:       funclist = ospf_opaque_type10_funclist;
        !           332:       break;
        !           333:     case OSPF_OPAQUE_AS_LSA:
        !           334:       funclist = ospf_opaque_type11_funclist;
        !           335:       break;
        !           336:     default:
        !           337:       zlog_warn ("ospf_get_opaque_funclist: Unexpected LSA-type(%u)", lsa_type);
        !           338:       break;
        !           339:     }
        !           340:   return funclist;
        !           341: }
        !           342: 
        !           343: /* XXX: such a huge argument list can /not/ be healthy... */
        !           344: int
        !           345: ospf_register_opaque_functab (
        !           346:   u_char lsa_type,
        !           347:   u_char opaque_type,
        !           348:   int (* new_if_hook)(struct interface *ifp),
        !           349:   int (* del_if_hook)(struct interface *ifp),
        !           350:   void (* ism_change_hook)(struct ospf_interface *oi, int old_status),
        !           351:   void (* nsm_change_hook)(struct ospf_neighbor *nbr, int old_status),
        !           352:   void (* config_write_router)(struct vty *vty),
        !           353:   void (* config_write_if    )(struct vty *vty, struct interface *ifp),
        !           354:   void (* config_write_debug )(struct vty *vty),
        !           355:   void (* show_opaque_info   )(struct vty *vty, struct ospf_lsa *lsa),
        !           356:   int  (* lsa_originator)(void *arg),
        !           357:   struct ospf_lsa *(* lsa_refresher )(struct ospf_lsa *lsa),
        !           358:   int (* new_lsa_hook)(struct ospf_lsa *lsa),
        !           359:   int (* del_lsa_hook)(struct ospf_lsa *lsa))
        !           360: {
        !           361:   struct list *funclist;
        !           362:   struct ospf_opaque_functab *new;
        !           363:   int rc = -1;
        !           364: 
        !           365:   if ((funclist = ospf_get_opaque_funclist (lsa_type)) == NULL)
        !           366:     {
        !           367:       zlog_warn ("ospf_register_opaque_functab: Cannot get funclist"
        !           368:                  " for Type-%u LSAs?",
        !           369:                  lsa_type);
        !           370:       goto out;
        !           371:     }
        !           372:   else
        !           373:     {
        !           374:       struct listnode *node, *nnode;
        !           375:       struct ospf_opaque_functab *functab;
        !           376:       
        !           377:       for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           378:         if (functab->opaque_type == opaque_type)
        !           379:           {
        !           380:             zlog_warn ("ospf_register_opaque_functab: Duplicated entry?:"
        !           381:                        " lsa_type(%u), opaque_type(%u)", 
        !           382:                        lsa_type, opaque_type);
        !           383:             goto out;
        !           384:           }
        !           385:     }
        !           386: 
        !           387:   if ((new = XCALLOC (MTYPE_OSPF_OPAQUE_FUNCTAB,
        !           388:                      sizeof (struct ospf_opaque_functab))) == NULL)
        !           389:     {
        !           390:       zlog_warn ("ospf_register_opaque_functab: XMALLOC: %s",
        !           391:                  safe_strerror (errno));
        !           392:       goto out;
        !           393:     }
        !           394: 
        !           395:   new->opaque_type    = opaque_type;
        !           396:   new->oipt           = NULL;
        !           397:   new->new_if_hook    = new_if_hook;
        !           398:   new->del_if_hook    = del_if_hook;
        !           399:   new->ism_change_hook     = ism_change_hook;
        !           400:   new->nsm_change_hook     = nsm_change_hook;
        !           401:   new->config_write_router = config_write_router;
        !           402:   new->config_write_if     = config_write_if;
        !           403:   new->config_write_debug  = config_write_debug;
        !           404:   new->show_opaque_info    = show_opaque_info;
        !           405:   new->lsa_originator = lsa_originator;
        !           406:   new->lsa_refresher  = lsa_refresher;
        !           407:   new->new_lsa_hook   = new_lsa_hook;
        !           408:   new->del_lsa_hook   = del_lsa_hook;
        !           409: 
        !           410:   listnode_add (funclist, new);
        !           411:   rc = 0;
        !           412: 
        !           413: out:
        !           414:   return rc;
        !           415: }
        !           416: 
        !           417: void
        !           418: ospf_delete_opaque_functab (u_char lsa_type, u_char opaque_type)
        !           419: {
        !           420:   struct list *funclist;
        !           421:   struct listnode *node, *nnode;
        !           422:   struct ospf_opaque_functab *functab;
        !           423: 
        !           424:   if ((funclist = ospf_get_opaque_funclist (lsa_type)) != NULL)
        !           425:     for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           426:       {
        !           427:         if (functab->opaque_type == opaque_type)
        !           428:           {
        !           429:             /* Cleanup internal control information, if it still remains. */
        !           430:             if (functab->oipt != NULL)
        !           431:               free_opaque_info_per_type (functab->oipt);
        !           432: 
        !           433:             /* Dequeue listnode entry from the list. */
        !           434:             listnode_delete (funclist, functab);
        !           435: 
        !           436:             /* Avoid misjudgement in the next lookup. */
        !           437:             if (listcount (funclist) == 0)
        !           438:               funclist->head = funclist->tail = NULL;
        !           439: 
        !           440:             XFREE (MTYPE_OSPF_OPAQUE_FUNCTAB, functab);
        !           441:             break;
        !           442:          }
        !           443:       }
        !           444: 
        !           445:   return;
        !           446: }
        !           447: 
        !           448: static struct ospf_opaque_functab *
        !           449: ospf_opaque_functab_lookup (struct ospf_lsa *lsa)
        !           450: {
        !           451:   struct list *funclist;
        !           452:   struct listnode *node;
        !           453:   struct ospf_opaque_functab *functab;
        !           454:   u_char key = GET_OPAQUE_TYPE (ntohl (lsa->data->id.s_addr));
        !           455: 
        !           456:   if ((funclist = ospf_get_opaque_funclist (lsa->data->type)) != NULL)
        !           457:     for (ALL_LIST_ELEMENTS_RO (funclist, node, functab))
        !           458:       if (functab->opaque_type == key)
        !           459:         return functab;
        !           460: 
        !           461:   return NULL;
        !           462: }
        !           463: 
        !           464: /*------------------------------------------------------------------------*
        !           465:  * Followings are management functions for self-originated LSA entries.
        !           466:  *------------------------------------------------------------------------*/
        !           467: 
        !           468: /*
        !           469:  * Opaque-LSA control information per opaque-type.
        !           470:  * Single Opaque-Type may have multiple instances; each of them will be
        !           471:  * identified by their opaque-id.
        !           472:  */
        !           473: struct opaque_info_per_type
        !           474: {
        !           475:   u_char lsa_type;
        !           476:   u_char opaque_type;
        !           477: 
        !           478:   enum { PROC_NORMAL, PROC_SUSPEND } status;
        !           479: 
        !           480:   /*
        !           481:    * Thread for (re-)origination scheduling for this opaque-type.
        !           482:    *
        !           483:    * Initial origination of Opaque-LSAs is controlled by generic
        !           484:    * Opaque-LSA handling module so that same opaque-type entries are
        !           485:    * called all at once when certain conditions are met.
        !           486:    * However, there might be cases that some Opaque-LSA clients need
        !           487:    * to (re-)originate their own Opaque-LSAs out-of-sync with others.
        !           488:    * This thread is prepared for that specific purpose.
        !           489:    */
        !           490:   struct thread *t_opaque_lsa_self;
        !           491: 
        !           492:   /*
        !           493:    * Backpointer to an "owner" which is LSA-type dependent.
        !           494:    *   type-9:  struct ospf_interface
        !           495:    *   type-10: struct ospf_area
        !           496:    *   type-11: struct ospf
        !           497:    */
        !           498:   void *owner;
        !           499: 
        !           500:   /* Collection of callback functions for this opaque-type. */
        !           501:   struct ospf_opaque_functab *functab;
        !           502: 
        !           503:   /* List of Opaque-LSA control informations per opaque-id. */
        !           504:   struct list *id_list;
        !           505: };
        !           506: 
        !           507: /* Opaque-LSA control information per opaque-id. */
        !           508: struct opaque_info_per_id
        !           509: {
        !           510:   u_int32_t opaque_id;
        !           511: 
        !           512:   /* Thread for refresh/flush scheduling for this opaque-type/id. */
        !           513:   struct thread *t_opaque_lsa_self;
        !           514: 
        !           515:   /* Backpointer to Opaque-LSA control information per opaque-type. */
        !           516:   struct opaque_info_per_type *opqctl_type;
        !           517: 
        !           518:   /* Here comes an actual Opaque-LSA entry for this opaque-type/id. */
        !           519:   struct ospf_lsa *lsa;
        !           520: };
        !           521: 
        !           522: static struct opaque_info_per_type *register_opaque_info_per_type (struct ospf_opaque_functab *functab, struct ospf_lsa *new);
        !           523: static struct opaque_info_per_type *lookup_opaque_info_by_type (struct ospf_lsa *lsa);
        !           524: static struct opaque_info_per_id *register_opaque_info_per_id (struct opaque_info_per_type *oipt, struct ospf_lsa *new);
        !           525: static struct opaque_info_per_id *lookup_opaque_info_by_id (struct opaque_info_per_type *oipt, struct ospf_lsa *lsa);
        !           526: static struct opaque_info_per_id *register_opaque_lsa (struct ospf_lsa *new);
        !           527: 
        !           528: 
        !           529: static struct opaque_info_per_type *
        !           530: register_opaque_info_per_type (struct ospf_opaque_functab *functab,
        !           531:                                struct ospf_lsa *new)
        !           532: {
        !           533:   struct ospf *top;
        !           534:   struct opaque_info_per_type *oipt;
        !           535: 
        !           536:   if ((oipt = XCALLOC (MTYPE_OPAQUE_INFO_PER_TYPE,
        !           537:                       sizeof (struct opaque_info_per_type))) == NULL)
        !           538:     {
        !           539:       zlog_warn ("register_opaque_info_per_type: XMALLOC: %s", safe_strerror (errno));
        !           540:       goto out;
        !           541:     }
        !           542: 
        !           543:   switch (new->data->type)
        !           544:     {
        !           545:     case OSPF_OPAQUE_LINK_LSA:
        !           546:       oipt->owner = new->oi;
        !           547:       listnode_add (new->oi->opaque_lsa_self, oipt);
        !           548:       break;
        !           549:     case OSPF_OPAQUE_AREA_LSA:
        !           550:       oipt->owner = new->area;
        !           551:       listnode_add (new->area->opaque_lsa_self, oipt);
        !           552:       break;
        !           553:     case OSPF_OPAQUE_AS_LSA:
        !           554:       top = ospf_lookup ();
        !           555:       if (new->area != NULL && (top = new->area->ospf) == NULL)
        !           556:         {
        !           557:           free_opaque_info_per_type ((void *) oipt);
        !           558:           oipt = NULL;
        !           559:           goto out; /* This case may not exist. */
        !           560:         }
        !           561:       oipt->owner = top;
        !           562:       listnode_add (top->opaque_lsa_self, oipt);
        !           563:       break;
        !           564:     default:
        !           565:       zlog_warn ("register_opaque_info_per_type: Unexpected LSA-type(%u)", new->data->type);
        !           566:       free_opaque_info_per_type ((void *) oipt);
        !           567:       oipt = NULL;
        !           568:       goto out; /* This case may not exist. */
        !           569:     }
        !           570: 
        !           571:   oipt->lsa_type = new->data->type;
        !           572:   oipt->opaque_type = GET_OPAQUE_TYPE (ntohl (new->data->id.s_addr));
        !           573:   oipt->status = PROC_NORMAL;
        !           574:   oipt->t_opaque_lsa_self = NULL;
        !           575:   oipt->functab = functab;
        !           576:   functab->oipt = oipt;
        !           577:   oipt->id_list = list_new ();
        !           578:   oipt->id_list->del = free_opaque_info_per_id;
        !           579: 
        !           580: out:
        !           581:   return oipt;
        !           582: }
        !           583: 
        !           584: static void
        !           585: free_opaque_info_per_type (void *val)
        !           586: {
        !           587:   struct opaque_info_per_type *oipt = (struct opaque_info_per_type *) val;
        !           588:   struct opaque_info_per_id *oipi;
        !           589:   struct ospf_lsa *lsa;
        !           590:   struct listnode *node, *nnode;
        !           591: 
        !           592:   /* Control information per opaque-id may still exist. */
        !           593:   for (ALL_LIST_ELEMENTS (oipt->id_list, node, nnode, oipi))
        !           594:     {
        !           595:       if ((lsa = oipi->lsa) == NULL)
        !           596:         continue;
        !           597:       if (IS_LSA_MAXAGE (lsa))
        !           598:         continue;
        !           599:       ospf_opaque_lsa_flush_schedule (lsa);
        !           600:     }
        !           601: 
        !           602:   /* Remove "oipt" from its owner's self-originated LSA list. */
        !           603:   switch (oipt->lsa_type)
        !           604:     {
        !           605:     case OSPF_OPAQUE_LINK_LSA:
        !           606:       {
        !           607:         struct ospf_interface *oi = (struct ospf_interface *)(oipt->owner);
        !           608:         listnode_delete (oi->opaque_lsa_self, oipt);
        !           609:         break;
        !           610:       }
        !           611:     case OSPF_OPAQUE_AREA_LSA:
        !           612:       {
        !           613:         struct ospf_area *area = (struct ospf_area *)(oipt->owner);
        !           614:         listnode_delete (area->opaque_lsa_self, oipt);
        !           615:         break;
        !           616:       }
        !           617:     case OSPF_OPAQUE_AS_LSA:
        !           618:       {
        !           619:         struct ospf *top = (struct ospf *)(oipt->owner);
        !           620:         listnode_delete (top->opaque_lsa_self, oipt);
        !           621:         break;
        !           622:       }
        !           623:     default:
        !           624:       zlog_warn ("free_opaque_info_per_type: Unexpected LSA-type(%u)", oipt->lsa_type);
        !           625:       break; /* This case may not exist. */
        !           626:     }
        !           627: 
        !           628:   OSPF_TIMER_OFF (oipt->t_opaque_lsa_self);
        !           629:   list_delete (oipt->id_list);
        !           630:   XFREE (MTYPE_OPAQUE_INFO_PER_TYPE, oipt);
        !           631:   return;
        !           632: }
        !           633: 
        !           634: static struct opaque_info_per_type *
        !           635: lookup_opaque_info_by_type (struct ospf_lsa *lsa)
        !           636: {
        !           637:   struct ospf *top;
        !           638:   struct ospf_area *area;
        !           639:   struct ospf_interface *oi;
        !           640:   struct list *listtop = NULL;
        !           641:   struct listnode *node, *nnode;
        !           642:   struct opaque_info_per_type *oipt = NULL;
        !           643:   u_char key = GET_OPAQUE_TYPE (ntohl (lsa->data->id.s_addr));
        !           644: 
        !           645:   switch (lsa->data->type)
        !           646:     {
        !           647:     case OSPF_OPAQUE_LINK_LSA:
        !           648:       if ((oi = lsa->oi) != NULL)
        !           649:         listtop = oi->opaque_lsa_self;
        !           650:       else
        !           651:         zlog_warn ("Type-9 Opaque-LSA: Reference to OI is missing?");
        !           652:       break;
        !           653:     case OSPF_OPAQUE_AREA_LSA:
        !           654:       if ((area = lsa->area) != NULL)
        !           655:         listtop = area->opaque_lsa_self;
        !           656:       else
        !           657:         zlog_warn ("Type-10 Opaque-LSA: Reference to AREA is missing?");
        !           658:       break;
        !           659:     case OSPF_OPAQUE_AS_LSA:
        !           660:       top = ospf_lookup ();
        !           661:       if ((area = lsa->area) != NULL && (top = area->ospf) == NULL)
        !           662:         {
        !           663:           zlog_warn ("Type-11 Opaque-LSA: Reference to OSPF is missing?");
        !           664:           break; /* Unlikely to happen. */
        !           665:         }
        !           666:       listtop = top->opaque_lsa_self;
        !           667:       break;
        !           668:     default:
        !           669:       zlog_warn ("lookup_opaque_info_by_type: Unexpected LSA-type(%u)", lsa->data->type);
        !           670:       break;
        !           671:     }
        !           672: 
        !           673:   if (listtop != NULL)
        !           674:     for (ALL_LIST_ELEMENTS (listtop, node, nnode, oipt))
        !           675:       if (oipt->opaque_type == key)
        !           676:         return oipt;
        !           677: 
        !           678:   return NULL;
        !           679: }
        !           680: 
        !           681: static struct opaque_info_per_id *
        !           682: register_opaque_info_per_id (struct opaque_info_per_type *oipt,
        !           683:                              struct ospf_lsa *new)
        !           684: {
        !           685:   struct opaque_info_per_id *oipi;
        !           686: 
        !           687:   if ((oipi = XCALLOC (MTYPE_OPAQUE_INFO_PER_ID,
        !           688:                       sizeof (struct opaque_info_per_id))) == NULL)
        !           689:     {
        !           690:       zlog_warn ("register_opaque_info_per_id: XMALLOC: %s", safe_strerror (errno));
        !           691:       goto out;
        !           692:     }
        !           693:   oipi->opaque_id = GET_OPAQUE_ID (ntohl (new->data->id.s_addr));
        !           694:   oipi->t_opaque_lsa_self = NULL;
        !           695:   oipi->opqctl_type = oipt;
        !           696:   oipi->lsa = ospf_lsa_lock (new);
        !           697: 
        !           698:   listnode_add (oipt->id_list, oipi);
        !           699: 
        !           700: out:
        !           701:   return oipi;
        !           702: }
        !           703: 
        !           704: static void
        !           705: free_opaque_info_per_id (void *val)
        !           706: {
        !           707:   struct opaque_info_per_id *oipi = (struct opaque_info_per_id *) val;
        !           708: 
        !           709:   OSPF_TIMER_OFF (oipi->t_opaque_lsa_self);
        !           710:   if (oipi->lsa != NULL)
        !           711:     ospf_lsa_unlock (&oipi->lsa);
        !           712:   XFREE (MTYPE_OPAQUE_INFO_PER_ID, oipi);
        !           713:   return;
        !           714: }
        !           715: 
        !           716: static struct opaque_info_per_id *
        !           717: lookup_opaque_info_by_id (struct opaque_info_per_type *oipt,
        !           718:                           struct ospf_lsa *lsa)
        !           719: {
        !           720:   struct listnode *node, *nnode;
        !           721:   struct opaque_info_per_id   *oipi;
        !           722:   u_int32_t key = GET_OPAQUE_ID (ntohl (lsa->data->id.s_addr));
        !           723: 
        !           724:   for (ALL_LIST_ELEMENTS (oipt->id_list, node, nnode, oipi))
        !           725:     if (oipi->opaque_id == key)
        !           726:       return oipi;
        !           727: 
        !           728:   return NULL;
        !           729: }
        !           730: 
        !           731: static struct opaque_info_per_id *
        !           732: register_opaque_lsa (struct ospf_lsa *new)
        !           733: {
        !           734:   struct ospf_opaque_functab *functab;
        !           735:   struct opaque_info_per_type *oipt;
        !           736:   struct opaque_info_per_id *oipi = NULL;
        !           737: 
        !           738:   if ((functab = ospf_opaque_functab_lookup (new)) == NULL)
        !           739:     goto out;
        !           740: 
        !           741:   if ((oipt = lookup_opaque_info_by_type (new)) == NULL
        !           742:   &&  (oipt = register_opaque_info_per_type (functab, new)) == NULL)
        !           743:     goto out;
        !           744: 
        !           745:   if ((oipi = register_opaque_info_per_id (oipt, new)) == NULL)
        !           746:     goto out;
        !           747: 
        !           748: out:
        !           749:   return oipi;
        !           750: }
        !           751: 
        !           752: /*------------------------------------------------------------------------*
        !           753:  * Followings are (vty) configuration functions for Opaque-LSAs handling.
        !           754:  *------------------------------------------------------------------------*/
        !           755: 
        !           756: DEFUN (capability_opaque,
        !           757:        capability_opaque_cmd,
        !           758:        "capability opaque",
        !           759:        "Enable specific OSPF feature\n"
        !           760:        "Opaque LSA\n")
        !           761: {
        !           762:   struct ospf *ospf = (struct ospf *) vty->index;
        !           763: 
        !           764:   /* Turn on the "master switch" of opaque-lsa capability. */
        !           765:   if (!CHECK_FLAG (ospf->config, OSPF_OPAQUE_CAPABLE))
        !           766:     {
        !           767:       if (IS_DEBUG_OSPF_EVENT)
        !           768:         zlog_debug ("Opaque capability: OFF -> ON");
        !           769: 
        !           770:       SET_FLAG (ospf->config, OSPF_OPAQUE_CAPABLE);
        !           771:       ospf_renegotiate_optional_capabilities (ospf);
        !           772:     }
        !           773:   return CMD_SUCCESS;
        !           774: }
        !           775: 
        !           776: ALIAS (capability_opaque,
        !           777:        ospf_opaque_capable_cmd,
        !           778:        "ospf opaque-lsa",
        !           779:        "OSPF specific commands\n"
        !           780:        "Enable the Opaque-LSA capability (rfc2370)\n")
        !           781: 
        !           782: DEFUN (no_capability_opaque,
        !           783:        no_capability_opaque_cmd,
        !           784:        "no capability opaque",
        !           785:        NO_STR
        !           786:        "Enable specific OSPF feature\n"
        !           787:        "Opaque LSA\n")
        !           788: {
        !           789:   struct ospf *ospf = (struct ospf *) vty->index;
        !           790: 
        !           791:   /* Turn off the "master switch" of opaque-lsa capability. */
        !           792:   if (CHECK_FLAG (ospf->config, OSPF_OPAQUE_CAPABLE))
        !           793:     {
        !           794:       if (IS_DEBUG_OSPF_EVENT)
        !           795:         zlog_debug ("Opaque capability: ON -> OFF");
        !           796: 
        !           797:       UNSET_FLAG (ospf->config, OSPF_OPAQUE_CAPABLE);
        !           798:       ospf_renegotiate_optional_capabilities (ospf);
        !           799:     }
        !           800:   return CMD_SUCCESS;
        !           801: }
        !           802: 
        !           803: ALIAS (no_capability_opaque,
        !           804:        no_ospf_opaque_capable_cmd,
        !           805:        "no ospf opaque-lsa",
        !           806:        NO_STR
        !           807:        "OSPF specific commands\n"
        !           808:        "Disable the Opaque-LSA capability (rfc2370)\n")
        !           809: 
        !           810: static void
        !           811: ospf_opaque_register_vty (void)
        !           812: {
        !           813:   install_element (OSPF_NODE, &capability_opaque_cmd);
        !           814:   install_element (OSPF_NODE, &no_capability_opaque_cmd);
        !           815:   install_element (OSPF_NODE, &ospf_opaque_capable_cmd);
        !           816:   install_element (OSPF_NODE, &no_ospf_opaque_capable_cmd);
        !           817:   return;
        !           818: }
        !           819: 
        !           820: /*------------------------------------------------------------------------*
        !           821:  * Followings are collection of user-registered function callers.
        !           822:  *------------------------------------------------------------------------*/
        !           823: 
        !           824: static int
        !           825: opaque_lsa_new_if_callback (struct list *funclist, struct interface *ifp)
        !           826: {
        !           827:   struct listnode *node, *nnode;
        !           828:   struct ospf_opaque_functab *functab;
        !           829:   int rc = -1;
        !           830: 
        !           831:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           832:     if (functab->new_if_hook != NULL)
        !           833:       if ((* functab->new_if_hook)(ifp) != 0)
        !           834:         goto out;
        !           835:   rc = 0;
        !           836: out:
        !           837:   return rc;
        !           838: }
        !           839: 
        !           840: static int
        !           841: opaque_lsa_del_if_callback (struct list *funclist, struct interface *ifp)
        !           842: {
        !           843:   struct listnode *node, *nnode;
        !           844:   struct ospf_opaque_functab *functab;
        !           845:   int rc = -1;
        !           846: 
        !           847:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           848:     if (functab->del_if_hook != NULL)
        !           849:       if ((* functab->del_if_hook)(ifp) != 0)
        !           850:         goto out;
        !           851:   rc = 0;
        !           852: out:
        !           853:   return rc;
        !           854: }
        !           855: 
        !           856: static void
        !           857: opaque_lsa_ism_change_callback (struct list *funclist,
        !           858:                                 struct ospf_interface *oi, int old_status)
        !           859: {
        !           860:   struct listnode *node, *nnode;
        !           861:   struct ospf_opaque_functab *functab;
        !           862: 
        !           863:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           864:     if (functab->ism_change_hook != NULL)
        !           865:       (* functab->ism_change_hook)(oi, old_status);
        !           866: 
        !           867:   return;
        !           868: }
        !           869: 
        !           870: static void
        !           871: opaque_lsa_nsm_change_callback (struct list *funclist,
        !           872:                                 struct ospf_neighbor *nbr, int old_status)
        !           873: {
        !           874:   struct listnode *node, *nnode;
        !           875:   struct ospf_opaque_functab *functab;
        !           876: 
        !           877:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           878:     if (functab->nsm_change_hook != NULL)
        !           879:       (* functab->nsm_change_hook)(nbr, old_status);
        !           880:   return;
        !           881: }
        !           882: 
        !           883: static void
        !           884: opaque_lsa_config_write_router_callback (struct list *funclist, 
        !           885:                                          struct vty *vty)
        !           886: {
        !           887:   struct listnode *node, *nnode;
        !           888:   struct ospf_opaque_functab *functab;
        !           889: 
        !           890:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           891:     if (functab->config_write_router != NULL)
        !           892:       (* functab->config_write_router)(vty);
        !           893:   return;
        !           894: }
        !           895: 
        !           896: static void
        !           897: opaque_lsa_config_write_if_callback (struct list *funclist,
        !           898:                                      struct vty *vty, struct interface *ifp)
        !           899: {
        !           900:   struct listnode *node, *nnode;
        !           901:   struct ospf_opaque_functab *functab;
        !           902: 
        !           903:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           904:     if (functab->config_write_if != NULL)
        !           905:       (* functab->config_write_if)(vty, ifp);
        !           906:   return;
        !           907: }
        !           908: 
        !           909: static void
        !           910: opaque_lsa_config_write_debug_callback (struct list *funclist, struct vty *vty)
        !           911: {
        !           912:   struct listnode *node, *nnode;
        !           913:   struct ospf_opaque_functab *functab;
        !           914: 
        !           915:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           916:     if (functab->config_write_debug != NULL)
        !           917:       (* functab->config_write_debug)(vty);
        !           918:   return;
        !           919: }
        !           920: 
        !           921: static int
        !           922: opaque_lsa_originate_callback (struct list *funclist, void *lsa_type_dependent)
        !           923: {
        !           924:   struct listnode *node, *nnode;
        !           925:   struct ospf_opaque_functab *functab;
        !           926:   int rc = -1;
        !           927: 
        !           928:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           929:     if (functab->lsa_originator != NULL)
        !           930:       if ((* functab->lsa_originator)(lsa_type_dependent) != 0)
        !           931:          goto out;
        !           932:   rc = 0;
        !           933: out:
        !           934:   return rc;
        !           935: }
        !           936: 
        !           937: static int
        !           938: new_lsa_callback (struct list *funclist, struct ospf_lsa *lsa)
        !           939: {
        !           940:   struct listnode *node, *nnode;
        !           941:   struct ospf_opaque_functab *functab;
        !           942:   int rc = -1;
        !           943: 
        !           944:   /* This function handles ALL types of LSAs, not only opaque ones. */
        !           945:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           946:     if (functab->new_lsa_hook != NULL)
        !           947:       if ((* functab->new_lsa_hook)(lsa) != 0)
        !           948:         goto out;
        !           949:   rc = 0;
        !           950: out:
        !           951:   return rc;
        !           952: }
        !           953: 
        !           954: static int
        !           955: del_lsa_callback (struct list *funclist, struct ospf_lsa *lsa)
        !           956: {
        !           957:   struct listnode *node, *nnode;
        !           958:   struct ospf_opaque_functab *functab;
        !           959:   int rc = -1;
        !           960: 
        !           961:   /* This function handles ALL types of LSAs, not only opaque ones. */
        !           962:   for (ALL_LIST_ELEMENTS (funclist, node, nnode, functab))
        !           963:     if (functab->del_lsa_hook != NULL)
        !           964:       if ((* functab->del_lsa_hook)(lsa) != 0)
        !           965:         goto out;
        !           966:   rc = 0;
        !           967: out:
        !           968:   return rc;
        !           969: }
        !           970: 
        !           971: /*------------------------------------------------------------------------*
        !           972:  * Followings are glue functions to call Opaque-LSA specific processing.
        !           973:  *------------------------------------------------------------------------*/
        !           974: 
        !           975: int
        !           976: ospf_opaque_new_if (struct interface *ifp)
        !           977: {
        !           978:   struct list *funclist;
        !           979:   int rc = -1;
        !           980: 
        !           981:   funclist = ospf_opaque_wildcard_funclist;
        !           982:   if (opaque_lsa_new_if_callback (funclist, ifp) != 0)
        !           983:     goto out;
        !           984: 
        !           985:   funclist = ospf_opaque_type9_funclist;
        !           986:   if (opaque_lsa_new_if_callback (funclist, ifp) != 0)
        !           987:     goto out;
        !           988: 
        !           989:   funclist = ospf_opaque_type10_funclist;
        !           990:   if (opaque_lsa_new_if_callback (funclist, ifp) != 0)
        !           991:     goto out;
        !           992: 
        !           993:   funclist = ospf_opaque_type11_funclist;
        !           994:   if (opaque_lsa_new_if_callback (funclist, ifp) != 0)
        !           995:     goto out;
        !           996: 
        !           997:   rc = 0;
        !           998: out:
        !           999:   return rc;
        !          1000: }
        !          1001: 
        !          1002: int
        !          1003: ospf_opaque_del_if (struct interface *ifp)
        !          1004: {
        !          1005:   struct list *funclist;
        !          1006:   int rc = -1;
        !          1007: 
        !          1008:   funclist = ospf_opaque_wildcard_funclist;
        !          1009:   if (opaque_lsa_del_if_callback (funclist, ifp) != 0)
        !          1010:     goto out;
        !          1011: 
        !          1012:   funclist = ospf_opaque_type9_funclist;
        !          1013:   if (opaque_lsa_del_if_callback (funclist, ifp) != 0)
        !          1014:     goto out;
        !          1015: 
        !          1016:   funclist = ospf_opaque_type10_funclist;
        !          1017:   if (opaque_lsa_del_if_callback (funclist, ifp) != 0)
        !          1018:     goto out;
        !          1019: 
        !          1020:   funclist = ospf_opaque_type11_funclist;
        !          1021:   if (opaque_lsa_del_if_callback (funclist, ifp) != 0)
        !          1022:     goto out;
        !          1023: 
        !          1024:   rc = 0;
        !          1025: out:
        !          1026:   return rc;
        !          1027: }
        !          1028: 
        !          1029: void
        !          1030: ospf_opaque_ism_change (struct ospf_interface *oi, int old_status)
        !          1031: {
        !          1032:   struct list *funclist;
        !          1033: 
        !          1034:   funclist = ospf_opaque_wildcard_funclist;
        !          1035:   opaque_lsa_ism_change_callback (funclist, oi, old_status);
        !          1036: 
        !          1037:   funclist = ospf_opaque_type9_funclist;
        !          1038:   opaque_lsa_ism_change_callback (funclist, oi, old_status);
        !          1039: 
        !          1040:   funclist = ospf_opaque_type10_funclist;
        !          1041:   opaque_lsa_ism_change_callback (funclist, oi, old_status);
        !          1042: 
        !          1043:   funclist = ospf_opaque_type11_funclist;
        !          1044:   opaque_lsa_ism_change_callback (funclist, oi, old_status);
        !          1045: 
        !          1046:   return;
        !          1047: }
        !          1048: 
        !          1049: void
        !          1050: ospf_opaque_nsm_change (struct ospf_neighbor *nbr, int old_state)
        !          1051: {
        !          1052:   struct ospf *top;
        !          1053:   struct list *funclist;
        !          1054: 
        !          1055:   if ((top = oi_to_top (nbr->oi)) == NULL)
        !          1056:     goto out;
        !          1057: 
        !          1058:   if (old_state != NSM_Full && nbr->state == NSM_Full)
        !          1059:     {
        !          1060:       if (CHECK_FLAG (nbr->options, OSPF_OPTION_O))
        !          1061:         {
        !          1062:           if (! CHECK_FLAG (top->opaque, OPAQUE_OPERATION_READY_BIT))
        !          1063:             {
        !          1064:               if (IS_DEBUG_OSPF_EVENT)
        !          1065:                 zlog_debug ("Opaque-LSA: Now get operational!");
        !          1066: 
        !          1067:               SET_FLAG (top->opaque, OPAQUE_OPERATION_READY_BIT);
        !          1068:             }
        !          1069: 
        !          1070:           ospf_opaque_lsa_originate_schedule (nbr->oi, NULL);
        !          1071:         }
        !          1072:     }
        !          1073:   else
        !          1074:   if (old_state == NSM_Full && nbr->state != NSM_Full)
        !          1075:     {
        !          1076: #ifdef NOTYET
        !          1077:       /*
        !          1078:        * If no more opaque-capable full-state neighbor remains in the
        !          1079:        * flooding scope which corresponds to Opaque-LSA type, periodic
        !          1080:        * LS flooding should be stopped.
        !          1081:        */
        !          1082: #endif /* NOTYET */
        !          1083:       ;
        !          1084:     }
        !          1085: 
        !          1086:   funclist = ospf_opaque_wildcard_funclist;
        !          1087:   opaque_lsa_nsm_change_callback (funclist, nbr, old_state);
        !          1088: 
        !          1089:   funclist = ospf_opaque_type9_funclist;
        !          1090:   opaque_lsa_nsm_change_callback (funclist, nbr, old_state);
        !          1091: 
        !          1092:   funclist = ospf_opaque_type10_funclist;
        !          1093:   opaque_lsa_nsm_change_callback (funclist, nbr, old_state);
        !          1094: 
        !          1095:   funclist = ospf_opaque_type11_funclist;
        !          1096:   opaque_lsa_nsm_change_callback (funclist, nbr, old_state);
        !          1097: 
        !          1098: out:
        !          1099:   return;
        !          1100: }
        !          1101: 
        !          1102: void
        !          1103: ospf_opaque_config_write_router (struct vty *vty, struct ospf *ospf)
        !          1104: {
        !          1105:   struct list *funclist;
        !          1106: 
        !          1107:   if (CHECK_FLAG (ospf->config, OSPF_OPAQUE_CAPABLE))
        !          1108:     vty_out (vty, " capability opaque%s", VTY_NEWLINE);
        !          1109: 
        !          1110:   funclist = ospf_opaque_wildcard_funclist;
        !          1111:   opaque_lsa_config_write_router_callback (funclist, vty);
        !          1112: 
        !          1113:   funclist = ospf_opaque_type9_funclist;
        !          1114:   opaque_lsa_config_write_router_callback (funclist, vty);
        !          1115: 
        !          1116:   funclist = ospf_opaque_type10_funclist;
        !          1117:   opaque_lsa_config_write_router_callback (funclist, vty);
        !          1118: 
        !          1119:   funclist = ospf_opaque_type11_funclist;
        !          1120:   opaque_lsa_config_write_router_callback (funclist, vty);
        !          1121: 
        !          1122:   return;
        !          1123: }
        !          1124: 
        !          1125: void
        !          1126: ospf_opaque_config_write_if (struct vty *vty, struct interface *ifp)
        !          1127: {
        !          1128:   struct list *funclist;
        !          1129: 
        !          1130:   funclist = ospf_opaque_wildcard_funclist;
        !          1131:   opaque_lsa_config_write_if_callback (funclist, vty, ifp);
        !          1132: 
        !          1133:   funclist = ospf_opaque_type9_funclist;
        !          1134:   opaque_lsa_config_write_if_callback (funclist, vty, ifp);
        !          1135: 
        !          1136:   funclist = ospf_opaque_type10_funclist;
        !          1137:   opaque_lsa_config_write_if_callback (funclist, vty, ifp);
        !          1138: 
        !          1139:   funclist = ospf_opaque_type11_funclist;
        !          1140:   opaque_lsa_config_write_if_callback (funclist, vty, ifp);
        !          1141: 
        !          1142:   return;
        !          1143: }
        !          1144: 
        !          1145: void
        !          1146: ospf_opaque_config_write_debug (struct vty *vty)
        !          1147: {
        !          1148:   struct list *funclist;
        !          1149: 
        !          1150:   funclist = ospf_opaque_wildcard_funclist;
        !          1151:   opaque_lsa_config_write_debug_callback (funclist, vty);
        !          1152: 
        !          1153:   funclist = ospf_opaque_type9_funclist;
        !          1154:   opaque_lsa_config_write_debug_callback (funclist, vty);
        !          1155: 
        !          1156:   funclist = ospf_opaque_type10_funclist;
        !          1157:   opaque_lsa_config_write_debug_callback (funclist, vty);
        !          1158: 
        !          1159:   funclist = ospf_opaque_type11_funclist;
        !          1160:   opaque_lsa_config_write_debug_callback (funclist, vty);
        !          1161: 
        !          1162:   return;
        !          1163: }
        !          1164: 
        !          1165: void
        !          1166: show_opaque_info_detail (struct vty *vty, struct ospf_lsa *lsa)
        !          1167: {
        !          1168:   struct lsa_header *lsah = (struct lsa_header *) lsa->data;
        !          1169:   u_int32_t lsid = ntohl (lsah->id.s_addr);
        !          1170:   u_char    opaque_type = GET_OPAQUE_TYPE (lsid);
        !          1171:   u_int32_t opaque_id = GET_OPAQUE_ID (lsid);
        !          1172:   struct ospf_opaque_functab *functab;
        !          1173: 
        !          1174:   /* Switch output functionality by vty address. */
        !          1175:   if (vty != NULL)
        !          1176:     {
        !          1177:       vty_out (vty, "  Opaque-Type %u (%s)%s", opaque_type,
        !          1178:               ospf_opaque_type_name (opaque_type), VTY_NEWLINE);
        !          1179:       vty_out (vty, "  Opaque-ID   0x%x%s", opaque_id, VTY_NEWLINE);
        !          1180: 
        !          1181:       vty_out (vty, "  Opaque-Info: %u octets of data%s%s",
        !          1182:                ntohs (lsah->length) - OSPF_LSA_HEADER_SIZE,
        !          1183:                VALID_OPAQUE_INFO_LEN(lsah) ? "" : "(Invalid length?)",
        !          1184:                VTY_NEWLINE);
        !          1185:     }
        !          1186:   else
        !          1187:     {
        !          1188:       zlog_debug ("    Opaque-Type %u (%s)", opaque_type,
        !          1189:                 ospf_opaque_type_name (opaque_type));
        !          1190:       zlog_debug ("    Opaque-ID   0x%x", opaque_id);
        !          1191: 
        !          1192:       zlog_debug ("    Opaque-Info: %u octets of data%s",
        !          1193:                ntohs (lsah->length) - OSPF_LSA_HEADER_SIZE,
        !          1194:                VALID_OPAQUE_INFO_LEN(lsah) ? "" : "(Invalid length?)");
        !          1195:     }
        !          1196: 
        !          1197:   /* Call individual output functions. */
        !          1198:   if ((functab = ospf_opaque_functab_lookup (lsa)) != NULL)
        !          1199:     if (functab->show_opaque_info != NULL)
        !          1200:       (* functab->show_opaque_info)(vty, lsa);
        !          1201: 
        !          1202:   return;
        !          1203: }
        !          1204: 
        !          1205: void
        !          1206: ospf_opaque_lsa_dump (struct stream *s, u_int16_t length)
        !          1207: {
        !          1208:   struct ospf_lsa lsa;
        !          1209: 
        !          1210:   lsa.data = (struct lsa_header *) STREAM_PNT (s);
        !          1211:   show_opaque_info_detail (NULL, &lsa);
        !          1212:   return;
        !          1213: }
        !          1214: 
        !          1215: static int
        !          1216: ospf_opaque_lsa_install_hook (struct ospf_lsa *lsa)
        !          1217: {
        !          1218:   struct list *funclist;
        !          1219:   int rc = -1;
        !          1220: 
        !          1221:   /*
        !          1222:    * Some Opaque-LSA user may want to monitor every LSA installation
        !          1223:    * into the LSDB, regardless with target LSA type.
        !          1224:    */
        !          1225:   funclist = ospf_opaque_wildcard_funclist;
        !          1226:   if (new_lsa_callback (funclist, lsa) != 0)
        !          1227:     goto out;
        !          1228: 
        !          1229:   funclist = ospf_opaque_type9_funclist;
        !          1230:   if (new_lsa_callback (funclist, lsa) != 0)
        !          1231:     goto out;
        !          1232: 
        !          1233:   funclist = ospf_opaque_type10_funclist;
        !          1234:   if (new_lsa_callback (funclist, lsa) != 0)
        !          1235:     goto out;
        !          1236: 
        !          1237:   funclist = ospf_opaque_type11_funclist;
        !          1238:   if (new_lsa_callback (funclist, lsa) != 0)
        !          1239:     goto out;
        !          1240: 
        !          1241:   rc = 0;
        !          1242: out:
        !          1243:   return rc;
        !          1244: }
        !          1245: 
        !          1246: static int
        !          1247: ospf_opaque_lsa_delete_hook (struct ospf_lsa *lsa)
        !          1248: {
        !          1249:   struct list *funclist;
        !          1250:   int rc = -1;
        !          1251: 
        !          1252:   /*
        !          1253:    * Some Opaque-LSA user may want to monitor every LSA deletion
        !          1254:    * from the LSDB, regardless with target LSA type.
        !          1255:    */
        !          1256:   funclist = ospf_opaque_wildcard_funclist;
        !          1257:   if (del_lsa_callback (funclist, lsa) != 0)
        !          1258:     goto out;
        !          1259: 
        !          1260:   funclist = ospf_opaque_type9_funclist;
        !          1261:   if (del_lsa_callback (funclist, lsa) != 0)
        !          1262:     goto out;
        !          1263: 
        !          1264:   funclist = ospf_opaque_type10_funclist;
        !          1265:   if (del_lsa_callback (funclist, lsa) != 0)
        !          1266:     goto out;
        !          1267: 
        !          1268:   funclist = ospf_opaque_type11_funclist;
        !          1269:   if (del_lsa_callback (funclist, lsa) != 0)
        !          1270:     goto out;
        !          1271: 
        !          1272:   rc = 0;
        !          1273: out:
        !          1274:   return rc;
        !          1275: }
        !          1276: 
        !          1277: /*------------------------------------------------------------------------*
        !          1278:  * Followings are Opaque-LSA origination/refresh management functions.
        !          1279:  *------------------------------------------------------------------------*/
        !          1280: 
        !          1281: static int ospf_opaque_type9_lsa_originate (struct thread *t);
        !          1282: static int ospf_opaque_type10_lsa_originate (struct thread *t);
        !          1283: static int ospf_opaque_type11_lsa_originate (struct thread *t);
        !          1284: static void ospf_opaque_lsa_reoriginate_resume (struct list *listtop, void *arg);
        !          1285: 
        !          1286: void
        !          1287: ospf_opaque_lsa_originate_schedule (struct ospf_interface *oi, int *delay0)
        !          1288: {
        !          1289:   struct ospf *top;
        !          1290:   struct ospf_area *area;
        !          1291:   struct listnode *node, *nnode;
        !          1292:   struct opaque_info_per_type *oipt;
        !          1293:   int delay = 0;
        !          1294: 
        !          1295:   if ((top = oi_to_top (oi)) == NULL || (area = oi->area) == NULL)
        !          1296:     {
        !          1297:       zlog_warn ("ospf_opaque_lsa_originate_schedule: Invalid argument?");
        !          1298:       goto out;
        !          1299:     }
        !          1300: 
        !          1301:   /* It may not a right time to schedule origination now. */
        !          1302:   if (! CHECK_FLAG (top->opaque, OPAQUE_OPERATION_READY_BIT))
        !          1303:     {
        !          1304:       if (IS_DEBUG_OSPF_EVENT)
        !          1305:         zlog_debug ("ospf_opaque_lsa_originate_schedule: Not operational.");
        !          1306:       goto out; /* This is not an error. */
        !          1307:     }
        !          1308:   if (IS_OPAQUE_LSA_ORIGINATION_BLOCKED (top->opaque))
        !          1309:     {
        !          1310:       if (IS_DEBUG_OSPF_EVENT)
        !          1311:         zlog_debug ("ospf_opaque_lsa_originate_schedule: Under blockade.");
        !          1312:       goto out; /* This is not an error, too. */
        !          1313:     }
        !          1314: 
        !          1315:   if (delay0 != NULL)
        !          1316:     delay = *delay0;
        !          1317: 
        !          1318:   /*
        !          1319:    * There might be some entries that have been waiting for triggering
        !          1320:    * of per opaque-type re-origination get resumed.
        !          1321:    */
        !          1322:   ospf_opaque_lsa_reoriginate_resume (  oi->opaque_lsa_self, (void *)   oi);
        !          1323:   ospf_opaque_lsa_reoriginate_resume (area->opaque_lsa_self, (void *) area);
        !          1324:   ospf_opaque_lsa_reoriginate_resume ( top->opaque_lsa_self, (void *)  top);
        !          1325: 
        !          1326:   /*
        !          1327:    * Now, schedule origination of all Opaque-LSAs per opaque-type.
        !          1328:    */
        !          1329:   if (! list_isempty (ospf_opaque_type9_funclist)
        !          1330:   &&    list_isempty (oi->opaque_lsa_self)
        !          1331:   &&    oi->t_opaque_lsa_self == NULL)
        !          1332:     {
        !          1333:       if (IS_DEBUG_OSPF_EVENT)
        !          1334:         zlog_debug ("Schedule Type-9 Opaque-LSA origination in %d sec later.", delay);
        !          1335:       oi->t_opaque_lsa_self =
        !          1336:        thread_add_timer (master, ospf_opaque_type9_lsa_originate, oi, delay);
        !          1337:       delay += OSPF_MIN_LS_INTERVAL;
        !          1338:     }
        !          1339: 
        !          1340:   if (! list_isempty (ospf_opaque_type10_funclist)
        !          1341:   &&    list_isempty (area->opaque_lsa_self)
        !          1342:   &&    area->t_opaque_lsa_self == NULL)
        !          1343:     {
        !          1344:       /*
        !          1345:        * One AREA may contain multiple OIs, but above 2nd and 3rd
        !          1346:        * conditions prevent from scheduling the originate function
        !          1347:        * again and again.
        !          1348:        */
        !          1349:       if (IS_DEBUG_OSPF_EVENT)
        !          1350:         zlog_debug ("Schedule Type-10 Opaque-LSA origination in %d sec later.", delay);
        !          1351:       area->t_opaque_lsa_self =
        !          1352:         thread_add_timer (master, ospf_opaque_type10_lsa_originate,
        !          1353:                           area, delay);
        !          1354:       delay += OSPF_MIN_LS_INTERVAL;
        !          1355:     }
        !          1356: 
        !          1357:   if (! list_isempty (ospf_opaque_type11_funclist)
        !          1358:   &&    list_isempty (top->opaque_lsa_self)
        !          1359:   &&    top->t_opaque_lsa_self == NULL)
        !          1360:     {
        !          1361:       /*
        !          1362:        * One OSPF may contain multiple AREAs, but above 2nd and 3rd
        !          1363:        * conditions prevent from scheduling the originate function
        !          1364:        * again and again.
        !          1365:        */
        !          1366:       if (IS_DEBUG_OSPF_EVENT)
        !          1367:         zlog_debug ("Schedule Type-11 Opaque-LSA origination in %d sec later.", delay);
        !          1368:       top->t_opaque_lsa_self =
        !          1369:         thread_add_timer (master, ospf_opaque_type11_lsa_originate,
        !          1370:                           top, delay);
        !          1371:       delay += OSPF_MIN_LS_INTERVAL;
        !          1372:     }
        !          1373: 
        !          1374:   /*
        !          1375:    * Following section treats a special situation that this node's
        !          1376:    * opaque capability has changed as "ON -> OFF -> ON".
        !          1377:    */
        !          1378:   if (! list_isempty (ospf_opaque_type9_funclist)
        !          1379:   &&  ! list_isempty (oi->opaque_lsa_self))
        !          1380:     {
        !          1381:       for (ALL_LIST_ELEMENTS (oi->opaque_lsa_self, node, nnode, oipt))
        !          1382:         {
        !          1383:          /* 
        !          1384:           * removed the test for
        !          1385:           *   (! list_isempty (oipt->id_list))   * Handler is already active. *
        !          1386:            * because opaque cababilities ON -> OFF -> ON result in list_isempty (oipt->id_list)
        !          1387:           * not being empty.
        !          1388:           */
        !          1389:           if (oipt->t_opaque_lsa_self != NULL /* Waiting for a thread call. */
        !          1390:               || oipt->status == PROC_SUSPEND)   /* Cannot originate now. */
        !          1391:               continue;
        !          1392: 
        !          1393:           ospf_opaque_lsa_reoriginate_schedule ((void *) oi,
        !          1394:             OSPF_OPAQUE_LINK_LSA, oipt->opaque_type);
        !          1395:         }
        !          1396:     }
        !          1397: 
        !          1398:   if (! list_isempty (ospf_opaque_type10_funclist)
        !          1399:   &&  ! list_isempty (area->opaque_lsa_self))
        !          1400:     {
        !          1401:       for (ALL_LIST_ELEMENTS (area->opaque_lsa_self, node, nnode, oipt))
        !          1402:         {
        !          1403:          /* 
        !          1404:           * removed the test for
        !          1405:           *   (! list_isempty (oipt->id_list))   * Handler is already active. *
        !          1406:            * because opaque cababilities ON -> OFF -> ON result in list_isempty (oipt->id_list)
        !          1407:           * not being empty.
        !          1408:           */
        !          1409:           if (oipt->t_opaque_lsa_self != NULL /* Waiting for a thread call. */
        !          1410:               || oipt->status == PROC_SUSPEND)   /* Cannot originate now. */
        !          1411:             continue;
        !          1412: 
        !          1413:           ospf_opaque_lsa_reoriginate_schedule ((void *) area,
        !          1414:             OSPF_OPAQUE_AREA_LSA, oipt->opaque_type);
        !          1415:         }
        !          1416:     }
        !          1417: 
        !          1418:   if (! list_isempty (ospf_opaque_type11_funclist)
        !          1419:   &&  ! list_isempty (top->opaque_lsa_self))
        !          1420:     {
        !          1421:       for (ALL_LIST_ELEMENTS (top->opaque_lsa_self, node, nnode, oipt))
        !          1422:         {
        !          1423:          /* 
        !          1424:           * removed the test for
        !          1425:           *   (! list_isempty (oipt->id_list))   * Handler is already active. *
        !          1426:            * because opaque cababilities ON -> OFF -> ON result in list_isempty (oipt->id_list)
        !          1427:           * not being empty.
        !          1428:           */
        !          1429:           if (oipt->t_opaque_lsa_self != NULL /* Waiting for a thread call. */
        !          1430:               || oipt->status == PROC_SUSPEND)   /* Cannot originate now. */
        !          1431:             continue;
        !          1432: 
        !          1433:           ospf_opaque_lsa_reoriginate_schedule ((void *) top,
        !          1434:             OSPF_OPAQUE_AS_LSA, oipt->opaque_type);
        !          1435:         }
        !          1436:     }
        !          1437: 
        !          1438:   if (delay0 != NULL)
        !          1439:     *delay0 = delay;
        !          1440: 
        !          1441: out:
        !          1442:   return;
        !          1443: }
        !          1444: 
        !          1445: static int
        !          1446: ospf_opaque_type9_lsa_originate (struct thread *t)
        !          1447: {
        !          1448:   struct ospf_interface *oi;
        !          1449:   int rc;
        !          1450: 
        !          1451:   oi = THREAD_ARG (t);
        !          1452:   oi->t_opaque_lsa_self = NULL;
        !          1453: 
        !          1454:   if (IS_DEBUG_OSPF_EVENT)
        !          1455:     zlog_debug ("Timer[Type9-LSA]: Originate Opaque-LSAs for OI %s",
        !          1456:                 IF_NAME (oi));
        !          1457: 
        !          1458:   rc = opaque_lsa_originate_callback (ospf_opaque_type9_funclist, oi);
        !          1459: 
        !          1460:   return rc;
        !          1461: }
        !          1462: 
        !          1463: static int
        !          1464: ospf_opaque_type10_lsa_originate (struct thread *t)
        !          1465: {
        !          1466:   struct ospf_area *area;
        !          1467:   int rc;
        !          1468: 
        !          1469:   area = THREAD_ARG (t);
        !          1470:   area->t_opaque_lsa_self = NULL;
        !          1471: 
        !          1472:   if (IS_DEBUG_OSPF_EVENT)
        !          1473:     zlog_debug ("Timer[Type10-LSA]: Originate Opaque-LSAs for Area %s",
        !          1474:                 inet_ntoa (area->area_id));
        !          1475: 
        !          1476:   rc = opaque_lsa_originate_callback (ospf_opaque_type10_funclist, area);
        !          1477: 
        !          1478:   return rc;
        !          1479: }
        !          1480: 
        !          1481: static int
        !          1482: ospf_opaque_type11_lsa_originate (struct thread *t)
        !          1483: {
        !          1484:   struct ospf *top;
        !          1485:   int rc;
        !          1486: 
        !          1487:   top = THREAD_ARG (t);
        !          1488:   top->t_opaque_lsa_self = NULL;
        !          1489: 
        !          1490:   if (IS_DEBUG_OSPF_EVENT)
        !          1491:     zlog_debug ("Timer[Type11-LSA]: Originate AS-External Opaque-LSAs");
        !          1492: 
        !          1493:   rc = opaque_lsa_originate_callback (ospf_opaque_type11_funclist, top);
        !          1494: 
        !          1495:   return rc;
        !          1496: }
        !          1497: 
        !          1498: static void
        !          1499: ospf_opaque_lsa_reoriginate_resume (struct list *listtop, void *arg)
        !          1500: {
        !          1501:   struct listnode *node, *nnode;
        !          1502:   struct opaque_info_per_type *oipt;
        !          1503:   struct ospf_opaque_functab *functab;
        !          1504: 
        !          1505:   if (listtop == NULL)
        !          1506:     goto out;
        !          1507: 
        !          1508:   /*
        !          1509:    * Pickup oipt entries those which in SUSPEND status, and give
        !          1510:    * them a chance to start re-origination now.
        !          1511:    */
        !          1512:   for (ALL_LIST_ELEMENTS (listtop, node, nnode, oipt))
        !          1513:     {
        !          1514:       if (oipt->status != PROC_SUSPEND)
        !          1515:           continue;
        !          1516: 
        !          1517:       oipt->status = PROC_NORMAL;
        !          1518: 
        !          1519:       if ((functab = oipt->functab) == NULL
        !          1520:           || functab->lsa_originator  == NULL)
        !          1521:         continue;
        !          1522: 
        !          1523:       if ((* functab->lsa_originator)(arg) != 0)
        !          1524:         {
        !          1525:           zlog_warn ("ospf_opaque_lsa_reoriginate_resume: Failed (opaque-type=%u)", oipt->opaque_type);
        !          1526:           continue;
        !          1527:         }
        !          1528:     }
        !          1529: 
        !          1530: out:
        !          1531:   return;
        !          1532: }
        !          1533: 
        !          1534: struct ospf_lsa *
        !          1535: ospf_opaque_lsa_install (struct ospf_lsa *lsa, int rt_recalc)
        !          1536: {
        !          1537:   struct ospf_lsa *new = NULL;
        !          1538:   struct opaque_info_per_type *oipt;
        !          1539:   struct opaque_info_per_id *oipi;
        !          1540:   struct ospf *top;
        !          1541: 
        !          1542:   /* Don't take "rt_recalc" into consideration for now. *//* XXX */
        !          1543: 
        !          1544:   if (! IS_LSA_SELF (lsa))
        !          1545:     {
        !          1546:       new = lsa; /* Don't touch this LSA. */
        !          1547:       goto out;
        !          1548:     }
        !          1549: 
        !          1550:   if (IS_DEBUG_OSPF (lsa, LSA_INSTALL))
        !          1551:     zlog_debug ("Install Type-%u Opaque-LSA: [opaque-type=%u, opaque-id=%x]", lsa->data->type, GET_OPAQUE_TYPE (ntohl (lsa->data->id.s_addr)), GET_OPAQUE_ID (ntohl (lsa->data->id.s_addr)));
        !          1552: 
        !          1553:   /* Replace the existing lsa with the new one. */
        !          1554:   if ((oipt = lookup_opaque_info_by_type (lsa)) != NULL
        !          1555:       && (oipi = lookup_opaque_info_by_id (oipt, lsa)) != NULL)
        !          1556:     {
        !          1557:       ospf_lsa_unlock (&oipi->lsa);
        !          1558:       oipi->lsa = ospf_lsa_lock (lsa);
        !          1559:     }
        !          1560:   /* Register the new lsa entry and get its control info. */
        !          1561:   else
        !          1562:   if ((oipi = register_opaque_lsa (lsa)) == NULL)
        !          1563:     {
        !          1564:       zlog_warn ("ospf_opaque_lsa_install: register_opaque_lsa() ?");
        !          1565:       goto out;
        !          1566:     }
        !          1567: 
        !          1568:   /*
        !          1569:    * Make use of a common mechanism (ospf_lsa_refresh_walker)
        !          1570:    * for periodic refresh of self-originated Opaque-LSAs.
        !          1571:    */
        !          1572:   switch (lsa->data->type)
        !          1573:     {
        !          1574:     case OSPF_OPAQUE_LINK_LSA:
        !          1575:       if ((top = oi_to_top (lsa->oi)) == NULL)
        !          1576:         {
        !          1577:           /* Above conditions must have passed. */
        !          1578:           zlog_warn ("ospf_opaque_lsa_install: Sonmething wrong?");
        !          1579:           goto out;
        !          1580:         }
        !          1581:       break;
        !          1582:     case OSPF_OPAQUE_AREA_LSA:
        !          1583:       if (lsa->area == NULL || (top = lsa->area->ospf) == NULL)
        !          1584:         {
        !          1585:           /* Above conditions must have passed. */
        !          1586:           zlog_warn ("ospf_opaque_lsa_install: Sonmething wrong?");
        !          1587:           goto out;
        !          1588:         }
        !          1589:       break;
        !          1590:     case OSPF_OPAQUE_AS_LSA:
        !          1591:       top = ospf_lookup ();
        !          1592:       if (lsa->area != NULL && (top = lsa->area->ospf) == NULL)
        !          1593:         {
        !          1594:           /* Above conditions must have passed. */
        !          1595:           zlog_warn ("ospf_opaque_lsa_install: Sonmething wrong?");
        !          1596:           goto out;
        !          1597:         }
        !          1598:       break;
        !          1599:     default:
        !          1600:       zlog_warn ("ospf_opaque_lsa_install: Unexpected LSA-type(%u)", lsa->data->type);
        !          1601:       goto out;
        !          1602:     }
        !          1603: 
        !          1604:   ospf_refresher_register_lsa (top, lsa);
        !          1605:   new = lsa;
        !          1606: 
        !          1607: out:
        !          1608:   return new;
        !          1609: }
        !          1610: 
        !          1611: struct ospf_lsa *
        !          1612: ospf_opaque_lsa_refresh (struct ospf_lsa *lsa)
        !          1613: {
        !          1614:   struct ospf *ospf;
        !          1615:   struct ospf_opaque_functab *functab;
        !          1616:   struct ospf_lsa *new = NULL;
        !          1617:   
        !          1618:   ospf = ospf_lookup ();
        !          1619: 
        !          1620:   if ((functab = ospf_opaque_functab_lookup (lsa)) == NULL
        !          1621:       || functab->lsa_refresher == NULL)
        !          1622:     {
        !          1623:       /*
        !          1624:        * Though this LSA seems to have originated on this node, the
        !          1625:        * handling module for this "lsa-type and opaque-type" was
        !          1626:        * already deleted sometime ago.
        !          1627:        * Anyway, this node still has a responsibility to flush this
        !          1628:        * LSA from the routing domain.
        !          1629:        */
        !          1630:       if (IS_DEBUG_OSPF_EVENT)
        !          1631:         zlog_debug ("LSA[Type%d:%s]: Flush stray Opaque-LSA", lsa->data->type, inet_ntoa (lsa->data->id));
        !          1632: 
        !          1633:       lsa->data->ls_age = htons (OSPF_LSA_MAXAGE);
        !          1634:       ospf_lsa_flush (ospf, lsa);
        !          1635:     }
        !          1636:   else
        !          1637:     new = (* functab->lsa_refresher)(lsa);
        !          1638: 
        !          1639:   return new;
        !          1640: }
        !          1641: 
        !          1642: /*------------------------------------------------------------------------*
        !          1643:  * Followings are re-origination/refresh/flush operations of Opaque-LSAs,
        !          1644:  * triggered by external interventions (vty session, signaling, etc).
        !          1645:  *------------------------------------------------------------------------*/
        !          1646: 
        !          1647: #define OSPF_OPAQUE_TIMER_ON(T,F,L,V) \
        !          1648:       if (!(T)) \
        !          1649:         (T) = thread_add_timer (master, (F), (L), (V))
        !          1650: 
        !          1651: static struct ospf_lsa *pseudo_lsa (struct ospf_interface *oi, struct ospf_area *area, u_char lsa_type, u_char opaque_type);
        !          1652: static int ospf_opaque_type9_lsa_reoriginate_timer (struct thread *t);
        !          1653: static int ospf_opaque_type10_lsa_reoriginate_timer (struct thread *t);
        !          1654: static int ospf_opaque_type11_lsa_reoriginate_timer (struct thread *t);
        !          1655: static int ospf_opaque_lsa_refresh_timer (struct thread *t);
        !          1656: 
        !          1657: void
        !          1658: ospf_opaque_lsa_reoriginate_schedule (void *lsa_type_dependent,
        !          1659:                                       u_char lsa_type, u_char opaque_type)
        !          1660: {
        !          1661:   struct ospf *top;
        !          1662:   struct ospf_area dummy, *area = NULL;
        !          1663:   struct ospf_interface *oi = NULL;
        !          1664: 
        !          1665:   struct ospf_lsa *lsa;
        !          1666:   struct opaque_info_per_type *oipt;
        !          1667:   int (*func) (struct thread * t) = NULL;
        !          1668:   int delay;
        !          1669: 
        !          1670:   switch (lsa_type)
        !          1671:     {
        !          1672:     case OSPF_OPAQUE_LINK_LSA:
        !          1673:       if ((oi = (struct ospf_interface *) lsa_type_dependent) == NULL)
        !          1674:         {
        !          1675:           zlog_warn ("ospf_opaque_lsa_reoriginate_schedule:"
        !          1676:                      " Type-9 Opaque-LSA: Invalid parameter?");
        !          1677:           goto out;
        !          1678:         }
        !          1679:       if ((top = oi_to_top (oi)) == NULL)
        !          1680:         {
        !          1681:           zlog_warn ("ospf_opaque_lsa_reoriginate_schedule: OI(%s) -> TOP?",
        !          1682:                      IF_NAME (oi));
        !          1683:           goto out;
        !          1684:         }
        !          1685:       if (!list_isempty (ospf_opaque_type9_funclist)
        !          1686:           && list_isempty (oi->opaque_lsa_self)
        !          1687:           && oi->t_opaque_lsa_self != NULL)
        !          1688:         {
        !          1689:           zlog_warn ("Type-9 Opaque-LSA (opaque_type=%u):"
        !          1690:                      " Common origination for OI(%s) has already started",
        !          1691:                      opaque_type, IF_NAME (oi));
        !          1692:           goto out;
        !          1693:         }
        !          1694:       func = ospf_opaque_type9_lsa_reoriginate_timer;
        !          1695:       break;
        !          1696:     case OSPF_OPAQUE_AREA_LSA:
        !          1697:       if ((area = (struct ospf_area *) lsa_type_dependent) == NULL)
        !          1698:         {
        !          1699:           zlog_warn ("ospf_opaque_lsa_reoriginate_schedule:"
        !          1700:                      " Type-10 Opaque-LSA: Invalid parameter?");
        !          1701:           goto out;
        !          1702:         }
        !          1703:       if ((top = area->ospf) == NULL)
        !          1704:         {
        !          1705:           zlog_warn ("ospf_opaque_lsa_reoriginate_schedule:"
        !          1706:                      " AREA(%s) -> TOP?", inet_ntoa (area->area_id));
        !          1707:           goto out;
        !          1708:         }
        !          1709:       if (!list_isempty (ospf_opaque_type10_funclist)
        !          1710:           && list_isempty (area->opaque_lsa_self)
        !          1711:           && area->t_opaque_lsa_self != NULL)
        !          1712:         {
        !          1713:           zlog_warn ("Type-10 Opaque-LSA (opaque_type=%u):"
        !          1714:                      " Common origination for AREA(%s) has already started",
        !          1715:                      opaque_type, inet_ntoa (area->area_id));
        !          1716:           goto out;
        !          1717:         }
        !          1718:       func = ospf_opaque_type10_lsa_reoriginate_timer;
        !          1719:       break;
        !          1720:     case OSPF_OPAQUE_AS_LSA:
        !          1721:       if ((top = (struct ospf *) lsa_type_dependent) == NULL)
        !          1722:         {
        !          1723:           zlog_warn ("ospf_opaque_lsa_reoriginate_schedule:"
        !          1724:                      " Type-11 Opaque-LSA: Invalid parameter?");
        !          1725:           goto out;
        !          1726:         }
        !          1727:       if (!list_isempty (ospf_opaque_type11_funclist)
        !          1728:           && list_isempty (top->opaque_lsa_self)
        !          1729:           && top->t_opaque_lsa_self != NULL)
        !          1730:         {
        !          1731:           zlog_warn ("Type-11 Opaque-LSA (opaque_type=%u):"
        !          1732:                      " Common origination has already started", opaque_type);
        !          1733:           goto out;
        !          1734:         }
        !          1735: 
        !          1736:       /* Fake "area" to pass "ospf" to a lookup function later. */
        !          1737:       dummy.ospf = top;
        !          1738:       area = &dummy;
        !          1739: 
        !          1740:       func = ospf_opaque_type11_lsa_reoriginate_timer;
        !          1741:       break;
        !          1742:     default:
        !          1743:       zlog_warn ("ospf_opaque_lsa_reoriginate_schedule:"
        !          1744:                  " Unexpected LSA-type(%u)",
        !          1745:                  lsa_type);
        !          1746:       goto out;
        !          1747:     }
        !          1748: 
        !          1749:   /* It may not a right time to schedule reorigination now. */
        !          1750:   if (!CHECK_FLAG (top->opaque, OPAQUE_OPERATION_READY_BIT))
        !          1751:     {
        !          1752:       if (IS_DEBUG_OSPF_EVENT)
        !          1753:         zlog_debug ("ospf_opaque_lsa_reoriginate_schedule: Not operational.");
        !          1754:       goto out;                 /* This is not an error. */
        !          1755:     }
        !          1756:   if (IS_OPAQUE_LSA_ORIGINATION_BLOCKED (top->opaque))
        !          1757:     {
        !          1758:       if (IS_DEBUG_OSPF_EVENT)
        !          1759:         zlog_debug ("ospf_opaque_lsa_reoriginate_schedule: Under blockade.");
        !          1760:       goto out;                 /* This is not an error, too. */
        !          1761:     }
        !          1762: 
        !          1763:   /* Generate a dummy lsa to be passed for a lookup function. */
        !          1764:   lsa = pseudo_lsa (oi, area, lsa_type, opaque_type);
        !          1765: 
        !          1766:   if ((oipt = lookup_opaque_info_by_type (lsa)) == NULL)
        !          1767:     {
        !          1768:       struct ospf_opaque_functab *functab;
        !          1769:       if ((functab = ospf_opaque_functab_lookup (lsa)) == NULL)
        !          1770:         {
        !          1771:           zlog_warn ("ospf_opaque_lsa_reoriginate_schedule:"
        !          1772:                      " No associated function?: lsa_type(%u),"
        !          1773:                      " opaque_type(%u)",
        !          1774:                      lsa_type, opaque_type);
        !          1775:           goto out;
        !          1776:         }
        !          1777:       if ((oipt = register_opaque_info_per_type (functab, lsa)) == NULL)
        !          1778:         {
        !          1779:           zlog_warn ("ospf_opaque_lsa_reoriginate_schedule:"
        !          1780:                      " Cannot get a control info?: lsa_type(%u),"
        !          1781:                      " opaque_type(%u)",
        !          1782:                      lsa_type, opaque_type);
        !          1783:           goto out;
        !          1784:         }
        !          1785:     }
        !          1786: 
        !          1787:   if (oipt->t_opaque_lsa_self != NULL)
        !          1788:     {
        !          1789:       if (IS_DEBUG_OSPF_EVENT)
        !          1790:         zlog_debug ("Type-%u Opaque-LSA has already scheduled to"
        !          1791:                    " RE-ORIGINATE: [opaque-type=%u]",
        !          1792:                    lsa_type, GET_OPAQUE_TYPE (ntohl (lsa->data->id.s_addr)));
        !          1793:       goto out;
        !          1794:     }
        !          1795: 
        !          1796:   /*
        !          1797:    * Different from initial origination time, in which various conditions
        !          1798:    * (opaque capability, neighbor status etc) are assured by caller of
        !          1799:    * the originating function "ospf_opaque_lsa_originate_schedule ()",
        !          1800:    * it is highly possible that these conditions might not be satisfied
        !          1801:    * at the time of re-origination function is to be called.
        !          1802:    */
        !          1803:   delay = OSPF_MIN_LS_INTERVAL; /* XXX */
        !          1804: 
        !          1805:   if (IS_DEBUG_OSPF_EVENT)
        !          1806:     zlog_debug ("Schedule Type-%u Opaque-LSA to RE-ORIGINATE in %d"
        !          1807:                " sec later: [opaque-type=%u]", 
        !          1808:                lsa_type, delay, 
        !          1809:                GET_OPAQUE_TYPE (ntohl (lsa->data->id.s_addr)));
        !          1810: 
        !          1811:   OSPF_OPAQUE_TIMER_ON (oipt->t_opaque_lsa_self, func, oipt, delay);
        !          1812: 
        !          1813: out:
        !          1814:   return;
        !          1815: }
        !          1816: 
        !          1817: static struct ospf_lsa *
        !          1818: pseudo_lsa (struct ospf_interface *oi, struct ospf_area *area,
        !          1819:             u_char lsa_type, u_char opaque_type)
        !          1820: {
        !          1821:   static struct ospf_lsa lsa = { 0 };
        !          1822:   static struct lsa_header lsah = { 0 };
        !          1823:   u_int32_t tmp;
        !          1824: 
        !          1825:   lsa.oi   = oi;
        !          1826:   lsa.area = area;
        !          1827:   lsa.data = &lsah;
        !          1828: 
        !          1829:   lsah.type = lsa_type;
        !          1830:   tmp = SET_OPAQUE_LSID (opaque_type, 0); /* Opaque-ID is unused here. */
        !          1831:   lsah.id.s_addr = htonl (tmp);
        !          1832: 
        !          1833:   return &lsa;
        !          1834: }
        !          1835: 
        !          1836: static int
        !          1837: ospf_opaque_type9_lsa_reoriginate_timer (struct thread *t)
        !          1838: {
        !          1839:   struct opaque_info_per_type *oipt;
        !          1840:   struct ospf_opaque_functab *functab;
        !          1841:   struct ospf *top;
        !          1842:   struct ospf_interface *oi;
        !          1843:   int rc = -1;
        !          1844: 
        !          1845:   oipt = THREAD_ARG (t);
        !          1846:   oipt->t_opaque_lsa_self = NULL;
        !          1847: 
        !          1848:   if ((functab = oipt->functab) == NULL
        !          1849:   ||   functab->lsa_originator == NULL)
        !          1850:     {
        !          1851:       zlog_warn ("ospf_opaque_type9_lsa_reoriginate_timer: No associated function?");
        !          1852:       goto out;
        !          1853:     }
        !          1854: 
        !          1855:   oi = (struct ospf_interface *) oipt->owner;
        !          1856:   if ((top = oi_to_top (oi)) == NULL)
        !          1857:     {
        !          1858:       zlog_warn ("ospf_opaque_type9_lsa_reoriginate_timer: Something wrong?");
        !          1859:       goto out;
        !          1860:     }
        !          1861: 
        !          1862:   if (! CHECK_FLAG (top->config, OSPF_OPAQUE_CAPABLE)
        !          1863:   ||  ! ospf_if_is_enable (oi)
        !          1864:   ||    ospf_nbr_count_opaque_capable (oi) == 0)
        !          1865:     {
        !          1866:       if (IS_DEBUG_OSPF_EVENT)
        !          1867:         zlog_debug ("Suspend re-origination of Type-9 Opaque-LSAs (opaque-type=%u) for a while...", oipt->opaque_type);
        !          1868:     
        !          1869:       oipt->status = PROC_SUSPEND;
        !          1870:       rc = 0;
        !          1871:       goto out;
        !          1872:     }
        !          1873: 
        !          1874:   if (IS_DEBUG_OSPF_EVENT)
        !          1875:     zlog_debug ("Timer[Type9-LSA]: Re-originate Opaque-LSAs (opaque-type=%u) for OI (%s)", oipt->opaque_type, IF_NAME (oi));
        !          1876: 
        !          1877:   rc = (* functab->lsa_originator)(oi);
        !          1878: out:
        !          1879:   return rc;
        !          1880: }
        !          1881: 
        !          1882: static int
        !          1883: ospf_opaque_type10_lsa_reoriginate_timer (struct thread *t)
        !          1884: {
        !          1885:   struct opaque_info_per_type *oipt;
        !          1886:   struct ospf_opaque_functab *functab;
        !          1887:   struct listnode *node, *nnode;
        !          1888:   struct ospf *top;
        !          1889:   struct ospf_area *area;
        !          1890:   struct ospf_interface *oi;
        !          1891:   int n, rc = -1;
        !          1892: 
        !          1893:   oipt = THREAD_ARG (t);
        !          1894:   oipt->t_opaque_lsa_self = NULL;
        !          1895: 
        !          1896:   if ((functab = oipt->functab) == NULL
        !          1897:   ||   functab->lsa_originator == NULL)
        !          1898:     {
        !          1899:       zlog_warn ("ospf_opaque_type10_lsa_reoriginate_timer: No associated function?");
        !          1900:       goto out;
        !          1901:     }
        !          1902: 
        !          1903:   area = (struct ospf_area *) oipt->owner;
        !          1904:   if (area == NULL || (top = area->ospf) == NULL)
        !          1905:     {
        !          1906:       zlog_warn ("ospf_opaque_type10_lsa_reoriginate_timer: Something wrong?");
        !          1907:       goto out;
        !          1908:     }
        !          1909: 
        !          1910:   /* There must be at least one "opaque-capable, full-state" neighbor. */
        !          1911:   n = 0;
        !          1912:   for (ALL_LIST_ELEMENTS (area->oiflist, node, nnode, oi))
        !          1913:     {
        !          1914:       if ((n = ospf_nbr_count_opaque_capable (oi)) > 0)
        !          1915:         break;
        !          1916:     }
        !          1917: 
        !          1918:   if (n == 0 || ! CHECK_FLAG (top->config, OSPF_OPAQUE_CAPABLE))
        !          1919:     {
        !          1920:       if (IS_DEBUG_OSPF_EVENT)
        !          1921:         zlog_debug ("Suspend re-origination of Type-10 Opaque-LSAs"
        !          1922:                    " (opaque-type=%u) for a while...", 
        !          1923:                    oipt->opaque_type);
        !          1924: 
        !          1925:       oipt->status = PROC_SUSPEND;
        !          1926:       rc = 0;
        !          1927:       goto out;
        !          1928:     }
        !          1929: 
        !          1930:   if (IS_DEBUG_OSPF_EVENT)
        !          1931:     zlog_debug ("Timer[Type10-LSA]: Re-originate Opaque-LSAs"
        !          1932:                " (opaque-type=%u) for Area %s", 
        !          1933:                oipt->opaque_type, inet_ntoa (area->area_id));
        !          1934: 
        !          1935:   rc = (* functab->lsa_originator)(area);
        !          1936: out:
        !          1937:   return rc;
        !          1938: }
        !          1939: 
        !          1940: static int
        !          1941: ospf_opaque_type11_lsa_reoriginate_timer (struct thread *t)
        !          1942: {
        !          1943:   struct opaque_info_per_type *oipt;
        !          1944:   struct ospf_opaque_functab *functab;
        !          1945:   struct ospf *top;
        !          1946:   int rc = -1;
        !          1947: 
        !          1948:   oipt = THREAD_ARG (t);
        !          1949:   oipt->t_opaque_lsa_self = NULL;
        !          1950: 
        !          1951:   if ((functab = oipt->functab) == NULL
        !          1952:       || functab->lsa_originator == NULL)
        !          1953:     {
        !          1954:       zlog_warn ("ospf_opaque_type11_lsa_reoriginate_timer:"
        !          1955:                  " No associated function?");
        !          1956:       goto out;
        !          1957:     }
        !          1958: 
        !          1959:   if ((top = (struct ospf *) oipt->owner) == NULL)
        !          1960:     {
        !          1961:       zlog_warn ("ospf_opaque_type11_lsa_reoriginate_timer: Something wrong?");
        !          1962:       goto out;
        !          1963:     }
        !          1964: 
        !          1965:   if (! CHECK_FLAG (top->config, OSPF_OPAQUE_CAPABLE))
        !          1966:     {
        !          1967:       if (IS_DEBUG_OSPF_EVENT)
        !          1968:         zlog_debug ("Suspend re-origination of Type-11 Opaque-LSAs (opaque-type=%u) for a while...", oipt->opaque_type);
        !          1969:     
        !          1970:       oipt->status = PROC_SUSPEND;
        !          1971:       rc = 0;
        !          1972:       goto out;
        !          1973:     }
        !          1974: 
        !          1975:   if (IS_DEBUG_OSPF_EVENT)
        !          1976:     zlog_debug ("Timer[Type11-LSA]: Re-originate Opaque-LSAs (opaque-type=%u).", oipt->opaque_type);
        !          1977: 
        !          1978:   rc = (* functab->lsa_originator)(top);
        !          1979: out:
        !          1980:   return rc;
        !          1981: }
        !          1982: 
        !          1983: void
        !          1984: ospf_opaque_lsa_refresh_schedule (struct ospf_lsa *lsa0)
        !          1985: {
        !          1986:   struct opaque_info_per_type *oipt;
        !          1987:   struct opaque_info_per_id *oipi;
        !          1988:   struct ospf_lsa *lsa;
        !          1989:   int delay;
        !          1990: 
        !          1991:   if ((oipt = lookup_opaque_info_by_type (lsa0)) == NULL
        !          1992:   ||  (oipi = lookup_opaque_info_by_id (oipt, lsa0)) == NULL)
        !          1993:     {
        !          1994:       zlog_warn ("ospf_opaque_lsa_refresh_schedule: Invalid parameter?");
        !          1995:       goto out;
        !          1996:     }
        !          1997: 
        !          1998:   /* Given "lsa0" and current "oipi->lsa" may different, but harmless. */
        !          1999:   if ((lsa = oipi->lsa) == NULL)
        !          2000:     {
        !          2001:       zlog_warn ("ospf_opaque_lsa_refresh_schedule: Something wrong?");
        !          2002:       goto out;
        !          2003:     }
        !          2004: 
        !          2005:   if (oipi->t_opaque_lsa_self != NULL)
        !          2006:     {
        !          2007:       if (IS_DEBUG_OSPF_EVENT)
        !          2008:         zlog_debug ("Type-%u Opaque-LSA has already scheduled to REFRESH: [opaque-type=%u, opaque-id=%x]", lsa->data->type, GET_OPAQUE_TYPE (ntohl (lsa->data->id.s_addr)), GET_OPAQUE_ID (ntohl (lsa->data->id.s_addr)));
        !          2009:       goto out;
        !          2010:     }
        !          2011: 
        !          2012:   /* Delete this lsa from neighbor retransmit-list. */
        !          2013:   switch (lsa->data->type)
        !          2014:     {
        !          2015:     case OSPF_OPAQUE_LINK_LSA:
        !          2016:     case OSPF_OPAQUE_AREA_LSA:
        !          2017:       ospf_ls_retransmit_delete_nbr_area (lsa->area, lsa);
        !          2018:       break;
        !          2019:     case OSPF_OPAQUE_AS_LSA:
        !          2020:       ospf_ls_retransmit_delete_nbr_as (lsa0->area->ospf, lsa);
        !          2021:       break;
        !          2022:     default:
        !          2023:       zlog_warn ("ospf_opaque_lsa_refresh_schedule: Unexpected LSA-type(%u)", lsa->data->type);
        !          2024:       goto out;
        !          2025:     }
        !          2026: 
        !          2027:   delay = ospf_lsa_refresh_delay (lsa);
        !          2028: 
        !          2029:   if (IS_DEBUG_OSPF_EVENT)
        !          2030:     zlog_debug ("Schedule Type-%u Opaque-LSA to REFRESH in %d sec later: [opaque-type=%u, opaque-id=%x]", lsa->data->type, delay, GET_OPAQUE_TYPE (ntohl (lsa->data->id.s_addr)), GET_OPAQUE_ID (ntohl (lsa->data->id.s_addr)));
        !          2031: 
        !          2032:   OSPF_OPAQUE_TIMER_ON (oipi->t_opaque_lsa_self,
        !          2033:                         ospf_opaque_lsa_refresh_timer, oipi, delay);
        !          2034: out:
        !          2035:   return;
        !          2036: }
        !          2037: 
        !          2038: static int
        !          2039: ospf_opaque_lsa_refresh_timer (struct thread *t)
        !          2040: {
        !          2041:   struct opaque_info_per_id *oipi;
        !          2042:   struct ospf_opaque_functab *functab;
        !          2043:   struct ospf_lsa *lsa;
        !          2044: 
        !          2045:   if (IS_DEBUG_OSPF_EVENT)
        !          2046:     zlog_debug ("Timer[Opaque-LSA]: (Opaque-LSA Refresh expire)");
        !          2047: 
        !          2048:   oipi = THREAD_ARG (t);
        !          2049:   oipi->t_opaque_lsa_self = NULL;
        !          2050: 
        !          2051:   if ((lsa = oipi->lsa) != NULL)
        !          2052:     if ((functab = oipi->opqctl_type->functab) != NULL)
        !          2053:       if (functab->lsa_refresher != NULL)
        !          2054:         (* functab->lsa_refresher)(lsa);
        !          2055: 
        !          2056:   return 0;
        !          2057: }
        !          2058: 
        !          2059: void
        !          2060: ospf_opaque_lsa_flush_schedule (struct ospf_lsa *lsa0)
        !          2061: {
        !          2062:   struct opaque_info_per_type *oipt;
        !          2063:   struct opaque_info_per_id *oipi;
        !          2064:   struct ospf_lsa *lsa;
        !          2065: 
        !          2066:   if ((oipt = lookup_opaque_info_by_type (lsa0)) == NULL
        !          2067:   ||  (oipi = lookup_opaque_info_by_id (oipt, lsa0)) == NULL)
        !          2068:     {
        !          2069:       zlog_warn ("ospf_opaque_lsa_flush_schedule: Invalid parameter?");
        !          2070:       goto out;
        !          2071:     }
        !          2072: 
        !          2073:   /* Given "lsa0" and current "oipi->lsa" may different, but harmless. */
        !          2074:   if ((lsa = oipi->lsa) == NULL)
        !          2075:     {
        !          2076:       zlog_warn ("ospf_opaque_lsa_flush_schedule: Something wrong?");
        !          2077:       goto out;
        !          2078:     }
        !          2079: 
        !          2080:   /* Delete this lsa from neighbor retransmit-list. */
        !          2081:   switch (lsa->data->type)
        !          2082:     {
        !          2083:     case OSPF_OPAQUE_LINK_LSA:
        !          2084:     case OSPF_OPAQUE_AREA_LSA:
        !          2085:       ospf_ls_retransmit_delete_nbr_area (lsa->area, lsa);
        !          2086:       break;
        !          2087:     case OSPF_OPAQUE_AS_LSA:
        !          2088:       ospf_ls_retransmit_delete_nbr_as (lsa0->area->ospf, lsa);
        !          2089:       break;
        !          2090:     default:
        !          2091:       zlog_warn ("ospf_opaque_lsa_flush_schedule: Unexpected LSA-type(%u)", lsa->data->type);
        !          2092:       goto out;
        !          2093:     }
        !          2094: 
        !          2095:   /* Dequeue listnode entry from the list. */
        !          2096:   listnode_delete (oipt->id_list, oipi);
        !          2097: 
        !          2098:   /* Avoid misjudgement in the next lookup. */
        !          2099:   if (listcount (oipt->id_list) == 0)
        !          2100:     oipt->id_list->head = oipt->id_list->tail = NULL;
        !          2101: 
        !          2102:   /* Disassociate internal control information with the given lsa. */
        !          2103:   free_opaque_info_per_id ((void *) oipi);
        !          2104: 
        !          2105:   /* Force given lsa's age to MaxAge. */
        !          2106:   lsa->data->ls_age = htons (OSPF_LSA_MAXAGE);
        !          2107: 
        !          2108:   if (IS_DEBUG_OSPF_EVENT)
        !          2109:     zlog_debug ("Schedule Type-%u Opaque-LSA to FLUSH: [opaque-type=%u, opaque-id=%x]", lsa->data->type, GET_OPAQUE_TYPE (ntohl (lsa->data->id.s_addr)), GET_OPAQUE_ID (ntohl (lsa->data->id.s_addr)));
        !          2110: 
        !          2111:   /* This lsa will be flushed and removed eventually. */
        !          2112:   ospf_lsa_flush (lsa0->area->ospf, lsa);
        !          2113: 
        !          2114: out:
        !          2115:   return;
        !          2116: }
        !          2117: 
        !          2118: /*------------------------------------------------------------------------*
        !          2119:  * Followings are control functions to block origination after restart.
        !          2120:  *------------------------------------------------------------------------*/
        !          2121: 
        !          2122: static void ospf_opaque_exclude_lsa_from_lsreq (struct route_table *nbrs, struct ospf_neighbor *inbr, struct ospf_lsa *lsa);
        !          2123: static void ospf_opaque_type9_lsa_rxmt_nbr_check (struct ospf_interface *oi);
        !          2124: static void ospf_opaque_type10_lsa_rxmt_nbr_check (struct ospf_area *area);
        !          2125: static void ospf_opaque_type11_lsa_rxmt_nbr_check (struct ospf *top);
        !          2126: static unsigned long ospf_opaque_nrxmt_self (struct route_table *nbrs, int lsa_type);
        !          2127: 
        !          2128: void
        !          2129: ospf_opaque_adjust_lsreq (struct ospf_neighbor *nbr, struct list *lsas)
        !          2130: {
        !          2131:   struct ospf *top;
        !          2132:   struct ospf_area *area;
        !          2133:   struct ospf_interface *oi;
        !          2134:   struct listnode *node1, *nnode1;
        !          2135:   struct listnode *node2, *nnode2;
        !          2136:   struct ospf_lsa *lsa;
        !          2137: 
        !          2138:   if ((top = oi_to_top (nbr->oi)) == NULL)
        !          2139:     goto out;
        !          2140: 
        !          2141:   /*
        !          2142:    * If an instance of self-originated Opaque-LSA is found in the given
        !          2143:    * LSA list, and it is not installed to LSDB yet, exclude it from the
        !          2144:    * list "nbr->ls_req". In this way, it is assured that an LSReq message,
        !          2145:    * which might be sent in the process of flooding, will not request for
        !          2146:    * the LSA to be flushed immediately; otherwise, depending on timing,
        !          2147:    * an LSUpd message will carry instances of target LSAs with MaxAge,
        !          2148:    * while other LSUpd message might carry old LSA instances (non-MaxAge).
        !          2149:    * Obviously, the latter would trigger miserable situations that repeat
        !          2150:    * installation and removal of unwanted LSAs indefinitely.
        !          2151:    */
        !          2152:   for (ALL_LIST_ELEMENTS (lsas, node1, nnode1, lsa))
        !          2153:     {
        !          2154:       /* Filter out unwanted LSAs. */
        !          2155:       if (! IS_OPAQUE_LSA (lsa->data->type))
        !          2156:         continue;
        !          2157:       if (! IPV4_ADDR_SAME (&lsa->data->adv_router, &top->router_id))
        !          2158:         continue;
        !          2159: 
        !          2160:       /*
        !          2161:        * Don't touch an LSA which has MaxAge; two possible cases.
        !          2162:        *
        !          2163:        *   1) This LSA has originally flushed by myself (received LSUpd
        !          2164:        *      message's router-id is equal to my router-id), and flooded
        !          2165:        *      back by an opaque-capable router.
        !          2166:        *
        !          2167:        *   2) This LSA has expired in an opaque-capable router and thus
        !          2168:        *      flushed by the router.
        !          2169:        */
        !          2170:       if (IS_LSA_MAXAGE (lsa))
        !          2171:         continue;
        !          2172: 
        !          2173:       /* If the LSA has installed in the LSDB, nothing to do here. */
        !          2174:       if (ospf_lsa_lookup_by_header (nbr->oi->area, lsa->data) != NULL)
        !          2175:         continue;
        !          2176: 
        !          2177:       /* Ok, here we go. */
        !          2178:       switch (lsa->data->type)
        !          2179:         {
        !          2180:         case OSPF_OPAQUE_LINK_LSA:
        !          2181:           oi = nbr->oi;
        !          2182:           ospf_opaque_exclude_lsa_from_lsreq (oi->nbrs, nbr, lsa);
        !          2183:           break;
        !          2184:         case OSPF_OPAQUE_AREA_LSA:
        !          2185:           area = nbr->oi->area;
        !          2186:           for (ALL_LIST_ELEMENTS (area->oiflist, node2, nnode2, oi))
        !          2187:             ospf_opaque_exclude_lsa_from_lsreq (oi->nbrs, nbr, lsa);
        !          2188:           break;
        !          2189:         case OSPF_OPAQUE_AS_LSA:
        !          2190:           for (ALL_LIST_ELEMENTS (top->oiflist, node2, nnode2, oi))
        !          2191:             ospf_opaque_exclude_lsa_from_lsreq (oi->nbrs, nbr, lsa);
        !          2192:           break;
        !          2193:         default:
        !          2194:           break;
        !          2195:         }
        !          2196:     }
        !          2197: 
        !          2198: out:
        !          2199:   return;
        !          2200: }
        !          2201: 
        !          2202: static void
        !          2203: ospf_opaque_exclude_lsa_from_lsreq (struct route_table *nbrs,
        !          2204:                                     struct ospf_neighbor *inbr,
        !          2205:                                     struct ospf_lsa *lsa)
        !          2206: {
        !          2207:   struct route_node *rn;
        !          2208:   struct ospf_neighbor *onbr;
        !          2209:   struct ospf_lsa *ls_req;
        !          2210: 
        !          2211:   for (rn = route_top (nbrs); rn; rn = route_next (rn))
        !          2212:     {
        !          2213:       if ((onbr = rn->info) == NULL)
        !          2214:         continue;
        !          2215:       if (onbr == inbr)
        !          2216:         continue;
        !          2217:       if ((ls_req = ospf_ls_request_lookup (onbr, lsa)) == NULL)
        !          2218:         continue;
        !          2219: 
        !          2220:       if (IS_DEBUG_OSPF_EVENT)
        !          2221:         zlog_debug ("LSA[%s]: Exclude this entry from LSReq to send.", dump_lsa_key (lsa));
        !          2222: 
        !          2223:       ospf_ls_request_delete (onbr, ls_req);
        !          2224: /*    ospf_check_nbr_loading (onbr);*//* XXX */
        !          2225:     }
        !          2226: 
        !          2227:   return;
        !          2228: }
        !          2229: 
        !          2230: void
        !          2231: ospf_opaque_self_originated_lsa_received (struct ospf_neighbor *nbr, 
        !          2232:                                           struct ospf_lsa *lsa)
        !          2233: {
        !          2234:   struct ospf *top;
        !          2235:   u_char before;
        !          2236: 
        !          2237:   if ((top = oi_to_top (nbr->oi)) == NULL)
        !          2238:     return;
        !          2239: 
        !          2240:   before = IS_OPAQUE_LSA_ORIGINATION_BLOCKED (top->opaque);
        !          2241: 
        !          2242:   /*
        !          2243:    * Since these LSA entries are not yet installed into corresponding
        !          2244:    * LSDB, just flush them without calling ospf_ls_maxage() afterward.
        !          2245:    */
        !          2246:   lsa->data->ls_age = htons (OSPF_LSA_MAXAGE);
        !          2247:   switch (lsa->data->type)
        !          2248:     {
        !          2249:     case OSPF_OPAQUE_LINK_LSA:
        !          2250:       SET_FLAG (top->opaque, OPAQUE_BLOCK_TYPE_09_LSA_BIT);
        !          2251:       ospf_flood_through_area (nbr->oi->area, NULL/*inbr*/, lsa);
        !          2252:       break;
        !          2253:     case OSPF_OPAQUE_AREA_LSA:
        !          2254:       SET_FLAG (top->opaque, OPAQUE_BLOCK_TYPE_10_LSA_BIT);
        !          2255:       ospf_flood_through_area (nbr->oi->area, NULL/*inbr*/, lsa);
        !          2256:       break;
        !          2257:     case OSPF_OPAQUE_AS_LSA:
        !          2258:       SET_FLAG (top->opaque, OPAQUE_BLOCK_TYPE_11_LSA_BIT);
        !          2259:       ospf_flood_through_as (top, NULL/*inbr*/, lsa);
        !          2260:       break;
        !          2261:     default:
        !          2262:       zlog_warn ("ospf_opaque_self_originated_lsa_received: Unexpected LSA-type(%u)", lsa->data->type);
        !          2263:       return;
        !          2264:     }
        !          2265: 
        !          2266:   ospf_lsa_discard (lsa); /* List "lsas" will be deleted by caller. */
        !          2267: 
        !          2268:   if (before == 0 && IS_OPAQUE_LSA_ORIGINATION_BLOCKED (top->opaque))
        !          2269:     {
        !          2270:       if (IS_DEBUG_OSPF_EVENT)
        !          2271:         zlog_debug ("Block Opaque-LSA origination: OFF -> ON");
        !          2272:     }
        !          2273: }
        !          2274: 
        !          2275: void
        !          2276: ospf_opaque_ls_ack_received (struct ospf_neighbor *nbr, struct ospf_lsa *lsa)
        !          2277: {
        !          2278:   struct ospf *top;
        !          2279:   int delay;
        !          2280:   struct ospf_interface *oi;
        !          2281:   struct listnode *node, *nnode;
        !          2282: 
        !          2283:   if ((top = oi_to_top (nbr->oi)) == NULL)
        !          2284:     return;
        !          2285:   
        !          2286:   if (!IS_OPAQUE_LSA_ORIGINATION_BLOCKED (top->opaque))
        !          2287:     return;
        !          2288:   
        !          2289:   switch (lsa->data->type)
        !          2290:     {
        !          2291:     case OSPF_OPAQUE_LINK_LSA:
        !          2292:       if (CHECK_FLAG (top->opaque, OPAQUE_BLOCK_TYPE_09_LSA_BIT))
        !          2293:         ospf_opaque_type9_lsa_rxmt_nbr_check (nbr->oi);
        !          2294:       /* Callback function... */
        !          2295:       break;
        !          2296:     case OSPF_OPAQUE_AREA_LSA:
        !          2297:       if (CHECK_FLAG (top->opaque, OPAQUE_BLOCK_TYPE_10_LSA_BIT))
        !          2298:         ospf_opaque_type10_lsa_rxmt_nbr_check (nbr->oi->area);
        !          2299:       /* Callback function... */
        !          2300:       break;
        !          2301:     case OSPF_OPAQUE_AS_LSA:
        !          2302:       if (CHECK_FLAG (top->opaque, OPAQUE_BLOCK_TYPE_11_LSA_BIT))
        !          2303:         ospf_opaque_type11_lsa_rxmt_nbr_check (top);
        !          2304:       /* Callback function... */
        !          2305:       break;
        !          2306:     default:
        !          2307:       zlog_warn ("ospf_opaque_ls_ack_received: Unexpected LSA-type(%u)", lsa->data->type);
        !          2308:       return;
        !          2309:     }
        !          2310:   
        !          2311:   if (IS_OPAQUE_LSA_ORIGINATION_BLOCKED (top->opaque))
        !          2312:     {
        !          2313:       if (IS_DEBUG_OSPF_EVENT)
        !          2314:         zlog_debug ("Block Opaque-LSA origination: ON -> OFF");
        !          2315:       return; /* Blocking still in progress. */
        !          2316:     }
        !          2317:   
        !          2318:   if (! CHECK_FLAG (top->config, OSPF_OPAQUE_CAPABLE))
        !          2319:     return; /* Opaque capability condition must have changed. */
        !          2320: 
        !          2321:   /* Ok, let's start origination of Opaque-LSAs. */
        !          2322:   delay = OSPF_MIN_LS_INTERVAL;
        !          2323: 
        !          2324:   for (ALL_LIST_ELEMENTS (top->oiflist, node, nnode, oi))
        !          2325:     {
        !          2326:       if (! ospf_if_is_enable (oi)
        !          2327:           || ospf_nbr_count_opaque_capable (oi) == 0)
        !          2328:         continue;
        !          2329: 
        !          2330:       ospf_opaque_lsa_originate_schedule (oi, &delay);
        !          2331:     }
        !          2332:     
        !          2333:   return;
        !          2334: }
        !          2335: 
        !          2336: static void
        !          2337: ospf_opaque_type9_lsa_rxmt_nbr_check (struct ospf_interface *oi)
        !          2338: {
        !          2339:   unsigned long n;
        !          2340: 
        !          2341:   n = ospf_opaque_nrxmt_self (oi->nbrs, OSPF_OPAQUE_LINK_LSA);
        !          2342:   if (n == 0)
        !          2343:     {
        !          2344:       if (IS_DEBUG_OSPF_EVENT)
        !          2345:         zlog_debug ("Self-originated type-9 Opaque-LSAs: OI(%s): Flush completed", IF_NAME (oi));
        !          2346: 
        !          2347:       UNSET_FLAG (oi->area->ospf->opaque, OPAQUE_BLOCK_TYPE_09_LSA_BIT);
        !          2348:     }
        !          2349:   return;
        !          2350: }
        !          2351: 
        !          2352: static void
        !          2353: ospf_opaque_type10_lsa_rxmt_nbr_check (struct ospf_area *area)
        !          2354: {
        !          2355:   struct listnode *node;
        !          2356:   struct ospf_interface *oi;
        !          2357:   unsigned long n = 0;
        !          2358: 
        !          2359:   for (ALL_LIST_ELEMENTS_RO (area->oiflist, node, oi))
        !          2360:     {
        !          2361:       if (area->area_id.s_addr != OSPF_AREA_BACKBONE
        !          2362:           && oi->type == OSPF_IFTYPE_VIRTUALLINK) 
        !          2363:         continue;
        !          2364: 
        !          2365:       n = ospf_opaque_nrxmt_self (oi->nbrs, OSPF_OPAQUE_AREA_LSA);
        !          2366:       if (n > 0)
        !          2367:         break;
        !          2368:     }
        !          2369: 
        !          2370:   if (n == 0)
        !          2371:     {
        !          2372:       if (IS_DEBUG_OSPF_EVENT)
        !          2373:         zlog_debug ("Self-originated type-10 Opaque-LSAs: AREA(%s): Flush completed", inet_ntoa (area->area_id));
        !          2374: 
        !          2375:       UNSET_FLAG (area->ospf->opaque, OPAQUE_BLOCK_TYPE_10_LSA_BIT);
        !          2376:     }
        !          2377: 
        !          2378:   return;
        !          2379: }
        !          2380: 
        !          2381: static void
        !          2382: ospf_opaque_type11_lsa_rxmt_nbr_check (struct ospf *top)
        !          2383: {
        !          2384:   struct listnode *node;
        !          2385:   struct ospf_interface *oi;
        !          2386:   unsigned long n = 0;
        !          2387: 
        !          2388:   for (ALL_LIST_ELEMENTS_RO (top->oiflist, node, oi))
        !          2389:     {
        !          2390:       switch (oi->type)
        !          2391:         {
        !          2392:         case OSPF_IFTYPE_VIRTUALLINK:
        !          2393:           continue;
        !          2394:         default:
        !          2395:           break;
        !          2396:         }
        !          2397: 
        !          2398:       n = ospf_opaque_nrxmt_self (oi->nbrs, OSPF_OPAQUE_AS_LSA);
        !          2399:       if (n > 0)
        !          2400:         goto out;
        !          2401:     }
        !          2402: 
        !          2403:   if (n == 0)
        !          2404:     {
        !          2405:       if (IS_DEBUG_OSPF_EVENT)
        !          2406:         zlog_debug ("Self-originated type-11 Opaque-LSAs: Flush completed");
        !          2407: 
        !          2408:       UNSET_FLAG (top->opaque, OPAQUE_BLOCK_TYPE_11_LSA_BIT);
        !          2409:     }
        !          2410: 
        !          2411: out:
        !          2412:   return;
        !          2413: }
        !          2414: 
        !          2415: static unsigned long
        !          2416: ospf_opaque_nrxmt_self (struct route_table *nbrs, int lsa_type)
        !          2417: {
        !          2418:   struct route_node *rn;
        !          2419:   struct ospf_neighbor *nbr;
        !          2420:   struct ospf *top;
        !          2421:   unsigned long n = 0;
        !          2422: 
        !          2423:   for (rn = route_top (nbrs); rn; rn = route_next (rn))
        !          2424:     {
        !          2425:       if ((nbr = rn->info) == NULL)
        !          2426:         continue;
        !          2427:       if ((top = oi_to_top (nbr->oi)) == NULL)
        !          2428:         continue;
        !          2429:       if (IPV4_ADDR_SAME (&nbr->router_id, &top->router_id))
        !          2430:         continue;
        !          2431:       n += ospf_ls_retransmit_count_self (nbr, lsa_type);
        !          2432:     }
        !          2433: 
        !          2434:   return n;
        !          2435: }
        !          2436: 
        !          2437: /*------------------------------------------------------------------------*
        !          2438:  * Followings are util functions; probably be used by Opaque-LSAs only...
        !          2439:  *------------------------------------------------------------------------*/
        !          2440: 
        !          2441: void
        !          2442: htonf (float *src, float *dst)
        !          2443: {
        !          2444:   u_int32_t lu1, lu2;
        !          2445: 
        !          2446:   memcpy (&lu1, src, sizeof (u_int32_t));
        !          2447:   lu2 = htonl (lu1);
        !          2448:   memcpy (dst, &lu2, sizeof (u_int32_t));
        !          2449:   return;
        !          2450: }
        !          2451: 
        !          2452: void
        !          2453: ntohf (float *src, float *dst)
        !          2454: {
        !          2455:   u_int32_t lu1, lu2;
        !          2456: 
        !          2457:   memcpy (&lu1, src, sizeof (u_int32_t));
        !          2458:   lu2 = ntohl (lu1);
        !          2459:   memcpy (dst, &lu2, sizeof (u_int32_t));
        !          2460:   return;
        !          2461: }
        !          2462: 
        !          2463: struct ospf *
        !          2464: oi_to_top (struct ospf_interface *oi)
        !          2465: {
        !          2466:   struct ospf *top = NULL;
        !          2467:   struct ospf_area *area;
        !          2468: 
        !          2469:   if (oi == NULL || (area = oi->area) == NULL || (top = area->ospf) == NULL)
        !          2470:     zlog_warn ("Broken relationship for \"OI -> AREA -> OSPF\"?");
        !          2471: 
        !          2472:   return top;
        !          2473: }
        !          2474: 
        !          2475: #endif /* HAVE_OPAQUE_LSA */

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