diff options
Diffstat (limited to 'ospf6d/ospf6_lsa.c')
-rw-r--r-- | ospf6d/ospf6_lsa.c | 1867 |
1 files changed, 446 insertions, 1421 deletions
diff --git a/ospf6d/ospf6_lsa.c b/ospf6d/ospf6_lsa.c index 712aa6866..13e9a0736 100644 --- a/ospf6d/ospf6_lsa.c +++ b/ospf6d/ospf6_lsa.c @@ -1,6 +1,5 @@ /* - * LSA function - * Copyright (C) 1999 Yasuhiro Ohara + * Copyright (C) 2003 Yasuhiro Ohara * * This file is part of GNU Zebra. * @@ -23,73 +22,56 @@ #include <zebra.h> /* Include other stuffs */ -#include <lib/version.h> #include "log.h" -#include "getopt.h" #include "linklist.h" -#include "thread.h" #include "command.h" #include "memory.h" -#include "sockunion.h" -#include "if.h" -#include "prefix.h" -#include "stream.h" #include "thread.h" -#include "filter.h" -#include "zclient.h" -#include "table.h" -#include "plist.h" +#include "ospf6d.h" #include "ospf6_proto.h" -#include "ospf6_prefix.h" #include "ospf6_lsa.h" #include "ospf6_lsdb.h" #include "ospf6_message.h" -#include "ospf6_dump.h" #include "ospf6_top.h" #include "ospf6_area.h" #include "ospf6_interface.h" #include "ospf6_neighbor.h" -#include "ospf6_ism.h" -#include "ospf6_nsm.h" -#include "ospf6_dbex.h" -#define HEADER_DEPENDENCY -#include "ospf6d.h" -#undef HEADER_DEPENDENCY +#include "ospf6_flood.h" -/* test LSAs identity */ -static int -ospf6_lsa_issame (struct ospf6_lsa_header__ *lsh1, - struct ospf6_lsa_header__ *lsh2) -{ - assert (lsh1 && lsh2); +unsigned char conf_debug_ospf6_lsa = 0; - if (lsh1->adv_router != lsh2->adv_router) - return 0; +struct ospf6_lstype ospf6_lstype[OSPF6_LSTYPE_SIZE]; - if (lsh1->id != lsh2->id) - return 0; +char *ospf6_lstype_str[OSPF6_LSTYPE_SIZE] = + {"Unknown", "Router", "Network", "Inter-Prefix", "Inter-Router", + "AS-External", "Group-Membership", "Type-7", "Link", "Intra-Prefix"}; - if (lsh1->type != lsh2->type) - return 0; +char * +ospf6_lstype_name (u_int16_t type) +{ + static char buf[8]; + int index = ntohs (type) & OSPF6_LSTYPE_FCODE_MASK; + + if (index < OSPF6_LSTYPE_SIZE && ospf6_lstype_str[index]) + return ospf6_lstype_str[index]; - return 1; + snprintf (buf, sizeof (buf), "0x%04hx", ntohs (type)); + return buf; } /* RFC2328: Section 13.2 */ int -ospf6_lsa_differ (struct ospf6_lsa *lsa1, - struct ospf6_lsa *lsa2) +ospf6_lsa_is_differ (struct ospf6_lsa *lsa1, + struct ospf6_lsa *lsa2) { - int diff, cmplen; + int len; - if (! ospf6_lsa_issame (lsa1->header, lsa2->header)) - return 1; + assert (OSPF6_LSA_IS_SAME (lsa1, lsa2)); - /* check Options field */ - /* xxx */ + /* XXX, Options ??? */ ospf6_lsa_age_current (lsa1); ospf6_lsa_age_current (lsa2); @@ -104,26 +86,26 @@ ospf6_lsa_differ (struct ospf6_lsa *lsa1, if (ntohs (lsa1->header->length) != ntohs (lsa2->header->length)) return 1; - cmplen = ntohs (lsa1->header->length) - sizeof (struct ospf6_lsa_header); - diff = memcmp (lsa1->header + 1, lsa2->header + 1, cmplen); - - return diff; + len = ntohs (lsa1->header->length) - sizeof (struct ospf6_lsa_header); + return memcmp (lsa1->header + 1, lsa2->header + 1, len); } int -ospf6_lsa_match (u_int16_t type, u_int32_t id, u_int32_t adv_router, - struct ospf6_lsa_header *lsh) +ospf6_lsa_is_changed (struct ospf6_lsa *lsa1, + struct ospf6_lsa *lsa2) { - if (lsh->advrtr != adv_router) - return 0; + int length; - if (lsh->ls_id != id) - return 0; + if (OSPF6_LSA_IS_MAXAGE (lsa1) ^ OSPF6_LSA_IS_MAXAGE (lsa2)) + return 1; + if (ntohs (lsa1->header->length) != ntohs (lsa2->header->length)) + return 1; - if (lsh->type != type) - return 0; + length = OSPF6_LSA_SIZE (lsa1->header) - sizeof (struct ospf6_lsa_header); + assert (length > 0); - return 1; + return memcmp (OSPF6_LSA_HEADER_END (lsa1->header), + OSPF6_LSA_HEADER_END (lsa2->header), length); } /* ospf6 age functions */ @@ -143,7 +125,8 @@ ospf6_lsa_age_set (struct ospf6_lsa *lsa) lsa->birth.tv_usec = now.tv_usec; if (ntohs (lsa->header->age) != MAXAGE) lsa->expire = thread_add_timer (master, ospf6_lsa_expire, lsa, - lsa->birth.tv_sec + MAXAGE - now.tv_sec); + MAXAGE + lsa->birth.tv_sec + - now.tv_sec); else lsa->expire = NULL; return; @@ -163,17 +146,14 @@ ospf6_lsa_age_current (struct ospf6_lsa *lsa) /* current time */ if (gettimeofday (&now, (struct timezone *)NULL) < 0) - zlog_warn ("LSA: gettimeofday failed, may fail ages: %s", + zlog_warn ("LSA: gettimeofday failed, may fail LSA AGEs: %s", strerror (errno)); /* calculate age */ ulage = now.tv_sec - lsa->birth.tv_sec; /* if over MAXAGE, set to it */ - if (ulage > MAXAGE) - age = MAXAGE; - else - age = ulage; + age = (ulage > MAXAGE ? MAXAGE : ulage); lsa->header->age = htons (age); return age; @@ -189,22 +169,17 @@ ospf6_lsa_age_update_to_send (struct ospf6_lsa *lsa, u_int32_t transdelay) if (age > MAXAGE) age = MAXAGE; lsa->header->age = htons (age); - return; } void ospf6_lsa_premature_aging (struct ospf6_lsa *lsa) { /* log */ - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: Premature aging: %s", lsa->str); + if (IS_OSPF6_DEBUG_LSA (ORIGINATE)) + zlog_info ("LSA: Premature aging: %s", lsa->name); - if (lsa->expire) - thread_cancel (lsa->expire); - lsa->expire = (struct thread *) NULL; - if (lsa->refresh) - thread_cancel (lsa->refresh); - lsa->refresh = (struct thread *) NULL; + THREAD_OFF (lsa->expire); + THREAD_OFF (lsa->refresh); memset (&lsa->birth, 0, sizeof (struct timeval)); thread_execute (master, ospf6_lsa_expire, lsa, 0); @@ -213,7 +188,7 @@ ospf6_lsa_premature_aging (struct ospf6_lsa *lsa) /* check which is more recent. if a is more recent, return -1; if the same, return 0; otherwise(b is more recent), return 1 */ int -ospf6_lsa_check_recent (struct ospf6_lsa *a, struct ospf6_lsa *b) +ospf6_lsa_compare (struct ospf6_lsa *a, struct ospf6_lsa *b) { signed long seqnuma, seqnumb; u_int16_t cksuma, cksumb; @@ -221,7 +196,7 @@ ospf6_lsa_check_recent (struct ospf6_lsa *a, struct ospf6_lsa *b) assert (a && a->header); assert (b && b->header); - assert (ospf6_lsa_issame (a->header, b->header)); + assert (OSPF6_LSA_IS_SAME (a, b)); seqnuma = ((signed long) ntohl (a->header->seqnum)) - (signed long) INITIAL_SEQUENCE_NUMBER; @@ -229,8 +204,7 @@ ospf6_lsa_check_recent (struct ospf6_lsa *a, struct ospf6_lsa *b) - (signed long) INITIAL_SEQUENCE_NUMBER; /* compare by sequence number */ - /* xxx, care about LS sequence number wrapping */ - recent_reason = "seqnum"; + /* XXX, LS sequence number wrapping */ if (seqnuma > seqnumb) return -1; else if (seqnuma < seqnumb) @@ -244,188 +218,89 @@ ospf6_lsa_check_recent (struct ospf6_lsa *a, struct ospf6_lsa *b) if (cksuma < cksumb) return 0; - /* Age check */ + /* Update Age */ agea = ospf6_lsa_age_current (a); ageb = ospf6_lsa_age_current (b); - /* MaxAge check */ - recent_reason = "max age"; - if (agea == OSPF6_LSA_MAXAGE && ageb != OSPF6_LSA_MAXAGE) + /* MaxAge check */ + if (agea == MAXAGE && ageb != MAXAGE) return -1; - else if (agea != OSPF6_LSA_MAXAGE && ageb == OSPF6_LSA_MAXAGE) + else if (agea != MAXAGE && ageb == MAXAGE) return 1; - recent_reason = "age differ"; - if (agea > ageb && agea - ageb >= OSPF6_LSA_MAXAGEDIFF) + /* Age check */ + if (agea > ageb && agea - ageb >= MAX_AGE_DIFF) return 1; - else if (agea < ageb && ageb - agea >= OSPF6_LSA_MAXAGEDIFF) + else if (agea < ageb && ageb - agea >= MAX_AGE_DIFF) return -1; /* neither recent */ - recent_reason = "the same instance"; return 0; } -int -ospf6_lsa_lsd_num (struct ospf6_lsa_header *lsa_header) -{ - int ldnum = 0; - u_int16_t len; - - len = ntohs (lsa_header->length); - len -= sizeof (struct ospf6_lsa_header); - if (lsa_header->type == htons (OSPF6_LSA_TYPE_ROUTER)) - { - len -= sizeof (struct ospf6_router_lsa); - ldnum = len / sizeof (struct ospf6_router_lsd); - } - else /* (lsa_header->type == htons (OSPF6_LSA_TYPE_NETWORK)) */ - { - len -= sizeof (struct ospf6_network_lsa); - ldnum = len / sizeof (u_int32_t); - } - - return ldnum; -} - -void * -ospf6_lsa_lsd_get (int index, struct ospf6_lsa_header *lsa_header) -{ - void *p; - struct ospf6_router_lsa *router_lsa; - struct ospf6_router_lsd *router_lsd; - struct ospf6_network_lsa *network_lsa; - struct ospf6_network_lsd *network_lsd; - - if (lsa_header->type == htons (OSPF6_LSA_TYPE_ROUTER)) - { - router_lsa = (struct ospf6_router_lsa *) (lsa_header + 1); - router_lsd = (struct ospf6_router_lsd *) (router_lsa + 1); - router_lsd += index; - p = (void *) router_lsd; - } - else if (lsa_header->type == htons (OSPF6_LSA_TYPE_NETWORK)) - { - network_lsa = (struct ospf6_network_lsa *) (lsa_header + 1); - network_lsd = (struct ospf6_network_lsd *) (network_lsa + 1); - network_lsd += index; - p = (void *) network_lsd; - } - else - { - p = (void *) NULL; - } - - return p; -} - -/* network_lsd <-> router_lsd */ -static int -ospf6_lsa_lsd_network_reference_match (struct ospf6_network_lsd *network_lsd1, - struct ospf6_lsa_header *lsa_header1, - struct ospf6_router_lsd *router_lsd2, - struct ospf6_lsa_header *lsa_header2) +char * +ospf6_lsa_printbuf (struct ospf6_lsa *lsa, char *buf, int size) { - if (network_lsd1->adv_router != lsa_header2->advrtr) - return 0; - if (router_lsd2->type != OSPF6_ROUTER_LSD_TYPE_TRANSIT_NETWORK) - return 0; - if (router_lsd2->neighbor_router_id != lsa_header1->advrtr) - return 0; - if (router_lsd2->neighbor_interface_id != lsa_header1->ls_id) - return 0; - return 1; + char id[16], adv_router[16]; + inet_ntop (AF_INET, &lsa->header->id, id, sizeof (id)); + inet_ntop (AF_INET, &lsa->header->adv_router, adv_router, + sizeof (adv_router)); + snprintf (buf, size, "[%s Id:%s Adv:%s]", + OSPF6_LSTYPE_NAME (lsa->header->type), id, adv_router); + return buf; } -/* router_lsd <-> router_lsd */ -static int -ospf6_lsa_lsd_router_reference_match (struct ospf6_router_lsd *router_lsd1, - struct ospf6_lsa_header *lsa_header1, - struct ospf6_router_lsd *router_lsd2, - struct ospf6_lsa_header *lsa_header2) +void +ospf6_lsa_header_print_raw (struct ospf6_lsa_header *header) { - if (router_lsd1->type != OSPF6_ROUTER_LSD_TYPE_POINTTOPOINT) - return 0; - if (router_lsd2->type != OSPF6_ROUTER_LSD_TYPE_POINTTOPOINT) - return 0; - if (router_lsd1->neighbor_router_id != lsa_header2->advrtr) - return 0; - if (router_lsd2->neighbor_router_id != lsa_header1->advrtr) - return 0; - if (router_lsd1->neighbor_interface_id != router_lsd2->interface_id) - return 0; - if (router_lsd2->neighbor_interface_id != router_lsd1->interface_id) - return 0; - return 1; + char id[16], adv_router[16]; + inet_ntop (AF_INET, &header->id, id, sizeof (id)); + inet_ntop (AF_INET, &header->adv_router, adv_router, + sizeof (adv_router)); + zlog_info (" [%s Id:%s Adv:%s]", + OSPF6_LSTYPE_NAME (header->type), id, adv_router); + zlog_info (" Age: %4hu SeqNum: %#08lx Cksum: %04hx Len: %d", + ntohs (header->age), (u_long) ntohl (header->seqnum), + ntohs (header->checksum), ntohs (header->length)); } -int -ospf6_lsa_lsd_is_refer_ok (int index1, struct ospf6_lsa_header *lsa_header1, - int index2, struct ospf6_lsa_header *lsa_header2) +void +ospf6_lsa_header_print (struct ospf6_lsa *lsa) { - struct ospf6_router_lsd *r1, *r2; - struct ospf6_network_lsd *n; - - r1 = (struct ospf6_router_lsd *) NULL; - r2 = (struct ospf6_router_lsd *) NULL; - n = (struct ospf6_network_lsd *) NULL; - if (lsa_header1->type == htons (OSPF6_LSA_TYPE_ROUTER)) - r1 = (struct ospf6_router_lsd *) ospf6_lsa_lsd_get (index1, lsa_header1); - else - n = (struct ospf6_network_lsd *) ospf6_lsa_lsd_get (index1, lsa_header1); - - if (lsa_header2->type == htons (OSPF6_LSA_TYPE_ROUTER)) - r2 = (struct ospf6_router_lsd *) ospf6_lsa_lsd_get (index2, lsa_header2); - else - n = (struct ospf6_network_lsd *) ospf6_lsa_lsd_get (index2, lsa_header2); - - if (r1 && r2) - return ospf6_lsa_lsd_router_reference_match (r1, lsa_header1, - r2, lsa_header2); - else if (r1 && n) - return ospf6_lsa_lsd_network_reference_match (n, lsa_header2, - r1, lsa_header1); - else if (n && r2) - return ospf6_lsa_lsd_network_reference_match (n, lsa_header1, - r2, lsa_header2); - return 0; + ospf6_lsa_age_current (lsa); + ospf6_lsa_header_print_raw (lsa->header); } void ospf6_lsa_show (struct vty *vty, struct ospf6_lsa *lsa) { - char adv_router[64], id[64], type[32]; + char adv_router[64], id[64]; + int index; - assert (lsa); - assert (lsa->header); + assert (lsa && lsa->header); - ospf6_lsa_type_string (lsa->header->type, type, sizeof (type)); inet_ntop (AF_INET, &lsa->header->id, id, sizeof (id)); inet_ntop (AF_INET, &lsa->header->adv_router, adv_router, sizeof (adv_router)); vty_out (vty, "%s", VTY_NEWLINE); vty_out (vty, "Age: %4hu Type: %s%s", ospf6_lsa_age_current (lsa), - type, VTY_NEWLINE); + OSPF6_LSTYPE_NAME (lsa->header->type), VTY_NEWLINE); vty_out (vty, "Link State ID: %s%s", id, VTY_NEWLINE); vty_out (vty, "Advertising Router: %s%s", adv_router, VTY_NEWLINE); - vty_out (vty, "LS Sequence Number: %#lx%s", (u_long)ntohl (lsa->header->seqnum), - VTY_NEWLINE); - vty_out (vty, "CheckSum: %#hx Length: %hu%s", ntohs (lsa->header->checksum), + vty_out (vty, "LS Sequence Number: %#010lx%s", + (u_long) ntohl (lsa->header->seqnum), VTY_NEWLINE); + vty_out (vty, "CheckSum: %#06hx Length: %hu%s", + ntohs (lsa->header->checksum), ntohs (lsa->header->length), VTY_NEWLINE); - { - struct ospf6_lsa_slot *slot; - slot = ospf6_lsa_slot_get (lsa->header->type); - if (slot) - { - (*slot->func_show) (vty, lsa); - vty_out (vty, "%s", VTY_NEWLINE); - return; - } - } - - vty_out (vty, "%sUnknown LSA type ...%s", VTY_NEWLINE, VTY_NEWLINE); + index = OSPF6_LSTYPE_INDEX (ntohs (lsa->header->type)); + if (ospf6_lstype[index].show) + (*ospf6_lstype[index].show) (vty, lsa); + else + vty_out (vty, "%sUnknown LSA type ...%s", VTY_NEWLINE, VTY_NEWLINE); + + vty_out (vty, "%s", VTY_NEWLINE); } void @@ -439,25 +314,24 @@ ospf6_lsa_show_summary_header (struct vty *vty) void ospf6_lsa_show_summary (struct vty *vty, struct ospf6_lsa *lsa) { - char adv_router[16], id[16], type[16]; + char adv_router[16], id[16]; struct timeval now, res; char duration[16]; assert (lsa); assert (lsa->header); - memset (type, 0, sizeof (type)); - ospf6_lsa_type_string (lsa->header->type, type, 13); inet_ntop (AF_INET, &lsa->header->id, id, sizeof (id)); inet_ntop (AF_INET, &lsa->header->adv_router, adv_router, sizeof (adv_router)); gettimeofday (&now, NULL); - ospf6_timeval_sub (&now, &lsa->installed, &res); - ospf6_timeval_string_summary (&res, duration, sizeof (duration)); + timersub (&now, &lsa->installed, &res); + timerstring (&res, duration, sizeof (duration)); vty_out (vty, "%-12s %-15s %-15s %4hu %8lx %04hx %4hu %8s%s", - type, id, adv_router, ospf6_lsa_age_current (lsa), + OSPF6_LSTYPE_NAME (lsa->header->type), + id, adv_router, ospf6_lsa_age_current (lsa), (u_long) ntohl (lsa->header->seqnum), ntohs (lsa->header->checksum), ntohs (lsa->header->length), duration, VTY_NEWLINE); @@ -473,7 +347,7 @@ ospf6_lsa_show_dump (struct vty *vty, struct ospf6_lsa *lsa) end = (char *) lsa->header + ntohs (lsa->header->length); vty_out (vty, "%s", VTY_NEWLINE); - vty_out (vty, "%s:%s", lsa->str, VTY_NEWLINE); + vty_out (vty, "%s:%s", lsa->name, VTY_NEWLINE); for (current = start; current < end; current ++) { @@ -489,149 +363,136 @@ ospf6_lsa_show_dump (struct vty *vty, struct ospf6_lsa *lsa) vty_out (vty, "%s%s", VTY_NEWLINE, VTY_NEWLINE); } +void +ospf6_lsa_show_internal (struct vty *vty, struct ospf6_lsa *lsa) +{ + char adv_router[64], id[64]; + + assert (lsa && lsa->header); + + inet_ntop (AF_INET, &lsa->header->id, id, sizeof (id)); + inet_ntop (AF_INET, &lsa->header->adv_router, + adv_router, sizeof (adv_router)); + + vty_out (vty, "%s", VTY_NEWLINE); + vty_out (vty, "Age: %4hu Type: %s%s", ospf6_lsa_age_current (lsa), + OSPF6_LSTYPE_NAME (lsa->header->type), VTY_NEWLINE); + vty_out (vty, "Link State ID: %s%s", id, VTY_NEWLINE); + vty_out (vty, "Advertising Router: %s%s", adv_router, VTY_NEWLINE); + vty_out (vty, "LS Sequence Number: %#010lx%s", + (u_long) ntohl (lsa->header->seqnum), VTY_NEWLINE); + vty_out (vty, "CheckSum: %#06hx Length: %hu%s", + ntohs (lsa->header->checksum), + ntohs (lsa->header->length), VTY_NEWLINE); + vty_out (vty, " Prev: %p This: %p Next: %p%s", + lsa->prev, lsa, lsa->next, VTY_NEWLINE); + vty_out (vty, " Reference count: %ld%s", lsa->refcnt, VTY_NEWLINE); + vty_out (vty, " Reference source: %s (%p) %s", + (lsa->refsrc ? lsa->refsrc->name : "None"), + lsa->refsrc, VTY_NEWLINE); + vty_out (vty, "%s", VTY_NEWLINE); +} + /* OSPFv3 LSA creation/deletion function */ /* calculate LS sequence number for my new LSA. return value is network byte order */ -static signed long -ospf6_lsa_seqnum_new (u_int16_t type, u_int32_t id, u_int32_t adv_router, +u_int32_t +ospf6_lsa_new_seqnum (u_int16_t type, u_int32_t id, u_int32_t adv_router, void *scope) { + struct ospf6_lsdb *lsdb = NULL; struct ospf6_lsa *lsa; - signed long seqnum; + signed long seqnum = 0; /* get current database copy */ - lsa = ospf6_lsdb_lookup (type, id, adv_router, scope); + lsdb = ospf6_get_scoped_lsdb (type, scope); + if (lsdb == NULL) + { + zlog_warn ("Can't decide scoped LSDB"); + return ((u_int32_t) htonl (INITIAL_SEQUENCE_NUMBER)); + } /* if current database copy not found, return InitialSequenceNumber */ - if (!lsa) + lsa = ospf6_lsdb_lookup (type, id, adv_router, lsdb); + if (lsa == NULL) seqnum = INITIAL_SEQUENCE_NUMBER; else seqnum = (signed long) ntohl (lsa->header->seqnum) + 1; - return (htonl (seqnum)); -} - -#if 0 -static void -ospf6_lsa_header_set (u_int16_t type, u_int32_t ls_id, u_int32_t advrtr, - struct ospf6_lsa_header *lsa_header, int bodysize) -{ - /* fill LSA header */ - lsa_header->age = 0; - lsa_header->type = type; - lsa_header->ls_id = ls_id; - lsa_header->advrtr = advrtr; - lsa_header->seqnum = - ospf6_lsa_seqnum_new (lsa_header->type, lsa_header->ls_id, - lsa_header->advrtr); - lsa_header->length = htons (sizeof (struct ospf6_lsa_header) + bodysize); - - /* LSA checksum */ - ospf6_lsa_checksum (lsa_header); + return ((u_int32_t) htonl (seqnum)); } -#endif /*0*/ struct ospf6_lsa * -ospf6_lsa_create (struct ospf6_lsa_header *source) +ospf6_lsa_create (struct ospf6_lsa_header *header) { struct ospf6_lsa *lsa = NULL; - struct ospf6_lsa_header *lsa_header = NULL; + struct ospf6_lsa_header *new_header = NULL; u_int16_t lsa_size = 0; - char buf_router[16], buf_id[16], typebuf[32]; - /* whole length of this LSA */ - lsa_size = ntohs (source->length); + /* size of the entire LSA */ + lsa_size = ntohs (header->length); /* XXX vulnerable */ /* allocate memory for this LSA */ - lsa_header = (struct ospf6_lsa_header *) + new_header = (struct ospf6_lsa_header *) XMALLOC (MTYPE_OSPF6_LSA, lsa_size); - if (! lsa_header) - { - zlog_err ("Can't allocate memory for LSA Header"); - return (struct ospf6_lsa *) NULL; - } - memset (lsa_header, 0, lsa_size); - /* copy LSA from source */ - memcpy (lsa_header, source, lsa_size); + /* copy LSA from original header */ + memcpy (new_header, header, lsa_size); /* LSA information structure */ /* allocate memory */ lsa = (struct ospf6_lsa *) - XMALLOC (MTYPE_OSPF6_LSA, sizeof (struct ospf6_lsa)); + XMALLOC (MTYPE_OSPF6_LSA, sizeof (struct ospf6_lsa)); memset (lsa, 0, sizeof (struct ospf6_lsa)); - lsa->lsa_hdr = (struct ospf6_lsa_hdr *) lsa_header; - lsa->header = (struct ospf6_lsa_header__ *) lsa_header; - - lsa->summary = 0; /* this is not LSA summary */ + lsa->header = (struct ospf6_lsa_header *) new_header; + lsa->headeronly = 0; /* this is not header only */ /* dump string */ - inet_ntop (AF_INET, &lsa->header->id, buf_id, sizeof (buf_id)); - inet_ntop (AF_INET, &lsa->header->adv_router, buf_router, - sizeof (buf_router)); - snprintf (lsa->str, sizeof (lsa->str), "[%s ID=%s Adv=%s]", - ospf6_lsa_type_string (lsa_header->type, typebuf, - sizeof (typebuf)), - buf_id, buf_router); + ospf6_lsa_printbuf (lsa, lsa->name, sizeof (lsa->name)); /* calculate birth, expire and refresh of this lsa */ ospf6_lsa_age_set (lsa); -#ifdef DEBUG - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Create: %s (%p/%p)", lsa->str, lsa, lsa->header); -#endif /*DEBUG*/ + if (IS_OSPF6_DEBUG_LSA (MEMORY)) + zlog_info ("Create LSA Memory: %s (%p/%p)", + lsa->name, lsa, lsa->header); return lsa; } struct ospf6_lsa * -ospf6_lsa_summary_create (struct ospf6_lsa_header__ *source) +ospf6_lsa_create_headeronly (struct ospf6_lsa_header *header) { struct ospf6_lsa *lsa = NULL; - struct ospf6_lsa_header *lsa_header = NULL; - u_int16_t lsa_size = 0; - char buf_router[16], buf_id[16], typebuf[16]; - - /* LSA summary contains LSA Header only */ - lsa_size = sizeof (struct ospf6_lsa_header); + struct ospf6_lsa_header *new_header = NULL; /* allocate memory for this LSA */ - lsa_header = (struct ospf6_lsa_header *) - XMALLOC (MTYPE_OSPF6_LSA_SUMMARY, lsa_size); - memset (lsa_header, 0, lsa_size); + new_header = (struct ospf6_lsa_header *) + XMALLOC (MTYPE_OSPF6_LSA, sizeof (struct ospf6_lsa_header)); - /* copy LSA from source */ - memcpy (lsa_header, source, lsa_size); + /* copy LSA from original header */ + memcpy (new_header, header, sizeof (struct ospf6_lsa_header)); /* LSA information structure */ /* allocate memory */ lsa = (struct ospf6_lsa *) - XMALLOC (MTYPE_OSPF6_LSA_SUMMARY, sizeof (struct ospf6_lsa)); + XMALLOC (MTYPE_OSPF6_LSA, sizeof (struct ospf6_lsa)); memset (lsa, 0, sizeof (struct ospf6_lsa)); - lsa->lsa_hdr = (struct ospf6_lsa_hdr *) lsa_header; - lsa->header = (struct ospf6_lsa_header__ *) lsa_header; - lsa->summary = 1; /* this is LSA summary */ + lsa->header = (struct ospf6_lsa_header *) new_header; + lsa->headeronly = 1; /* this is header only */ /* dump string */ - inet_ntop (AF_INET, &lsa->header->id, buf_id, sizeof (buf_id)); - inet_ntop (AF_INET, &lsa->header->adv_router, buf_router, - sizeof (buf_router)); - snprintf (lsa->str, sizeof (lsa->str), "[%s Summary ID=%s Adv=%s]", - ospf6_lsa_type_string (lsa->header->type, typebuf, - sizeof (typebuf)), - buf_id, buf_router); + ospf6_lsa_printbuf (lsa, lsa->name, sizeof (lsa->name)); /* calculate birth, expire and refresh of this lsa */ ospf6_lsa_age_set (lsa); -#ifdef DEBUG - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Create: %s (%p/%p)", lsa->str, lsa, lsa->header); -#endif /*DEBUG*/ + if (IS_OSPF6_DEBUG_LSA (MEMORY)) + zlog_info ("Create LSA (Header-only) Memory: %s (%p/%p)", + lsa->name, lsa, lsa->header); return lsa; } @@ -639,40 +500,48 @@ ospf6_lsa_summary_create (struct ospf6_lsa_header__ *source) void ospf6_lsa_delete (struct ospf6_lsa *lsa) { - /* just to make sure */ - if (lsa->lock != 0) - { - zlog_err ("Can't delete %s: lock: %ld", lsa->str, lsa->lock); - return; - } + assert (lsa->lock == 0); /* cancel threads */ - if (lsa->expire) - thread_cancel (lsa->expire); - lsa->expire = (struct thread *) NULL; - if (lsa->refresh) - thread_cancel (lsa->refresh); - lsa->refresh = (struct thread *) NULL; + THREAD_OFF (lsa->expire); + THREAD_OFF (lsa->refresh); -#ifdef DEBUG - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Delete %s (%p/%p)", lsa->str, lsa, lsa->header); -#endif /*DEBUG*/ + if (IS_OSPF6_DEBUG_LSA (MEMORY)) + zlog_info ("Delete LSA %s Memory: %s (%p/%p)", + (lsa->headeronly ? "(Header-only) " : ""), + lsa->name, lsa, lsa->header); /* do free */ - if (lsa->summary) - XFREE (MTYPE_OSPF6_LSA_SUMMARY, lsa->header); - else - XFREE (MTYPE_OSPF6_LSA, lsa->header); - lsa->header = NULL; + XFREE (MTYPE_OSPF6_LSA, lsa->header); + XFREE (MTYPE_OSPF6_LSA, lsa); +} - if (lsa->summary) - XFREE (MTYPE_OSPF6_LSA_SUMMARY, lsa); +struct ospf6_lsa * +ospf6_lsa_copy (struct ospf6_lsa *lsa) +{ + struct ospf6_lsa *copy = NULL; + + if (IS_OSPF6_DEBUG_LSA (MEMORY)) + zlog_info ("Create LSA Copy from %s", lsa->name); + + ospf6_lsa_age_current (lsa); + if (lsa->headeronly) + copy = ospf6_lsa_create_headeronly (lsa->header); else - XFREE (MTYPE_OSPF6_LSA, lsa); + copy = ospf6_lsa_create (lsa->header); + assert (copy->lock == 0); + + copy->installed = lsa->installed; + copy->originated = lsa->originated; + copy->scope = lsa->scope; + + copy->refsrc = lsa; + copy->refsrc->refcnt++; + + return copy; } -/* increment reference counter of struct ospf6_lsa */ +/* increment reference counter of struct ospf6_lsa */ void ospf6_lsa_lock (struct ospf6_lsa *lsa) { @@ -680,163 +549,156 @@ ospf6_lsa_lock (struct ospf6_lsa *lsa) return; } -/* decrement reference counter of struct ospf6_lsa */ +/* decrement reference counter of struct ospf6_lsa */ void ospf6_lsa_unlock (struct ospf6_lsa *lsa) { /* decrement reference counter */ - if (lsa->lock > 0) - lsa->lock--; - else - zlog_warn ("Can't unlock %s: already no lock", lsa->str); + assert (lsa->lock > 0); + lsa->lock--; - if (lsa->lock == 0) - ospf6_lsa_delete (lsa); + if (lsa->lock != 0) + return; + + if (lsa->refsrc) + lsa->refsrc->refcnt--; + + ospf6_lsa_delete (lsa); } void -ospf6_lsa_originate (u_int16_t type, u_int32_t id, u_int32_t adv_router, - char *data, int data_len, void *scope) +ospf6_lsa_originate (struct ospf6_lsa *lsa) { - char buffer[MAXLSASIZE]; - struct ospf6_lsa_header *lsa_header; - struct ospf6_lsa *lsa; struct ospf6_lsa *old; - - assert (data_len <= sizeof (buffer) - sizeof (struct ospf6_lsa_header)); - - lsa_header = (struct ospf6_lsa_header *) buffer; - - /* Copy LSA Body */ - memcpy (buffer + sizeof (struct ospf6_lsa_header), data, data_len); - - /* Fill LSA Header */ - lsa_header->age = 0; - lsa_header->type = type; - lsa_header->ls_id = id; - lsa_header->advrtr = adv_router; - lsa_header->seqnum = - ospf6_lsa_seqnum_new (lsa_header->type, lsa_header->ls_id, - lsa_header->advrtr, scope); - lsa_header->length = htons (sizeof (struct ospf6_lsa_header) + data_len); - - /* LSA checksum */ - ospf6_lsa_checksum (lsa_header); - - /* create LSA */ - lsa = ospf6_lsa_create ((struct ospf6_lsa_header *) buffer); - lsa->scope = scope; + struct ospf6_lsdb *lsdb = NULL; /* find previous LSA */ + lsdb = ospf6_get_scoped_lsdb (lsa->header->type, lsa->scope); + if (lsdb == NULL) + { + zlog_warn ("Can't decide scoped LSDB"); + ospf6_lsa_delete (lsa); + return; + } + old = ospf6_lsdb_lookup (lsa->header->type, lsa->header->id, - lsa->header->adv_router, lsa->scope); + lsa->header->adv_router, lsdb); if (old) { - /* Check if this is neither different instance nor refresh, return */ - if (! CHECK_FLAG (old->flag, OSPF6_LSA_FLAG_REFRESH) && - ! ospf6_lsa_differ (lsa, old)) + /* If this origination is neither different instance nor refresh, + suppress this origination */ + if (! CHECK_FLAG (old->flag, OSPF6_LSA_REFRESH) && + ! OSPF6_LSA_IS_DIFFER (lsa, old)) { - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: Suppress updating %s", lsa->str); + if (IS_OSPF6_DEBUG_LSA (ORIGINATE)) + zlog_info ("Suppress updating LSA: %s", lsa->name); ospf6_lsa_delete (lsa); return; } } lsa->refresh = thread_add_timer (master, ospf6_lsa_refresh, lsa, - OSPF6_LS_REFRESH_TIME); - gettimeofday (&lsa->originated, NULL); - - //if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: originate %s seq: %#x age: %hu %ld.%06ld", - lsa->str, ntohl (lsa->header->seqnum), - ospf6_lsa_age_current (lsa), - lsa->originated.tv_sec, lsa->originated.tv_usec); - - ospf6_dbex_remove_from_all_retrans_list (lsa); - ospf6_dbex_flood (lsa, NULL); - ospf6_lsdb_install (lsa); + LS_REFRESH_TIME); + + if (IS_OSPF6_DEBUG_LSA (ORIGINATE)) + { + zlog_info ("LSA Originate:"); + ospf6_lsa_header_print (lsa); + } + + if (old) + ospf6_flood_clear (old); + ospf6_flood_lsa (lsa, NULL); + ospf6_install_lsa (lsa, lsdb); +} + +void +ospf6_lsa_re_originate (struct ospf6_lsa *lsa) +{ + u_int16_t index; + + if (IS_OSPF6_DEBUG_LSA (ORIGINATE)) + { + zlog_info ("LSA Reoriginate:"); + ospf6_lsa_header_print (lsa); + } + + index = OSPF6_LSTYPE_INDEX (ntohs (lsa->header->type)); + if (ospf6_lstype[index].reoriginate) + (*ospf6_lstype[index].reoriginate) (lsa); + else + ospf6_lsa_premature_aging (lsa); } -/* ospf6_lsa expired */ +/* ospf6 lsa expiry */ int ospf6_lsa_expire (struct thread *thread) { struct ospf6_lsa *lsa; struct ospf6_lsdb *lsdb = NULL; - void (*hook) (struct ospf6_lsa *, struct ospf6_lsa *); lsa = (struct ospf6_lsa *) THREAD_ARG (thread); - assert (lsa && lsa->lsa_hdr); - /* assertion */ - assert (IS_LSA_MAXAGE (lsa)); - assert (!lsa->refresh); + assert (lsa && lsa->header); + assert (OSPF6_LSA_IS_MAXAGE (lsa)); + assert (! lsa->refresh); lsa->expire = (struct thread *) NULL; - /* log */ - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: Expire: %s", lsa->str); - - if (!lsa->summary) + if (IS_OSPF6_DEBUG_LSA (TIMER)) { - /* reflood lsa */ - ospf6_dbex_flood (lsa, NULL); - - /* get scoped lsdb, call remove hook */ - if (OSPF6_LSA_IS_SCOPE_LINKLOCAL (ntohs (lsa->header->type))) - lsdb = ((struct ospf6_interface *) lsa->scope)->lsdb; - else if (OSPF6_LSA_IS_SCOPE_AREA (ntohs (lsa->header->type))) - lsdb = ((struct ospf6_area *) lsa->scope)->lsdb; - else if (OSPF6_LSA_IS_SCOPE_AS (ntohs (lsa->header->type))) - lsdb = ((struct ospf6 *) lsa->scope)->lsdb; - else - assert (0); + zlog_info ("LSA Expire:"); + ospf6_lsa_header_print (lsa); + } - /* call LSDB hook to re-process LSA */ - hook = ospf6_lsdb_hook[ntohs (lsa->header->type) & - OSPF6_LSTYPE_CODE_MASK].hook; - if (hook) - (*hook) (NULL, lsa); + if (lsa->headeronly) + return 0; /* dbexchange will do something ... */ - /* do not free LSA, and do nothing about lslists. - wait event (ospf6_lsdb_check_maxage) */ + /* reflood lsa */ + ospf6_flood_lsa (lsa, NULL); + + /* reinstall lsa */ + lsdb = ospf6_get_scoped_lsdb (lsa->header->type, lsa->scope); + if (lsdb == NULL) + { + zlog_warn ("Can't decide scoped LSDB: %s", lsa->name); + return 0; } + if (IS_OSPF6_DEBUG_LSA (DATABASE)) + zlog_info ("Reinstall MaxAge %s", lsa->name); + ospf6_lsdb_add (lsa, lsdb); + + /* schedule maxage remover */ + ospf6_maxage_remove (ospf6); return 0; } +/* Below will become dummy thread. + refresh function must be set individually per each LSAs */ int ospf6_lsa_refresh (struct thread *thread) { struct ospf6_lsa *lsa; - struct ospf6_lsa_slot *slot; assert (thread); - lsa = (struct ospf6_lsa *) THREAD_ARG (thread); - assert (lsa && lsa->lsa_hdr); + lsa = (struct ospf6_lsa *) THREAD_ARG (thread); + assert (lsa && lsa->header); - /* this will be used later as flag to decide really originate */ lsa->refresh = (struct thread *) NULL; - SET_FLAG (lsa->flag, OSPF6_LSA_FLAG_REFRESH); - /* log */ - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA Refresh: %s", lsa->str); + /* this will be used later to decide really originate or not */ + SET_FLAG (lsa->flag, OSPF6_LSA_REFRESH); - slot = ospf6_lsa_slot_get (lsa->header->type); - if (slot) + if (IS_OSPF6_DEBUG_LSA (ORIGINATE)) { - zlog_info ("LSA Refresh: %s", slot->name); - (*slot->func_refresh) (lsa); - return 0; + zlog_info ("LSA Refresh:"); + ospf6_lsa_header_print (lsa); } - zlog_warn ("Can't Refresh LSA: Unknown type: %#x", - ntohs (lsa->header->type)); - return 1; + ospf6_lsa_re_originate (lsa); + return 0; } @@ -885,1042 +747,205 @@ ospf6_lsa_checksum (struct ospf6_lsa_header *lsa_header) } int -ospf6_lsa_is_known_type (struct ospf6_lsa_header *lsa_header) +ospf6_unknown_reoriginate (struct ospf6_lsa *lsa) { - struct ospf6_lsa_slot *slot; - - slot = ospf6_lsa_slot_get (lsa_header->type); - if (slot) - return 1; + ospf6_lsa_premature_aging (lsa); return 0; } -struct ospf6_lsa_slot *slot_head = NULL; - -struct ospf6_lsa_slot * -ospf6_lsa_slot_get (u_int16_t type) -{ - struct ospf6_lsa_slot *slot; - - for (slot = slot_head; slot; slot = slot->next) - { - if (slot->type == type) - return slot; - } - - return NULL; -} - int -ospf6_lsa_slot_register (struct ospf6_lsa_slot *src) +ospf6_unknown_show (struct vty *vty, struct ospf6_lsa *lsa) { - struct ospf6_lsa_slot *new, *slot; - - slot = ospf6_lsa_slot_get (src->type); - if (slot) - { - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: Slot register: already exists: %#x %s", - slot->type, slot->name); - return -1; - } - - new = (struct ospf6_lsa_slot *) - XMALLOC (MTYPE_OSPF6_LSA, sizeof (struct ospf6_lsa_slot)); - if (! new) - { - zlog_err ("Can't allocate memory for LSA slot: %s", strerror (errno)); - return -1; - } - memset (new, 0, sizeof (struct ospf6_lsa_slot)); - memcpy (new, src, sizeof (struct ospf6_lsa_slot)); + u_char *start, *end, *current; + char byte[4]; - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: Slot register: %#x %s", slot->type, slot->name); + start = (char *) lsa->header + sizeof (struct ospf6_lsa_header); + end = (char *) lsa->header + ntohs (lsa->header->length); - if (slot_head == NULL) + vty_out (vty, " Unknown contents:%s", VTY_NEWLINE); + for (current = start; current < end; current ++) { - new->prev = NULL; - new->next = NULL; - slot_head = new; - return 0; - } - - slot = slot_head; - while (slot->next) - slot = slot->next; - - slot->next = new; - new->prev = slot; - - return 0; -} - -int -ospf6_lsa_slot_unregister (u_int16_t type) -{ - struct ospf6_lsa_slot *slot; + if ((current - start) % 16 == 0) + vty_out (vty, "%s ", VTY_NEWLINE); + else if ((current - start) % 4 == 0) + vty_out (vty, " "); - slot = ospf6_lsa_slot_get (type); - if (slot == NULL) - { - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Registering LSA slot: no such slot: %#x", type); - return -1; + snprintf (byte, sizeof (byte), "%02x", *current); + vty_out (vty, "%s", byte); } - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Unregistering LSA Slot: %#x %s", slot->type, slot->name); - - if (slot->prev) - slot->prev->next = slot->next; - if (slot->next) - slot->next->prev = slot->prev; - - if (slot_head == slot) - slot_head = slot->next; - - XFREE (MTYPE_OSPF6_LSA, slot); + vty_out (vty, "%s%s", VTY_NEWLINE, VTY_NEWLINE); return 0; } -char * -ospf6_lsa_type_string (u_int16_t type, char *buf, int bufsize) +void +ospf6_lsa_init () { - struct ospf6_lsa_slot *slot; + memset (ospf6_lstype, 0, sizeof (ospf6_lstype)); - slot = ospf6_lsa_slot_get (type); - if (slot) - snprintf (buf, bufsize, "%s", slot->name); - else - snprintf (buf, bufsize, "Type=0x%04x", ntohs (type)); - - return buf; + ospf6_lstype[0].name = "Unknown"; + ospf6_lstype[0].reoriginate = ospf6_unknown_reoriginate; + ospf6_lstype[0].show = ospf6_unknown_show; } -/*******************/ -/* LSA Origination */ -/*******************/ - -#define CONTINUE_IF_ADDRESS_LINKLOCAL(addr)\ - if (IN6_IS_ADDR_LINKLOCAL (&(addr)->u.prefix6))\ - {\ - char buf[64];\ - prefix2str (addr, buf, sizeof (buf));\ - if (IS_OSPF6_DUMP_LSA)\ - zlog_info (" Filter out Linklocal: %s", buf);\ - continue;\ - } - -#define CONTINUE_IF_ADDRESS_UNSPECIFIED(addr)\ - if (IN6_IS_ADDR_UNSPECIFIED (&(addr)->u.prefix6))\ - {\ - char buf[64];\ - prefix2str (addr, buf, sizeof (buf));\ - if (IS_OSPF6_DUMP_LSA)\ - zlog_info (" Filter out Unspecified: %s", buf);\ - continue;\ - } - -#define CONTINUE_IF_ADDRESS_LOOPBACK(addr)\ - if (IN6_IS_ADDR_LOOPBACK (&(addr)->u.prefix6))\ - {\ - char buf[64];\ - prefix2str (addr, buf, sizeof (buf));\ - if (IS_OSPF6_DUMP_LSA)\ - zlog_info (" Filter out Loopback: %s", buf);\ - continue;\ - } - -#define CONTINUE_IF_ADDRESS_V4COMPAT(addr)\ - if (IN6_IS_ADDR_V4COMPAT (&(addr)->u.prefix6))\ - {\ - char buf[64];\ - prefix2str (addr, buf, sizeof (buf));\ - if (IS_OSPF6_DUMP_LSA)\ - zlog_info (" Filter out V4Compat: %s", buf);\ - continue;\ - } - -#define CONTINUE_IF_ADDRESS_V4MAPPED(addr)\ - if (IN6_IS_ADDR_V4MAPPED (&(addr)->u.prefix6))\ - {\ - char buf[64];\ - prefix2str (addr, buf, sizeof (buf));\ - if (IS_OSPF6_DUMP_LSA)\ - zlog_info (" Filter out V4Mapped: %s", buf);\ - continue;\ - } - -/******************************/ -/* RFC2740 3.4.3.1 Router-LSA */ -/******************************/ - -char * -ospf6_lsa_router_bits_string (u_char router_bits, char *buf, int size) -{ - char w, v, e, b; - - w = (router_bits & OSPF6_ROUTER_LSA_BIT_W ? 'W' : '-'); - v = (router_bits & OSPF6_ROUTER_LSA_BIT_V ? 'V' : '-'); - e = (router_bits & OSPF6_ROUTER_LSA_BIT_E ? 'E' : '-'); - b = (router_bits & OSPF6_ROUTER_LSA_BIT_B ? 'B' : '-'); - snprintf (buf, size, "----%c%c%c%c", w, v, e, b); - return buf; -} - -int -ospf6_lsa_router_show (struct vty *vty, struct ospf6_lsa *lsa) -{ - char *start, *end, *current; - char buf[32], name[32], bits[32], options[32]; - struct ospf6_router_lsa *router_lsa; - struct ospf6_router_lsd *lsdesc; - - assert (lsa->header); - - router_lsa = (struct ospf6_router_lsa *) - ((char *) lsa->header + sizeof (struct ospf6_lsa_header)); - - ospf6_lsa_router_bits_string (router_lsa->bits, bits, sizeof (bits)); - ospf6_options_string (router_lsa->options, options, sizeof (options)); - vty_out (vty, " Bits: %s Options: %s%s", bits, options, VTY_NEWLINE); - - start = (char *) router_lsa + sizeof (struct ospf6_router_lsa); - end = (char *) lsa->header + ntohs (lsa->header->length); - for (current = start; current + sizeof (struct ospf6_router_lsd) <= end; - current += sizeof (struct ospf6_router_lsd)) - { - lsdesc = (struct ospf6_router_lsd *) current; - - if (lsdesc->type == OSPF6_ROUTER_LSD_TYPE_POINTTOPOINT) - snprintf (name, sizeof (name), "Point-To-Point"); - else if (lsdesc->type == OSPF6_ROUTER_LSD_TYPE_TRANSIT_NETWORK) - snprintf (name, sizeof (name), "Transit-Network"); - else if (lsdesc->type == OSPF6_ROUTER_LSD_TYPE_STUB_NETWORK) - snprintf (name, sizeof (name), "Stub-Network"); - else if (lsdesc->type == OSPF6_ROUTER_LSD_TYPE_VIRTUAL_LINK) - snprintf (name, sizeof (name), "Virtual-Link"); - else - snprintf (name, sizeof (name), "Unknown (%#x)", lsdesc->type); - - vty_out (vty, " Type: %s Metric: %d%s", - name, ntohs (lsdesc->metric), VTY_NEWLINE); - vty_out (vty, " Interface ID: %s%s", - inet_ntop (AF_INET, &lsdesc->interface_id, - buf, sizeof (buf)), VTY_NEWLINE); - vty_out (vty, " Neighbor Interface ID: %s%s", - inet_ntop (AF_INET, &lsdesc->neighbor_interface_id, - buf, sizeof (buf)), VTY_NEWLINE); - vty_out (vty, " Neighbor Router ID: %s%s", - inet_ntop (AF_INET, &lsdesc->neighbor_router_id, - buf, sizeof (buf)), VTY_NEWLINE); - } - return 0; -} - -u_long -ospf6_lsa_has_elasped (u_int16_t type, u_int32_t id, - u_int32_t adv_router, void *scope) -{ - struct ospf6_lsa *old; - struct timeval now; - if (adv_router != ospf6->router_id) - zlog_info ("LSA: Router-ID changed ?"); - - old = ospf6_lsdb_lookup (type, id, adv_router, scope); - if (! old) - return OSPF6_LSA_MAXAGE; - - gettimeofday (&now, NULL); - return ((u_long) SEC_TVDIFF (&now, &old->originated)); -} - -int -ospf6_lsa_originate_router (struct thread *thread) -{ - char buffer [MAXLSASIZE]; - u_int16_t size; - struct ospf6_area *o6a; - int count; - u_int32_t area_id; - - struct ospf6_router_lsa *router_lsa; - struct ospf6_router_lsd *router_lsd; - listnode i; - struct ospf6_interface *o6i; - struct ospf6_neighbor *o6n = NULL; - - area_id = (u_int32_t) THREAD_ARG (thread); - - o6a = ospf6_area_lookup (area_id, ospf6); - if (! o6a) - { - inet_ntop (AF_INET, &area_id, buffer, sizeof (buffer)); - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: Update Router-LSA: No such area: %s", buffer); - return 0; +DEFUN (debug_ospf6_lsa_sendrecv, + debug_ospf6_lsa_sendrecv_cmd, + "debug ospf6 lsa (send|recv|originate|timer|database|memory|all)", + DEBUG_STR + OSPF6_STR + "Debug Link State Advertisements (LSAs)\n" + "Debug Sending LSAs\n" + "Debug Receiving LSAs\n" + "Debug Originating LSAs\n" + "Debug Timer Event of LSAs\n" + "Debug LSA Database\n" + "Debug Memory of LSAs\n" + "Debug LSAs all\n" + ) +{ + unsigned char level = 0; + + if (argc) + { + if (! strncmp (argv[0], "s", 1)) + level = OSPF6_DEBUG_LSA_SEND; + else if (! strncmp (argv[0], "r", 1)) + level = OSPF6_DEBUG_LSA_RECV; + else if (! strncmp (argv[0], "o", 1)) + level = OSPF6_DEBUG_LSA_ORIGINATE; + else if (! strncmp (argv[0], "t", 1)) + level = OSPF6_DEBUG_LSA_TIMER; + else if (! strncmp (argv[0], "d", 1)) + level = OSPF6_DEBUG_LSA_DATABASE; + else if (! strncmp (argv[0], "m", 1)) + level = OSPF6_DEBUG_LSA_MEMORY; + else if (! strncmp (argv[0], "a", 1)) + { + level = OSPF6_DEBUG_LSA_SEND | OSPF6_DEBUG_LSA_RECV | + OSPF6_DEBUG_LSA_ORIGINATE | OSPF6_DEBUG_LSA_TIMER | + OSPF6_DEBUG_LSA_DATABASE | OSPF6_DEBUG_LSA_MEMORY; + } } - - /* clear thread */ - o6a->thread_router_lsa = NULL; - - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: originate Router-LSA for Area %s", o6a->str); - - size = sizeof (struct ospf6_router_lsa); - memset (buffer, 0, sizeof (buffer)); - router_lsa = (struct ospf6_router_lsa *) buffer; - - OSPF6_OPT_CLEAR_ALL (router_lsa->options); - OSPF6_OPT_SET (router_lsa->options, OSPF6_OPT_V6); - OSPF6_OPT_SET (router_lsa->options, OSPF6_OPT_E); - OSPF6_OPT_CLEAR (router_lsa->options, OSPF6_OPT_MC); - OSPF6_OPT_CLEAR (router_lsa->options, OSPF6_OPT_N); - OSPF6_OPT_SET (router_lsa->options, OSPF6_OPT_R); - OSPF6_OPT_CLEAR (router_lsa->options, OSPF6_OPT_DC); - - OSPF6_ROUTER_LSA_CLEAR_ALL_BITS (router_lsa); - OSPF6_ROUTER_LSA_CLEAR (router_lsa, OSPF6_ROUTER_LSA_BIT_B); - - if (ospf6_is_asbr (o6a->ospf6)) - OSPF6_ROUTER_LSA_SET (router_lsa, OSPF6_ROUTER_LSA_BIT_E); else - OSPF6_ROUTER_LSA_CLEAR (router_lsa, OSPF6_ROUTER_LSA_BIT_E); - - OSPF6_ROUTER_LSA_CLEAR (router_lsa, OSPF6_ROUTER_LSA_BIT_V); - OSPF6_ROUTER_LSA_CLEAR (router_lsa, OSPF6_ROUTER_LSA_BIT_W); - - /* describe links for each interfaces */ - router_lsd = (struct ospf6_router_lsd *) (router_lsa + 1); - for (i = listhead (o6a->if_list); i; nextnode (i)) { - o6i = (struct ospf6_interface *) getdata (i); - assert (o6i); - - /* Interfaces in state Down or Loopback are not described */ - if (o6i->state == IFS_DOWN || o6i->state == IFS_LOOPBACK) - continue; - - /* Nor are interfaces without any full adjacencies described */ - count = 0; - o6i->foreach_nei (o6i, &count, NBS_FULL, ospf6_count_state); - if (count == 0) - continue; - - /* Point-to-Point interfaces */ - if (if_is_pointopoint (o6i->interface)) + level = OSPF6_DEBUG_LSA_SEND | OSPF6_DEBUG_LSA_RECV | + OSPF6_DEBUG_LSA_ORIGINATE | OSPF6_DEBUG_LSA_TIMER; + } + + OSPF6_DEBUG_LSA_ON (level); + return CMD_SUCCESS; +} + +ALIAS (debug_ospf6_lsa_sendrecv, + debug_ospf6_lsa_cmd, + "debug ospf6 lsa", + NO_STR + DEBUG_STR + OSPF6_STR + "Debug Link State Advertisements (LSAs)\n" + ); + +DEFUN (no_debug_ospf6_lsa_sendrecv, + no_debug_ospf6_lsa_sendrecv_cmd, + "no debug ospf6 lsa (send|recv|originate|timer|database|memory|all)", + NO_STR + DEBUG_STR + OSPF6_STR + "Debug Link State Advertisements (LSAs)\n" + "Debug Sending LSAs\n" + "Debug Receiving LSAs\n" + "Debug Originating LSAs\n" + "Debug Timer Event of LSAs\n" + "Debug LSA Database\n" + "Debug Memory of LSAs\n" + "Debug LSAs all\n" + ) +{ + unsigned char level = 0; + + if (argc) + { + if (! strncmp (argv[0], "s", 1)) + level = OSPF6_DEBUG_LSA_SEND; + else if (! strncmp (argv[0], "r", 1)) + level = OSPF6_DEBUG_LSA_RECV; + else if (! strncmp (argv[0], "o", 1)) + level = OSPF6_DEBUG_LSA_ORIGINATE; + else if (! strncmp (argv[0], "t", 1)) + level = OSPF6_DEBUG_LSA_TIMER; + else if (! strncmp (argv[0], "d", 1)) + level = OSPF6_DEBUG_LSA_DATABASE; + else if (! strncmp (argv[0], "m", 1)) + level = OSPF6_DEBUG_LSA_MEMORY; + else if (! strncmp (argv[0], "a", 1)) { - if (listcount (o6i->neighbor_list) == 0) - continue; - - if (listcount (o6i->neighbor_list) != 1) - zlog_warn ("LSA: Multiple neighbors on PoinToPoint: %s", - o6i->interface->name); - - o6n = (struct ospf6_neighbor *) - getdata (listhead (o6i->neighbor_list)); - assert (o6n); - - router_lsd->type = OSPF6_ROUTER_LSD_TYPE_POINTTOPOINT; - router_lsd->metric = htons (o6i->cost); - router_lsd->interface_id = htonl (o6i->if_id); - router_lsd->neighbor_interface_id = htonl (o6n->ifid); - router_lsd->neighbor_router_id = o6n->router_id; - - size += sizeof (struct ospf6_router_lsd); - router_lsd ++; - - continue; + level = OSPF6_DEBUG_LSA_SEND | OSPF6_DEBUG_LSA_RECV | + OSPF6_DEBUG_LSA_ORIGINATE | OSPF6_DEBUG_LSA_TIMER | + OSPF6_DEBUG_LSA_DATABASE | OSPF6_DEBUG_LSA_MEMORY; } - - /* Broadcast and NBMA interfaces */ - if (if_is_broadcast (o6i->interface)) - { - /* If this router is not DR, - and If this router not fully adjacent with DR, - this interface is not transit yet: ignore. */ - if (o6i->state != IFS_DR) - { - o6n = ospf6_neighbor_lookup (o6i->dr, o6i); /* find DR */ - if (o6n == NULL || o6n->state != NBS_FULL) - continue; - } - else - { - count = 0; - o6i->foreach_nei (o6i, &count, NBS_FULL, ospf6_count_state); - if (count == 0) - continue; - } - - router_lsd->type = OSPF6_ROUTER_LSD_TYPE_TRANSIT_NETWORK; - router_lsd->metric = htons (o6i->cost); - router_lsd->interface_id = htonl (o6i->if_id); - if (o6i->state != IFS_DR) - { - router_lsd->neighbor_interface_id = htonl (o6n->ifid); - router_lsd->neighbor_router_id = o6n->router_id; - } - else - { - router_lsd->neighbor_interface_id = htonl (o6i->if_id); - router_lsd->neighbor_router_id = o6i->area->ospf6->router_id; - } - - size += sizeof (struct ospf6_router_lsd); - router_lsd ++; - - continue; - } - - /* Virtual links */ - /* xxx */ - /* Point-to-Multipoint interfaces */ - /* xxx */ } - - ospf6_lsa_originate (htons (OSPF6_LSA_TYPE_ROUTER), - htonl (0), o6a->ospf6->router_id, - (char *) router_lsa, size, o6a); - return 0; -} - -void -ospf6_lsa_schedule_router (struct ospf6_area *area) -{ - u_long elasped_time, time = 0; - - if (area->thread_router_lsa) - { - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: schedule: Router-LSA for Area %s: another thread", - area->str); - return; - } - - elasped_time = - ospf6_lsa_has_elasped (htons (OSPF6_LSA_TYPE_ROUTER), htonl (0), - area->ospf6->router_id, area); - if (elasped_time < OSPF6_MIN_LS_INTERVAL) - time = (u_long) (OSPF6_MIN_LS_INTERVAL - elasped_time); - else - time = 0; - - if (IS_OSPF6_DUMP_LSA) - zlog_info ("LSA: schedule: Router-LSA for Area %s after %lu sec", - area->str, time); - - if (time) - area->thread_router_lsa = - thread_add_timer (master, ospf6_lsa_originate_router, - (void *) area->area_id, time); else - area->thread_router_lsa = - thread_add_event (master, ospf6_lsa_originate_router, - (void *) area->area_id, 0); -} - -int -ospf6_lsa_router_hook_neighbor (void *neighbor) -{ - struct ospf6_neighbor *o6n = neighbor; - if (o6n->ospf6_interface->area) - ospf6_lsa_schedule_router (o6n->ospf6_interface->area); - return 0; -} - -int -ospf6_lsa_router_hook_interface (void *interface) -{ - struct ospf6_interface *o6i = interface; - if (o6i->area) - ospf6_lsa_schedule_router (o6i->area); - return 0; -} - -int -ospf6_lsa_router_hook_area (void *area) -{ - struct ospf6_area *o6a = area; - ospf6_lsa_schedule_router (o6a); - return 0; -} - -int -ospf6_lsa_router_hook_top (void *ospf6) -{ - struct ospf6 *o6 = ospf6; - struct ospf6_area *o6a; - listnode node; - - for (node = listhead (o6->area_list); node; nextnode (node)) { - o6a = getdata (node); - ospf6_lsa_schedule_router (o6a); + level = OSPF6_DEBUG_LSA_SEND | OSPF6_DEBUG_LSA_RECV | + OSPF6_DEBUG_LSA_ORIGINATE | OSPF6_DEBUG_LSA_TIMER; } - return 0; -} - -int -ospf6_lsa_router_refresh (void *old) -{ - struct ospf6_lsa *lsa = old; - struct ospf6_area *o6a; - - o6a = lsa->scope; - ospf6_lsa_schedule_router (o6a); - return 0; -} - -void -ospf6_lsa_slot_register_router () -{ - struct ospf6_lsa_slot slot; - struct ospf6_hook hook; - - memset (&slot, 0, sizeof (struct ospf6_lsa_slot)); - slot.type = htons (OSPF6_LSA_TYPE_ROUTER); - slot.name = "Router"; - slot.func_show = ospf6_lsa_router_show; - slot.func_refresh = ospf6_lsa_router_refresh; - ospf6_lsa_slot_register (&slot); - - ospf6_lsdb_hook[OSPF6_LSA_TYPE_ROUTER & OSPF6_LSTYPE_CODE_MASK].hook = - ospf6_spf_database_hook; - - memset (&hook, 0, sizeof (hook)); - hook.name = "OriginateRouter"; - hook.hook_change = ospf6_lsa_router_hook_neighbor; - ospf6_hook_register (&hook, &neighbor_hook); - - memset (&hook, 0, sizeof (hook)); - hook.name = "OriginateRouter"; - hook.hook_change = ospf6_lsa_router_hook_interface; - ospf6_hook_register (&hook, &interface_hook); - - memset (&hook, 0, sizeof (hook)); - hook.name = "OriginateRouter"; - hook.hook_change = ospf6_lsa_router_hook_area; - ospf6_hook_register (&hook, &area_hook); - - memset (&hook, 0, sizeof (hook)); - hook.name = "OriginateRouter"; - hook.hook_change = ospf6_lsa_router_hook_top; - ospf6_hook_register (&hook, &top_hook); -} - -/*******************************/ -/* RFC2740 3.4.3.2 Network-LSA */ -/*******************************/ - -int -ospf6_lsa_network_show (struct vty *vty, struct ospf6_lsa *lsa) -{ - char *start, *end, *current; - struct ospf6_network_lsa *network_lsa; - u_int32_t *router_id; - char buf[128], options[32]; - - assert (lsa->header); - network_lsa = (struct ospf6_network_lsa *) (lsa->header + 1); - router_id = (u_int32_t *)(network_lsa + 1); - - ospf6_options_string (network_lsa->options, options, sizeof (options)); - vty_out (vty, " Options: %s%s", options, VTY_NEWLINE); - - start = (char *) network_lsa + sizeof (struct ospf6_network_lsa); - end = (char *) lsa->header + ntohs (lsa->header->length); - for (current = start; current + sizeof (u_int32_t) <= end; - current += sizeof (u_int32_t)) - { - router_id = (u_int32_t *) current; - inet_ntop (AF_INET, router_id, buf, sizeof (buf)); - vty_out (vty, " Attached Router: %s%s", buf, VTY_NEWLINE); - } - return 0; -} - -void -ospf6_lsa_network_update (char *ifname) -{ - char buffer [MAXLSASIZE]; - u_int16_t size; - struct ospf6_lsa *old; - struct interface *ifp; - struct ospf6_interface *o6i; - int count; - - struct ospf6_network_lsa *network_lsa; - struct ospf6_neighbor *o6n; - u_int32_t *router_id; - listnode node; - - ifp = if_lookup_by_name (ifname); - if (! ifp) - { - if (IS_OSPF6_DUMP_LSA) - zlog_warn ("Update Network: No such Interface: %s", ifname); - return; - } - - o6i = (struct ospf6_interface *) ifp->info; - if (! o6i || ! o6i->area) - { - if (IS_OSPF6_DUMP_LSA) - zlog_warn ("Update Network: Interface not enabled: %s", ifname); - return; - } - - /* find previous LSA */ - old = ospf6_lsdb_lookup (htons (OSPF6_LSA_TYPE_NETWORK), - htonl (o6i->if_id), - o6i->area->ospf6->router_id, o6i->area); - - /* Don't originate Network-LSA if not DR */ - if (o6i->state != IFS_DR) - { - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Update Network: Interface %s is not DR", - o6i->interface->name); - if (old) - ospf6_lsa_premature_aging (old); - return; - } - - /* If none of neighbor is adjacent to us */ - count = 0; - o6i->foreach_nei (o6i, &count, NBS_FULL, ospf6_count_state); - if (count == 0) - { - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Update Network: Interface %s is Stub", - o6i->interface->name); - if (old) - ospf6_lsa_premature_aging (old); - return; - } - - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Update Network: Interface %s", o6i->interface->name); - - /* prepare buffer */ - memset (buffer, 0, sizeof (buffer)); - size = sizeof (struct ospf6_network_lsa); - network_lsa = (struct ospf6_network_lsa *) buffer; - router_id = (u_int32_t *)(network_lsa + 1); - - /* set fields of myself */ - *router_id++ = o6i->area->ospf6->router_id; - size += sizeof (u_int32_t); - network_lsa->options[0] |= o6i->area->options[0]; - network_lsa->options[1] |= o6i->area->options[1]; - network_lsa->options[2] |= o6i->area->options[2]; - - /* Walk through neighbors */ - for (node = listhead (o6i->neighbor_list); node; nextnode (node)) - { - o6n = (struct ospf6_neighbor *) getdata (node); - - if (o6n->state != NBS_FULL) - continue; - - /* set this neighbor's Router-ID to LSA */ - *router_id++ = o6n->router_id; - size += sizeof (u_int32_t); - - /* options field is logical OR */ - network_lsa->options[0] |= o6n->options[0]; - network_lsa->options[1] |= o6n->options[1]; - network_lsa->options[2] |= o6n->options[2]; - } - - ospf6_lsa_originate (htons (OSPF6_LSA_TYPE_NETWORK), - htonl (o6i->if_id), o6i->area->ospf6->router_id, - (char *) network_lsa, size, o6i->area); -} -int -ospf6_lsa_network_hook_neighbor (void *neighbor) -{ - struct ospf6_neighbor *o6n = neighbor; - ospf6_lsa_network_update (o6n->ospf6_interface->interface->name); - return 0; + OSPF6_DEBUG_LSA_OFF (level); + return CMD_SUCCESS; } -int -ospf6_lsa_network_hook_interface (void *interface) -{ - struct ospf6_interface *o6i = interface; - if (o6i->area) - ospf6_lsa_network_update (o6i->interface->name); - return 0; -} +ALIAS (no_debug_ospf6_lsa_sendrecv, + no_debug_ospf6_lsa_cmd, + "no debug ospf6 lsa", + NO_STR + DEBUG_STR + OSPF6_STR + "Debug Link State Advertisements (LSAs)\n" + ); int -ospf6_lsa_network_refresh (void *old) +config_write_ospf6_debug_lsa (struct vty *vty) { - struct ospf6_lsa *lsa = old; - struct interface *ifp; - - ifp = if_lookup_by_index (ntohl (lsa->header->id)); - if (! ifp) - ospf6_lsa_premature_aging (old); + if (conf_debug_ospf6_lsa == OSPF6_DEBUG_LSA_ALL) + vty_out (vty, "debug ospf6 lsa all%s", VTY_NEWLINE); else - ospf6_lsa_network_update (ifp->name); - - return 0; -} - -void -ospf6_lsa_slot_register_network () -{ - struct ospf6_lsa_slot slot; - struct ospf6_hook hook; - - memset (&slot, 0, sizeof (struct ospf6_lsa_slot)); - slot.type = htons (OSPF6_LSA_TYPE_NETWORK); - slot.name = "Network"; - slot.func_show = ospf6_lsa_network_show; - slot.func_refresh = ospf6_lsa_network_refresh; - ospf6_lsa_slot_register (&slot); - - ospf6_lsdb_hook[OSPF6_LSA_TYPE_NETWORK & OSPF6_LSTYPE_CODE_MASK].hook = - ospf6_spf_database_hook; - - memset (&hook, 0, sizeof (hook)); - hook.name = "OriginateNetwork"; - hook.hook_change = ospf6_lsa_network_hook_neighbor; - ospf6_hook_register (&hook, &neighbor_hook); - - memset (&hook, 0, sizeof (hook)); - hook.name = "OriginateNetwork"; - hook.hook_change = ospf6_lsa_network_hook_interface; - ospf6_hook_register (&hook, &interface_hook); -} - -/****************************/ -/* RFC2740 3.4.3.6 Link-LSA */ -/****************************/ - -int -ospf6_lsa_link_show (struct vty *vty, struct ospf6_lsa *lsa) -{ - char *start, *end, *current; - struct ospf6_link_lsa *link_lsa; - int prefixnum; - struct ospf6_prefix *prefix; - char buf[128]; - struct in6_addr in6; - - assert (lsa->header); - - link_lsa = (struct ospf6_link_lsa *) (lsa->header + 1); - prefixnum = ntohl (link_lsa->llsa_prefix_num); - - inet_ntop (AF_INET6, (void *)&link_lsa->llsa_linklocal, buf, sizeof (buf)); - vty_out (vty, " LinkLocal Address: %s%s", buf, VTY_NEWLINE); - vty_out (vty, " Number of Prefix: %d%s", prefixnum, VTY_NEWLINE); - - start = (char *) link_lsa + sizeof (struct ospf6_link_lsa); - end = (char *) lsa->header + ntohs (lsa->header->length); - for (current = start; current < end; current += OSPF6_PREFIX_SIZE (prefix)) - { - prefix = (struct ospf6_prefix *) current; - if (current + OSPF6_PREFIX_SIZE (prefix) > end) - { - vty_out (vty, " Trailing %d byte garbage ... Malformed%s", - end - current, VTY_NEWLINE); - return -1; - } - - ospf6_prefix_options_str (prefix->prefix_options, buf, sizeof (buf)); - vty_out (vty, " Prefix Options: %s%s", buf, VTY_NEWLINE); - ospf6_prefix_in6_addr (prefix, &in6); - inet_ntop (AF_INET6, &in6, buf, sizeof (buf)); - vty_out (vty, " Prefix: %s/%d%s", - buf, prefix->prefix_length, VTY_NEWLINE); - } - - return 0; -} - - -void -ospf6_lsa_link_update (char *ifname) -{ - char *cp, buffer [MAXLSASIZE], buf[32]; - u_int16_t size; - struct ospf6_lsa *old; - struct interface *ifp; - struct ospf6_interface *o6i; - - struct ospf6_link_lsa *link_lsa; - struct ospf6_prefix *p; - list prefix_connected; - listnode node; - struct connected *c; - - ifp = if_lookup_by_name (ifname); - if (! ifp) - { - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Update Link: No such Interface: %s", ifname); - return; - } - - o6i = (struct ospf6_interface *) ifp->info; - if (! o6i || ! o6i->area) { - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Update Link: Interface not enabled: %s", ifname); - return; - } - -#if 0 - /* Link-LSA is on Broadcast or NBMA */ - if (! if_is_broadcast (o6i->interface) /* && ! NBMA xxx */) - { - return; - } -#endif /*0*/ - - /* find previous LSA */ - old = ospf6_lsdb_lookup (htons (OSPF6_LSA_TYPE_LINK), htonl (o6i->if_id), - ospf6->router_id, o6i->area); - - /* can't make Link-LSA if linklocal address not set */ - if (! o6i->lladdr) - { - if (IS_OSPF6_DUMP_LSA) - zlog_warn ("Update Link: No Linklocal Address: %s", - o6i->interface->name); - if (old) - ospf6_lsa_premature_aging (old); - return; - } - - if (IS_OSPF6_DUMP_LSA) - zlog_info ("Update Link: Interface %s", o6i->interface->name); - - if (! ospf6_interface_is_enabled (o6i->interface->ifindex)) - { - if (IS_OSPF6_DUMP_LSA) - zlog_info (" Interface %s not enabled", o6i->interface->name); - if (old) - ospf6_lsa_premature_aging (old); - return; - } - - /* check connected prefix */ - prefix_connected = list_new (); - for (node = listhead (o6i->interface->connected); node; nextnode (node)) - { - c = (struct connected *) getdata (node); - - /* filter prefix not IPv6 */ - if (c->address->family != AF_INET6) - continue; - - /* for log */ - prefix2str (c->address, buf, sizeof (buf)); - - CONTINUE_IF_ADDRESS_LINKLOCAL (c->address); - CONTINUE_IF_ADDRESS_UNSPECIFIED (c->address); - CONTINUE_IF_ADDRESS_LOOPBACK (c->address); - CONTINUE_IF_ADDRESS_V4COMPAT (c->address); - CONTINUE_IF_ADDRESS_V4MAPPED (c->address); - - /* filter prefix specified by configuration */ - if (o6i->plist_name) + if (conf_debug_ospf6_lsa == OSPF6_DEBUG_LSA_DEFAULT) + vty_out (vty, "debug ospf6 lsa%s", VTY_NEWLINE); + else { - struct prefix_list *plist; - enum prefix_list_type result = PREFIX_PERMIT; - - plist = prefix_list_lookup (AFI_IP6, o6i->plist_name); - if (plist) - result = prefix_list_apply (plist, c->address); - else if (IS_OSPF6_DUMP_LSA) - zlog_warn ("Update Intra-Prefix (Stub): " - "Prefix list \"%s\" not found", o6i->plist_name); - - if (result == PREFIX_DENY) - { - if (IS_OSPF6_DUMP_LSA) - zlog_info (" Filter out Prefix-list %s: %s", - o6i->plist_name, buf); - continue; - } + if (IS_OSPF6_DEBUG_LSA (SEND)) + vty_out (vty, "debug ospf6 lsa send%s", VTY_NEWLINE); + if (IS_OSPF6_DEBUG_LSA (RECV)) + vty_out (vty, "debug ospf6 lsa recv%s", VTY_NEWLINE); + if (IS_OSPF6_DEBUG_LSA (ORIGINATE)) + vty_out (vty, "debug ospf6 lsa originate%s", VTY_NEWLINE); + if (IS_OSPF6_DEBUG_LSA (TIMER)) + vty_out (vty, "debug ospf6 lsa timer%s", VTY_NEWLINE); } - if (IS_OSPF6_DUMP_LSA) - zlog_info (" Advertise %s", buf); - - /* hold prefix in list. duplicate is filtered in ospf6_prefix_add() */ - p = ospf6_prefix_create (0, 0, (struct prefix_ipv6 *) c->address); - ospf6_prefix_add (prefix_connected, p); - } - - /* Note: even if no prefix configured, still we have to create Link-LSA - for next-hop resolution */ - - memset (buffer, 0, sizeof (buffer)); - size = sizeof (struct ospf6_link_lsa); - link_lsa = (struct ospf6_link_lsa *) buffer; - - /* fill Link LSA and calculate size */ - link_lsa->llsa_rtr_pri = o6i->priority; - link_lsa->llsa_options[0] = o6i->area->options[0]; - link_lsa->llsa_options[1] = o6i->area->options[1]; - link_lsa->llsa_options[2] = o6i->area->options[2]; - - /* linklocal address */ - memcpy (&link_lsa->llsa_linklocal, o6i->lladdr, sizeof (struct in6_addr)); - -#ifdef KAME /* clear ifindex */ - if (link_lsa->llsa_linklocal.s6_addr[3] & 0x0f) - link_lsa->llsa_linklocal.s6_addr[3] &= ~((char)0x0f); -#endif /* KAME */ - - link_lsa->llsa_prefix_num = htonl (listcount (prefix_connected)); - cp = (char *)(link_lsa + 1); - for (node = listhead (prefix_connected); node; nextnode (node)) - { - p = (struct ospf6_prefix *) getdata (node); - size += OSPF6_PREFIX_SIZE (p); - memcpy (cp, p, OSPF6_PREFIX_SIZE (p)); - cp += OSPF6_PREFIX_SIZE (p); + if (IS_OSPF6_DEBUG_LSA (DATABASE)) + vty_out (vty, "debug ospf6 lsa database%s", VTY_NEWLINE); + if (IS_OSPF6_DEBUG_LSA (MEMORY)) + vty_out (vty, "debug ospf6 lsa memory%s", VTY_NEWLINE); } - for (node = listhead (prefix_connected); node; nextnode (node)) - { - p = (struct ospf6_prefix *) getdata (node); - ospf6_prefix_delete (p); - } - list_delete (prefix_connected); - - ospf6_lsa_originate (htons (OSPF6_LSA_TYPE_LINK), - htonl (o6i->if_id), o6i->area->ospf6->router_id, - (char *) link_lsa, size, o6i); -} - -int -ospf6_lsa_link_hook_interface (void *interface) -{ - struct ospf6_interface *o6i = interface; - if (o6i->area) - ospf6_lsa_link_update (o6i->interface->name); - return 0; -} - -int -ospf6_lsa_link_refresh (void *old) -{ - struct ospf6_lsa *lsa = old; - struct interface *ifp; - - ifp = if_lookup_by_index (ntohl (lsa->header->id)); - if (! ifp) - ospf6_lsa_premature_aging (old); - else - ospf6_lsa_link_update (ifp->name); - return 0; } void -ospf6_lsa_slot_register_link () +install_element_ospf6_debug_lsa () { - struct ospf6_lsa_slot slot; - - memset (&slot, 0, sizeof (struct ospf6_lsa_slot)); - slot.type = htons (OSPF6_LSA_TYPE_LINK); - slot.name = "Link"; - slot.func_show = ospf6_lsa_link_show; - slot.func_refresh = ospf6_lsa_link_refresh; - slot.hook_interface.name = "OriginateLink"; - slot.hook_interface.hook_change = ospf6_lsa_link_hook_interface; - ospf6_lsa_slot_register (&slot); - - /* - * Link LSA handling will be shift in ospf6_intra.c - * Currently, only database hook only moved to ospf6_intra.c - */ -#if 0 - ospf6_lsdb_hook[OSPF6_LSA_TYPE_LINK & OSPF6_LSTYPE_CODE_MASK].hook = - ospf6_spf_database_hook; -#endif /*0*/ + install_element (ENABLE_NODE, &debug_ospf6_lsa_cmd); + install_element (ENABLE_NODE, &debug_ospf6_lsa_sendrecv_cmd); + install_element (ENABLE_NODE, &no_debug_ospf6_lsa_cmd); + install_element (ENABLE_NODE, &no_debug_ospf6_lsa_sendrecv_cmd); + install_element (CONFIG_NODE, &debug_ospf6_lsa_cmd); + install_element (CONFIG_NODE, &debug_ospf6_lsa_sendrecv_cmd); + install_element (CONFIG_NODE, &no_debug_ospf6_lsa_cmd); + install_element (CONFIG_NODE, &no_debug_ospf6_lsa_sendrecv_cmd); } -int -ospf6_lsa_add_hook (void *data) -{ - struct ospf6_lsa *lsa = data; - struct ospf6_lsa_slot *sp; - - sp = ospf6_lsa_slot_get (lsa->header->type); - if (sp) - { - CALL_CHANGE_HOOK (&sp->database_hook, lsa); - } - else - zlog_warn ("Unknown LSA added to database: %s", lsa->str); - return 0; -} - -int -ospf6_lsa_change_hook (void *data) -{ - struct ospf6_lsa *lsa = data; - struct ospf6_lsa_slot *sp; - - sp = ospf6_lsa_slot_get (lsa->header->type); - if (sp) - { - CALL_CHANGE_HOOK (&sp->database_hook, lsa); - } - else - zlog_warn ("Unknown LSA changed in database: %s", lsa->str); - return 0; -} - -int -ospf6_lsa_remove_hook (void *data) -{ - struct ospf6_lsa *lsa = data; - struct ospf6_lsa_slot *sp; - - sp = ospf6_lsa_slot_get (lsa->header->type); - if (sp) - { - CALL_REMOVE_HOOK (&sp->database_hook, lsa); - } - else - zlog_warn ("Unknown LSA removed from database: %s", lsa->str); - return 0; -} - -/* Initialize LSA slots */ -void -ospf6_lsa_init () -{ - struct ospf6_hook hook; - - slot_head = NULL; - ospf6_lsa_slot_register_router (); - ospf6_lsa_slot_register_network (); - ospf6_lsa_slot_register_link (); -#if 0 - ospf6_lsa_slot_register_intra_prefix (); - ospf6_lsa_slot_register_as_external (); -#endif /*0*/ - - hook.name = "LSADatabaseHook"; - hook.hook_add = ospf6_lsa_add_hook; - hook.hook_change = ospf6_lsa_change_hook; - hook.hook_remove = ospf6_lsa_remove_hook; - ospf6_hook_register (&hook, &database_hook); -} |