| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762 | 
							- /*
 
-  Copyright (c) 2007 - 2010 RIPE NCC - All Rights Reserved
 
-  
 
-  Permission to use, copy, modify, and distribute this software and its
 
-  documentation for any purpose and without fee is hereby granted, provided
 
-  that the above copyright notice appear in all copies and that both that
 
-  copyright notice and this permission notice appear in supporting
 
-  documentation, and that the name of the author not be used in advertising or
 
-  publicity pertaining to distribution of the software without specific,
 
-  written prior permission.
 
-  
 
-  THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
 
-  ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS; IN NO EVENT SHALL
 
-  AUTHOR BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY
 
-  DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN
 
-  AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 
-  OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 
-  
 
- Parts of this code have been engineered after analiyzing GNU Zebra's
 
- source code and therefore might contain declarations/code from GNU
 
- Zebra, Copyright (C) 1999 Kunihiro Ishiguro. Zebra is a free routing
 
- software, distributed under the GNU General Public License. A copy of
 
- this license is included with libbgpdump.
 
- */
 
- #include "bgpdump-config.h"
 
- #include "cfile_tools.h"
 
- #include "bgpdump_lib.h"
 
- #include "bgpdump_mstream.h"
 
- #include "util.h"
 
- #include <sys/stat.h>
 
- #include <stdlib.h>
 
- #include <unistd.h>
 
- #include <stdbool.h>
 
- #include <netinet/in.h>
 
- #include <sys/socket.h>
 
- #include <arpa/inet.h>
 
- #include <zlib.h>
 
- #include <assert.h>
 
- void	  bgpdump_free_attr(attributes_t *attr);
 
- static    BGPDUMP_ENTRY* bgpdump_entry_create(BGPDUMP *dump);
 
- static    int process_mrtd_bgp(struct mstream *s, BGPDUMP_ENTRY *entry);
 
- static    int process_mrtd_table_dump(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    int process_mrtd_table_dump_v2(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    int process_mrtd_table_dump_v2_peer_index_table(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    int process_mrtd_table_dump_v2_ipv4_unicast(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    int process_mrtd_table_dump_v2_ipv6_unicast(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    int process_zebra_bgp(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    int process_zebra_bgp_state_change(struct mstream *s,BGPDUMP_ENTRY *entry, u_int8_t asn_len);
 
- static    int process_zebra_bgp_message(struct mstream *s,BGPDUMP_ENTRY *entry, u_int8_t asn_len);
 
- static    int process_zebra_bgp_message_update(struct mstream *s,BGPDUMP_ENTRY *entry, u_int8_t asn_len);
 
- static    int process_zebra_bgp_message_open(struct mstream *s,BGPDUMP_ENTRY *entry, u_int8_t asn_len);
 
- static    int process_zebra_bgp_message_notify(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    int process_zebra_bgp_entry(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    int process_zebra_bgp_snapshot(struct mstream *s,BGPDUMP_ENTRY *entry);
 
- static    attributes_t *process_attributes(struct mstream *s, u_int8_t asn_len, struct zebra_incomplete *incomplete, int is_addp);
 
- static    void process_attr_aspath_string(struct aspath *as);
 
- static    char aspath_delimiter_char (u_char type, u_char which);
 
- static    void process_attr_community_string(struct community *com);
 
- static    void process_attr_lcommunity_string(struct lcommunity *lcom);
 
- static    void process_mp_announce(struct mstream *s, struct mp_info *info, struct zebra_incomplete *incomplete, int is_addp);
 
- static    void process_mp_withdraw(struct mstream *s, struct mp_info *info, struct zebra_incomplete *incomplete, int is_addp);
 
- static    int read_prefix_list(struct mstream *s, u_int16_t af, struct prefix *prefixes, struct zebra_incomplete *incomplete, int is_addp);
 
- static    as_t read_asn(struct mstream *s, u_int8_t len);
 
- static    struct aspath *create_aspath(u_int16_t len, u_int8_t asn_len);
 
- static    void aspath_error(struct aspath *as);
 
- static    int check_new_aspath(struct aspath *aspath);
 
- static    void process_asn32_trans(attributes_t *attr, u_int8_t asn_len);
 
- static    struct aspath *asn32_merge_paths(struct aspath *path, struct aspath *newpath);
 
- static    void asn32_expand_16_to_32(char *dst, char *src, int len);
 
- #ifndef HAVE_STRLCAT
 
- static    size_t strlcat(char *dst, const char *src, size_t size);
 
- #endif
 
- char *bgpdump_version(void) {
 
-     return PACKAGE_VERSION;
 
- }
 
- BGPDUMP *bgpdump_open_dump(const char *filename) {
 
-     CFRFILE *f = cfr_open(filename);
 
-     if(! f) {
 
-         perror("can't open dumpfile");
 
-         return NULL;
 
-     }
 
-     BGPDUMP *this_dump = malloc(sizeof(BGPDUMP));
 
-     if (this_dump == NULL) {
 
-         perror("malloc");
 
-         return NULL;
 
-     }
 
-     strcpy(this_dump->filename, "[STDIN]");
 
-     if(filename && strcmp(filename, "-")) {
 
-         if (strlen(filename) >= BGPDUMP_MAX_FILE_LEN - 1) {
 
-             fprintf (stderr, "File name %s is too long.\n", filename);
 
-             exit(1);
 
-         }
 
- 	strcpy(this_dump->filename, filename);
 
-     }
 
-     this_dump->f = f;
 
-     this_dump->eof=0;
 
-     this_dump->parsed = 0;
 
-     this_dump->parsed_ok = 0;
 
-     // peer index table shared among entries
 
-     this_dump->table_dump_v2_peer_index_table = NULL;
 
-     return this_dump;
 
- }
 
- void bgpdump_close_dump(BGPDUMP *dump) {
 
-     if(dump == NULL) {
 
-         return;
 
-     }
 
-     if(dump->table_dump_v2_peer_index_table){
 
-         free(dump->table_dump_v2_peer_index_table->entries);
 
-     }
 
-     free(dump->table_dump_v2_peer_index_table);
 
- 	cfr_close(dump->f);
 
-     free(dump);
 
- }
 
- BGPDUMP_ENTRY* bgpdump_entry_create(BGPDUMP *dump){
 
-     BGPDUMP_ENTRY *this_entry = malloc(sizeof(BGPDUMP_ENTRY));
 
-     if(this_entry == NULL) {
 
-         err("%s: out of memory", __func__);
 
-         return NULL;
 
-     }
 
-     memset(this_entry, 0, sizeof(BGPDUMP_ENTRY));
 
-     this_entry->dump = dump;
 
-     return this_entry;
 
- }
 
- BGPDUMP_ENTRY*	bgpdump_read_next(BGPDUMP *dump) {
 
-     assert(dump);
 
-     struct mstream s;
 
-     u_char *buffer;
 
-     int ok=0;
 
-     u_int32_t bytes_read, t;
 
-     BGPDUMP_ENTRY *this_entry = bgpdump_entry_create(dump);
 
-     if(this_entry == NULL) {
 
-         err("%s: out of memmory", __func__);
 
-         dump->eof = 1;
 
-         return(NULL);
 
-     }
 
-     bytes_read = cfr_read_n(dump->f, &t, 4);
 
-     bytes_read += cfr_read_n(dump->f, &(this_entry->type), 2);
 
-     bytes_read += cfr_read_n(dump->f, &(this_entry->subtype), 2);
 
-     bytes_read += cfr_read_n(dump->f, &(this_entry->length), 4);
 
-     
 
-     if (bytes_read == 12) {
 
-         /* Intel byte ordering stuff ... */
 
-         this_entry->type = ntohs(this_entry->type);
 
-         this_entry->subtype = ntohs(this_entry->subtype);
 
-         this_entry->time = (time_t) ntohl (t);
 
-         this_entry->length = ntohl(this_entry->length);
 
-         
 
-         /* If Extended Header format, then reading the miscroseconds attribute */
 
-         if (this_entry->type == BGPDUMP_TYPE_ZEBRA_BGP_ET) {
 
-             bytes_read += cfr_read_n(dump->f, &(this_entry->ms), 4);
 
-             if (bytes_read == 16) {
 
-                 this_entry->ms = ntohl(this_entry->ms);
 
-                 /* "The Microsecond Timestamp is included in the computation of
 
-                  * the Length field value." (RFC6396 2011) */
 
-                 this_entry->length -= 4;
 
-                 ok = 1;
 
-             }
 
-         } else {
 
-             this_entry->ms = 0;
 
-             ok = 1;
 
-         }
 
-     }
 
-     if (!ok) {
 
-         if(bytes_read > 0) {
 
-             /* Malformed record */
 
-             err("bgpdump_read_next: incomplete MRT header (%d bytes read, expecting 12 or 16)",
 
-                 bytes_read);
 
-         }
 
-         /* Nothing more to read, quit */
 
-         free(this_entry);
 
-         dump->eof = 1;
 
-         return(NULL);
 
-     }
 
-     dump->parsed++;
 
-     this_entry->attr=NULL;
 
-     if(this_entry->length == 0) {
 
-         err("%s: invalid entry length: 0", __func__);
 
- 	free(this_entry);
 
- 	dump->eof=1;
 
- 	return(NULL);
 
-     }
 
-     if ((buffer = malloc(this_entry->length)) == NULL) {
 
- 	err("%s: out of memory", __func__);
 
- 	free(this_entry);
 
- 	dump->eof=1;
 
- 	return(NULL);
 
-     }
 
-     bytes_read = cfr_read_n(dump->f, buffer, this_entry->length);
 
-     if(bytes_read != this_entry->length) {
 
- 	err("bgpdump_read_next: incomplete dump record (%d bytes read, expecting %d)",
 
- 	       bytes_read, this_entry->length);
 
- 	free(this_entry);
 
- 	free(buffer);
 
- 	dump->eof=1;
 
- 	return(NULL);
 
-     }
 
-     ok=0;
 
-     mstream_init(&s,buffer,this_entry->length);
 
-     switch(this_entry->type) {
 
- 	case BGPDUMP_TYPE_MRTD_BGP:
 
- 		ok = process_mrtd_bgp(&s,this_entry);
 
- 		break;
 
- 	case BGPDUMP_TYPE_MRTD_TABLE_DUMP:
 
- 		ok = process_mrtd_table_dump(&s,this_entry);
 
- 		break;
 
- 	case BGPDUMP_TYPE_ZEBRA_BGP:
 
-     case BGPDUMP_TYPE_ZEBRA_BGP_ET:
 
- 		ok = process_zebra_bgp(&s,this_entry);
 
- 		break;
 
- 	case BGPDUMP_TYPE_TABLE_DUMP_V2:
 
- 		ok = process_mrtd_table_dump_v2(&s,this_entry);
 
- 		break;
 
-     }
 
-     free(buffer);
 
-     if(ok) {
 
- 	dump->parsed_ok++;
 
-     } else {
 
- 	bgpdump_free_mem(this_entry);
 
- 	return NULL;
 
-     }
 
-     return this_entry;
 
- }
 
- static void bgpdump_free_mp_info(struct mp_info *info) {
 
-     u_int16_t afi;
 
-     u_int8_t safi;
 
-     for(afi = 1; afi <= BGPDUMP_MAX_AFI; afi++) {
 
-     for(safi = 1; safi <= BGPDUMP_MAX_SAFI; safi++) {
 
-         if(info->announce[afi][safi]) {
 
-             free(info->announce[afi][safi]);
 
-             info->announce[afi][safi] = NULL;
 
-         }
 
-         if(info->withdraw[afi][safi]) {
 
-             free(info->withdraw[afi][safi]);
 
-             info->withdraw[afi][safi] = NULL;
 
-         }
 
-     }
 
-     }
 
-     free(info);
 
- }
 
- void bgpdump_free_mem(BGPDUMP_ENTRY *entry) {
 
-     if(entry!=NULL) {
 
- 	bgpdump_free_attr(entry->attr);
 
- 	switch(entry->type) {
 
- 	    case BGPDUMP_TYPE_ZEBRA_BGP:
 
- 	    case BGPDUMP_TYPE_ZEBRA_BGP_ET:
 
- 		switch(entry->subtype) {
 
- 		    case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE:
 
- 		    case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4:
 
- 		    case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL:
 
- 		    case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL:
 
- 		    case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_ADDPATH:
 
- 		    case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_ADDPATH:
 
- 		    case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL_ADDPATH:
 
- 		    case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL_ADDPATH:
 
- 			switch(entry->body.zebra_message.type) {
 
- 			    case BGP_MSG_NOTIFY:
 
- 				if(entry->body.zebra_message.notify_data)
 
- 				    free(entry->body.zebra_message.notify_data);
 
- 				break;
 
- 			    case BGP_MSG_OPEN:
 
- 				if(entry->body.zebra_message.opt_data)
 
- 				    free(entry->body.zebra_message.opt_data);
 
- 				break;
 
- 			}
 
- 			break;
 
- 		}
 
- 		break;
 
- 	    case BGPDUMP_TYPE_TABLE_DUMP_V2:
 
-         switch(entry->subtype) {
 
-             case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV4_UNICAST:
 
-             case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV6_UNICAST:
 
-             case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV4_UNICAST_ADDPATH:
 
-             case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV6_UNICAST_ADDPATH:
 
-             {
 
-                 BGPDUMP_TABLE_DUMP_V2_PREFIX *e;
 
-                 e = &entry->body.mrtd_table_dump_v2_prefix;
 
-                 int i;
 
-                 // Conditional as the entries have already been free'd in this case
 
-                 if(!(e->entry_count && entry->dump->table_dump_v2_peer_index_table == NULL)) {
 
-                     for(i = 0; i < e->entry_count; i++){
 
-                         bgpdump_free_attr(e->entries[i].attr);
 
-                     }
 
-                     free(e->entries);
 
-                 }
 
-             }
 
- 		}
 
- 		break;
 
- 	}
 
- 	free(entry);
 
-     }
 
- }
 
- void bgpdump_free_attr(attributes_t *attr){
 
- 	if(attr != NULL) {
 
-     	u_int16_t i;
 
-     	struct aspath *path, *pathstofree[3] = { attr->aspath, attr->old_aspath, attr->new_aspath };
 
- 	    for(i = 0; i < sizeof(pathstofree) / sizeof(pathstofree[0]); i++) {
 
- 	      path = pathstofree[i];
 
- 	      if(path) {
 
- 		if(path->data)
 
- 		  free(path->data);
 
- 		if(path->str)
 
- 		  free(path->str);
 
- 		free(path);
 
- 	      }
 
- 	    }
 
- 	    if(attr->community != NULL) {
 
- 		if(attr->community->val != NULL)
 
- 		    free(attr->community->val);
 
- 		if(attr->community->str != NULL)
 
- 		    free(attr->community->str);
 
- 		free(attr->community);
 
- 	    }
 
-         if(attr->lcommunity != NULL) {
 
-             if(attr->lcommunity->val != NULL)
 
-                 free(attr->lcommunity->val);
 
-             if(attr->lcommunity->str != NULL)
 
-                 free(attr->lcommunity->str);
 
-         free(attr->lcommunity);
 
-         }
 
- 	    if(attr->data != NULL)
 
- 		free(attr->data);
 
- 	    if(attr->mp_info != NULL)
 
- 		bgpdump_free_mp_info(attr->mp_info);
 
- 	    if(attr->cluster != NULL) {
 
- 			free(attr->cluster->list);
 
- 			free(attr->cluster);
 
- 		}
 
- 	    if (attr->unknown_num) {
 
- 		for (i = 0; i < attr->unknown_num; i++)
 
- 		    free(attr->unknown[i].raw);
 
- 		free(attr->unknown);
 
- 	    }
 
- 	    free(attr);
 
- 	}
 
- }
 
- /* Helper function to check if this is an Additional Paths enabled subtype (exported) */
 
- int is_addpath(BGPDUMP_ENTRY *entry) {
 
-     switch(entry->type) {
 
-         case BGPDUMP_TYPE_ZEBRA_BGP:
 
-         case BGPDUMP_TYPE_ZEBRA_BGP_ET:
 
-             switch(entry->subtype) {
 
-                 case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_ADDPATH:
 
-                 case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_ADDPATH:
 
-                 case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL_ADDPATH:
 
-                 case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL_ADDPATH:
 
-                     return 1;
 
-                 default:
 
-                     return 0;
 
-             }
 
-         case BGPDUMP_TYPE_TABLE_DUMP_V2:
 
-             switch(entry->subtype) {
 
-                 case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV4_UNICAST_ADDPATH:
 
-                 case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV4_MULTICAST_ADDPATH:
 
-                 case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV6_UNICAST_ADDPATH:
 
-                 case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV6_MULTICAST_ADDPATH:
 
-                 case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_GENERIC_ADDPATH:
 
-                     return 1;
 
-                 default:
 
-                     return 0;
 
-             }
 
-         default:
 
-             return 0;
 
-     }
 
- }
 
- int process_mrtd_bgp(struct mstream *s, BGPDUMP_ENTRY *entry) {
 
-     switch(entry->subtype) {
 
-     case BGPDUMP_SUBTYPE_MRTD_BGP_UPDATE:
 
-     case BGPDUMP_SUBTYPE_MRTD_BGP_KEEPALIVE:
 
- 	entry->body.mrtd_message.source_as = read_asn(s, ASN16_LEN);
 
- 	entry->body.mrtd_message.source_ip = mstream_get_ipv4(s);
 
- 	entry->body.mrtd_message.destination_as = read_asn(s, ASN16_LEN);
 
- 	entry->body.mrtd_message.destination_ip = mstream_get_ipv4(s);
 
- 	mstream_t withdraw_stream = mstream_copy(s, mstream_getw(s, NULL));
 
- 	entry->body.mrtd_message.withdraw_count = read_prefix_list(&withdraw_stream, AFI_IP,
 
- 								   entry->body.mrtd_message.withdraw,
 
- 								   &entry->body.mrtd_message.incomplete, 0);
 
- 	if((entry->attr = process_attributes(s, ASN16_LEN, &entry->body.mrtd_message.incomplete, 0)) == NULL)
 
- 	    return 0;
 
- 	entry->body.mrtd_message.announce_count = read_prefix_list(s, AFI_IP, 
 
- 								   entry->body.mrtd_message.announce,
 
- 								   &entry->body.mrtd_message.incomplete, 0);
 
- 	break;
 
-     case BGPDUMP_SUBTYPE_MRTD_BGP_STATE_CHANGE:
 
- 	entry->body.mrtd_state_change.destination_as = read_asn(s, ASN16_LEN);
 
- 	entry->body.mrtd_state_change.destination_ip = mstream_get_ipv4(s);
 
- 	entry->body.mrtd_state_change.old_state = mstream_getw(s, NULL);
 
- 	entry->body.mrtd_state_change.new_state = mstream_getw(s, NULL);
 
- 	break;
 
-     }
 
-     return 1;
 
- }
 
- int process_mrtd_table_dump(struct mstream *s,BGPDUMP_ENTRY *entry) {
 
-     int afi = entry->subtype;
 
-     u_int8_t asn_len;
 
-     u_int32_t temp_time = 0;
 
-     mstream_getw(s,&entry->body.mrtd_table_dump.view);
 
-     mstream_getw(s,&entry->body.mrtd_table_dump.sequence);
 
-     switch(afi) {
 
- 	case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP:
 
- 	case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP_32BIT_AS:
 
- 	    entry->body.mrtd_table_dump.prefix.v4_addr = mstream_get_ipv4(s);
 
- 	    break;
 
- #ifdef BGPDUMP_HAVE_IPV6
 
- 	case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP6:
 
- 	case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP6_32BIT_AS:
 
- 	    mstream_get(s, &entry->body.mrtd_table_dump.prefix.v6_addr.s6_addr, 16);
 
- 	    break;
 
- #endif
 
- 	default:
 
- 	    warn("process_mrtd_table_dump: unknown AFI %d",  afi);
 
- 	    mstream_get(s, NULL, mstream_can_read(s));
 
- 	    return 0;
 
-     }
 
-     mstream_getc(s,&entry->body.mrtd_table_dump.mask);
 
-     mstream_getc(s,&entry->body.mrtd_table_dump.status);
 
-     mstream_getl(s,&temp_time);
 
-     (entry->body).mrtd_table_dump.uptime = temp_time;
 
-     switch(afi) {
 
-       case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP:
 
-       case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP_32BIT_AS:
 
- 	entry->body.mrtd_table_dump.peer_ip.v4_addr = mstream_get_ipv4(s);
 
- 	break;
 
- #ifdef BGPDUMP_HAVE_IPV6
 
-       case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP6:
 
-       case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP6_32BIT_AS:
 
- 	mstream_get(s, &entry->body.mrtd_table_dump.peer_ip.v6_addr.s6_addr, 16);
 
- 	break;
 
- #endif
 
-     }
 
-     switch(afi) {
 
-       case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP:
 
-       case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP6:
 
- 	asn_len = ASN16_LEN;
 
- 	break;
 
-       case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP_32BIT_AS:
 
-       case BGPDUMP_SUBTYPE_MRTD_TABLE_DUMP_AFI_IP6_32BIT_AS:
 
- 	asn_len = ASN32_LEN;
 
- 	break;
 
-       default:
 
-             assert(0); // unreachable
 
-     }
 
-     entry->body.mrtd_table_dump.peer_as = read_asn(s, asn_len);
 
-     if((entry->attr = process_attributes(s, asn_len, NULL, 0)) == NULL)
 
-         return 0;
 
-     return 1;
 
- }
 
- int process_mrtd_table_dump_v2(struct mstream *s,BGPDUMP_ENTRY *entry) {
 
- 	switch(entry->subtype){
 
- 	case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_PEER_INDEX_TABLE:
 
- 		return process_mrtd_table_dump_v2_peer_index_table(s, entry);
 
- 	break;
 
- 	case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV4_UNICAST:
 
- 	case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV4_UNICAST_ADDPATH:
 
- 		return process_mrtd_table_dump_v2_ipv4_unicast(s, entry);
 
- 	break;
 
- 	case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV6_UNICAST:
 
- 	case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_IPV6_UNICAST_ADDPATH:
 
- 		return process_mrtd_table_dump_v2_ipv6_unicast(s, entry);
 
- 	break;
 
- 	case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_GENERIC:
 
- 	case BGPDUMP_SUBTYPE_TABLE_DUMP_V2_RIB_GENERIC_ADDPATH:
 
- 		//return process_mrtd_table_dump_v2_generic(s, entry);
 
- 	break;
 
- 	}
 
- 	return 0;
 
- }
 
- int process_mrtd_table_dump_v2_peer_index_table(struct mstream *s,BGPDUMP_ENTRY *entry) {
 
- 	BGPDUMP_TABLE_DUMP_V2_PEER_INDEX_TABLE *t;
 
- 	uint16_t i;
 
- 	uint8_t peertype;
 
- 	uint16_t view_name_len;
 
- 	if(entry->dump->table_dump_v2_peer_index_table) {
 
- 		free(entry->dump->table_dump_v2_peer_index_table->entries);
 
- 	}
 
- 	free(entry->dump->table_dump_v2_peer_index_table);
 
- 	if((entry->dump->table_dump_v2_peer_index_table = malloc(sizeof(BGPDUMP_TABLE_DUMP_V2_PEER_INDEX_TABLE))) == NULL) {
 
- 	    err("process_mrtd_table_dump_v2_peer_index_table: failed to allocate memory for index table");
 
- 	    return 0;
 
- 	}
 
- 	t = entry->dump->table_dump_v2_peer_index_table;
 
- 	t->entries = NULL;
 
- 	t->local_bgp_id = mstream_get_ipv4(s);
 
- 	mstream_getw(s,&view_name_len);
 
- 	strcpy(t->view_name, "");
 
- 	// view_name_len is without trailing \0
 
- 	if(view_name_len+1 > BGPDUMP_TYPE_TABLE_DUMP_V2_MAX_VIEWNAME_LEN) {
 
- 	    warn("process_mrtd_table_dump_v2_peer_index_table: view name length more than maximum length (%d), ignoring view name", BGPDUMP_TYPE_TABLE_DUMP_V2_MAX_VIEWNAME_LEN);
 
- 	} else {
 
- 		mstream_get(s, t->view_name, view_name_len);
 
- 		t->view_name[view_name_len] = 0;
 
- 	}
 
- 	mstream_getw(s,&t->peer_count);
 
- 	t->entries = malloc(sizeof(BGPDUMP_TABLE_DUMP_V2_PEER_INDEX_TABLE_ENTRY) * t->peer_count);
 
- 	if(t->entries == NULL){
 
- 	    err("process_mrtd_table_dump_v2_peer_index_table: failed to allocate memory for index table");
 
- 		return 0;
 
- 	}
 
- 	for(i=0; i < t->peer_count; i++) {
 
-     	mstream_getc(s,&peertype);
 
-                 t->entries[i].afi = AFI_IP;
 
- #ifdef BGPDUMP_HAVE_IPV6
 
- 		if(peertype & BGPDUMP_PEERTYPE_TABLE_DUMP_V2_AFI_IP6)
 
- 			t->entries[i].afi = AFI_IP6;
 
- #endif
 
-                 t->entries[i].peer_bgp_id = mstream_get_ipv4(s);
 
- 		if(t->entries[i].afi == AFI_IP)
 
- 			t->entries[i].peer_ip.v4_addr = mstream_get_ipv4(s);
 
- #ifdef BGPDUMP_HAVE_IPV6
 
- 		else
 
- 			mstream_get(s, &t->entries[i].peer_ip.v6_addr.s6_addr, 16);
 
- #endif
 
- 		if(peertype & BGPDUMP_PEERTYPE_TABLE_DUMP_V2_AS4)
 
- 			t->entries[i].peer_as = read_asn(s, ASN32_LEN);
 
- 		else
 
- 			t->entries[i].peer_as = read_asn(s, ASN16_LEN);
 
- 	}
 
- 	return 0;
 
- }
 
- int process_mrtd_table_dump_v2_ipv4_unicast(struct mstream *s, BGPDUMP_ENTRY *entry){
 
- 	BGPDUMP_TABLE_DUMP_V2_PREFIX *prefixdata;
 
- 	prefixdata = &entry->body.mrtd_table_dump_v2_prefix;
 
-     int addpath = is_addpath(entry);
 
- 	uint16_t i;
 
- 	prefixdata->afi = AFI_IP;
 
- 	prefixdata->safi = SAFI_UNICAST;
 
- 	mstream_getl(s, &prefixdata->seq);
 
- 	mstream_getc(s, &prefixdata->prefix_length);
 
- 	bzero(&prefixdata->prefix.v4_addr.s_addr, 4);
 
- 	mstream_get(s, &prefixdata->prefix.v4_addr.s_addr, (prefixdata->prefix_length+7)/8);
 
- 	mstream_getw(s, &prefixdata->entry_count);
 
- 	prefixdata->entries = malloc(sizeof(BGPDUMP_TABLE_DUMP_V2_ROUTE_ENTRY) * prefixdata->entry_count);
 
- 	if(prefixdata->entries == NULL){
 
- 	    err("process_mrtd_table_dump_v2_ipv4_unicast: failed to allocate memory for entry table");
 
- 		return 0;
 
- 	}
 
- 	if(prefixdata->entry_count && entry->dump->table_dump_v2_peer_index_table == NULL) {
 
- 	    free(prefixdata->entries);
 
- 	    err("%s: missing peer index table", __func__);
 
- 	    return 0;
 
-         }
 
- 	for(i=0; i < prefixdata->entry_count; i++){
 
- 		BGPDUMP_TABLE_DUMP_V2_ROUTE_ENTRY *e;
 
- 		e = &prefixdata->entries[i];
 
- 		mstream_getw(s, &e->peer_index);
 
- 		assert(e->peer_index < entry->dump->table_dump_v2_peer_index_table->peer_count);
 
- 		e->peer = &entry->dump->table_dump_v2_peer_index_table->entries[e->peer_index];
 
- 		mstream_getl(s, &e->originated_time);
 
- 		if (addpath)
 
- 		    mstream_getl(s, &e->path_id);
 
-             
 
- 		if((e->attr = process_attributes(s, 4, NULL, is_addpath(entry))) == NULL)
 
- 		    return 0;
 
- 	}
 
- 	return 1;
 
- }
 
- int process_mrtd_table_dump_v2_ipv6_unicast(struct mstream *s, BGPDUMP_ENTRY *entry){
 
- #ifdef BGPDUMP_HAVE_IPV6
 
- 	BGPDUMP_TABLE_DUMP_V2_PREFIX *prefixdata;
 
- 	prefixdata = &entry->body.mrtd_table_dump_v2_prefix;
 
-     int addpath = is_addpath(entry);
 
- 	uint16_t i;
 
- 	prefixdata->afi = AFI_IP6;
 
- 	prefixdata->safi = SAFI_UNICAST;
 
- 	mstream_getl(s, &prefixdata->seq);
 
- 	mstream_getc(s, &prefixdata->prefix_length);
 
- 	bzero(&prefixdata->prefix.v6_addr.s6_addr, 16);
 
- 	mstream_get(s, &prefixdata->prefix.v6_addr.s6_addr, (prefixdata->prefix_length+7)/8);
 
- 	mstream_getw(s, &prefixdata->entry_count);
 
- 	prefixdata->entries = malloc(sizeof(BGPDUMP_TABLE_DUMP_V2_ROUTE_ENTRY) * prefixdata->entry_count);
 
- 	if(prefixdata->entries == NULL){
 
- 	    err("process_mrtd_table_dump_v2_ipv6_unicast: failed to allocate memory for entry table");
 
- 		return 0;
 
- 	}
 
- 	if(prefixdata->entry_count && entry->dump->table_dump_v2_peer_index_table == NULL) {
 
- 	    free(prefixdata->entries);
 
- 	    err("%s: missing peer index table", __func__);
 
- 	    return 0;
 
-         }
 
- 	for(i=0; i < prefixdata->entry_count; i++){
 
- 		BGPDUMP_TABLE_DUMP_V2_ROUTE_ENTRY *e;
 
- 		e = &prefixdata->entries[i];
 
- 		mstream_getw(s, &e->peer_index);
 
- 		assert(e->peer_index < entry->dump->table_dump_v2_peer_index_table->peer_count);
 
- 		e->peer = &entry->dump->table_dump_v2_peer_index_table->entries[e->peer_index];
 
- 		mstream_getl(s, &e->originated_time);
 
- 		if (addpath)
 
- 		    mstream_getl(s, &e->path_id);
 
- 		if((e->attr = process_attributes(s, 4, NULL, is_addpath(entry))) == NULL)
 
- 		    return 0;
 
- 	}
 
- #endif
 
- 	return 1;
 
- }
 
- int process_zebra_bgp(struct mstream *s,BGPDUMP_ENTRY *entry) {
 
-     switch(entry->subtype) {
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_STATE_CHANGE:
 
- 	    return process_zebra_bgp_state_change(s, entry, ASN16_LEN);
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_STATE_CHANGE_AS4:
 
- 	    return process_zebra_bgp_state_change(s, entry, ASN32_LEN);
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE:
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL:
 
-     case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_ADDPATH:
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL_ADDPATH:
 
- 	    return process_zebra_bgp_message(s, entry, ASN16_LEN);
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4:
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL:
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_ADDPATH:
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL_ADDPATH:
 
- 	    return process_zebra_bgp_message(s, entry, ASN32_LEN);
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_ENTRY:
 
- 	    return process_zebra_bgp_entry(s,entry);
 
- 	case BGPDUMP_SUBTYPE_ZEBRA_BGP_SNAPSHOT:
 
- 	    return process_zebra_bgp_snapshot(s, entry);
 
- 	default:
 
- 	    warn("process_zebra_bgp: unknown subtype %d", entry->subtype);
 
- 	    return 0;
 
-     }
 
- }
 
- int process_zebra_bgp_state_change(struct mstream *s,BGPDUMP_ENTRY *entry, u_int8_t asn_len) {
 
-     entry->body.zebra_state_change.source_as = read_asn(s, asn_len);
 
-     entry->body.zebra_state_change.destination_as = read_asn(s, asn_len);
 
-     /* Work around Zebra dump corruption.
 
-      * N.B. I don't see this in quagga 0.96.4 any more. Is it fixed? */
 
-     if (entry->length == 8) {
 
- 	warn("process_zebra_bgp_state_change: 8-byte state change (zebra bug?)");
 
- 	mstream_getw(s,&entry->body.zebra_state_change.old_state);
 
- 	mstream_getw(s,&entry->body.zebra_state_change.new_state);
 
- 	/* Fill in with dummy values */
 
- 	entry->body.zebra_state_change.interface_index = 0;
 
- 	entry->body.zebra_state_change.address_family = AFI_IP;
 
- 	entry->body.zebra_state_change.source_ip.v4_addr.s_addr = 0;
 
- 	entry->body.zebra_state_change.destination_ip.v4_addr.s_addr = 0;
 
-    	return 1;
 
-     }
 
-     mstream_getw(s,&entry->body.zebra_state_change.interface_index);
 
-     mstream_getw(s,&entry->body.zebra_state_change.address_family);
 
-     switch(entry->body.zebra_state_change.address_family) {
 
- 	case AFI_IP:
 
- 	    // length could be 20 or 24 (asn16 vs asn32)
 
- 	    if(entry->length != 20 && entry->length != 24) {
 
- 		warn("process_zebra_bgp_state_change: bad length %d",
 
- 		       entry->length);
 
- 		return 0;
 
- 	    }
 
- 	    entry->body.zebra_state_change.source_ip.v4_addr = mstream_get_ipv4(s);
 
- 	    entry->body.zebra_state_change.destination_ip.v4_addr = mstream_get_ipv4(s);
 
- 	    break;
 
- #ifdef BGPDUMP_HAVE_IPV6
 
- 	case AFI_IP6:
 
- 	    // length could be 44 or 48 (asn16 vs asn32)
 
- 	    if(entry->length != 44 && entry->length != 48) {
 
- 		warn("process_zebra_bgp_state_change: bad length %d",
 
- 		       entry->length);
 
- 		return 0;
 
- 	    }
 
- 	    mstream_get(s, &entry->body.zebra_state_change.source_ip.v6_addr.s6_addr, 16);
 
- 	    mstream_get(s, &entry->body.zebra_state_change.destination_ip.v6_addr.s6_addr, 16);
 
- 	    break;
 
- #endif
 
- 	default:
 
- 	    warn("process_zebra_bgp_state_change: unknown AFI %d",
 
- 		   entry->body.zebra_state_change.address_family);
 
- 	    return 0;
 
-     }
 
-     mstream_getw(s,&entry->body.zebra_state_change.old_state);
 
-     mstream_getw(s,&entry->body.zebra_state_change.new_state);
 
-     return 1;
 
- }
 
- int process_zebra_bgp_message(struct mstream *s,BGPDUMP_ENTRY *entry, u_int8_t asn_len) {
 
-     u_char marker[16]; /* BGP marker */
 
- 	switch(entry->subtype) {
 
- 		case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL:
 
- 		case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL:
 
- 		case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL_ADDPATH:
 
- 		case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL_ADDPATH:
 
- 			entry->body.zebra_message.destination_as = read_asn(s, asn_len);
 
- 			entry->body.zebra_message.source_as = read_asn(s, asn_len);
 
- 			break;
 
- 		case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE:
 
- 		case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4:
 
- 		case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_ADDPATH:
 
- 		case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_ADDPATH:
 
- 		default:
 
- 			entry->body.zebra_message.source_as = read_asn(s, asn_len);
 
- 			entry->body.zebra_message.destination_as = read_asn(s, asn_len);
 
- 			break;
 
- 	}
 
-     mstream_getw(s,&entry->body.zebra_message.interface_index);
 
-     mstream_getw(s,&entry->body.zebra_message.address_family);
 
-     entry->body.zebra_message.opt_len = 0;
 
-     entry->body.zebra_message.opt_data = NULL;
 
-     entry->body.zebra_message.notify_len = 0;
 
-     entry->body.zebra_message.notify_data = NULL;
 
-     switch(entry->body.zebra_message.address_family) {
 
- 	case AFI_IP:
 
- 		switch(entry->subtype) {
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL_ADDPATH:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL_ADDPATH:
 
- 		    		entry->body.zebra_message.destination_ip.v4_addr = mstream_get_ipv4(s);
 
- 		    		entry->body.zebra_message.source_ip.v4_addr = mstream_get_ipv4(s);
 
- 				break;
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_ADDPATH:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_ADDPATH:
 
- 			default:
 
- 				entry->body.zebra_message.source_ip.v4_addr = mstream_get_ipv4(s);
 
- 				entry->body.zebra_message.destination_ip.v4_addr = mstream_get_ipv4(s);
 
- 				break;
 
- 		}
 
- 	    mstream_get (s, marker, 16);
 
- 	    break;
 
- #ifdef BGPDUMP_HAVE_IPV6
 
- 	case AFI_IP6:
 
- 		switch(entry->subtype) {
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_LOCAL_ADDPATH:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_LOCAL_ADDPATH:
 
- 				mstream_get(s,&entry->body.zebra_message.destination_ip.v6_addr.s6_addr, 16);
 
- 				mstream_get(s,&entry->body.zebra_message.source_ip.v6_addr.s6_addr, 16);
 
- 				break;
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_ADDPATH:
 
- 			case BGPDUMP_SUBTYPE_ZEBRA_BGP_MESSAGE_AS4_ADDPATH:
 
- 			default:
 
- 				mstream_get(s,&entry->body.zebra_message.source_ip.v6_addr.s6_addr, 16);
 
- 				mstream_get(s,&entry->body.zebra_message.destination_ip.v6_addr.s6_addr, 16);
 
- 				break;
 
- 		}
 
- 	    mstream_get (s, marker, 16);
 
- 	    break;
 
- #endif
 
- 	case 0xFFFF:
 
- 	    /* Zebra doesn't dump ifindex or src/dest IPs in OPEN
 
- 	     * messages. Work around it. */
 
- 	    if (entry->body.zebra_message.interface_index == 0xFFFF) {
 
- 		memset(marker, 0xFF, 4);
 
- 		mstream_get (s, marker + 4, 12);
 
- 		entry->body.zebra_message.interface_index = 0;
 
- 		entry->body.zebra_message.address_family = AFI_IP;
 
- 		entry->body.zebra_message.source_ip.v4_addr.s_addr = 0;
 
- 		entry->body.zebra_message.destination_ip.v4_addr.s_addr = 0;
 
- 		break;
 
- 	    }
 
- 	    /* Note fall through! If we don't recognize this type of data corruption, we say
 
- 	     * the address family is unsupported (since FFFF is not a valid address family) */
 
- 	default:
 
- 	    /* unsupported address family */
 
- 	    warn("process_zebra_bgp_message: unsupported AFI %d",
 
- 		   entry->body.zebra_message.address_family);
 
- 	    return 0;
 
-     }
 
-     if(memcmp(marker, "\377\377\377\377\377\377\377\377\377\377\377\377\377\377\377\377", 16) != 0) {
 
- 	/* bad marker... ignore packet */
 
- 	warn(
 
- 	       "bgp_message: bad marker: %02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x",
 
- 	       marker[0],marker[1],marker[2],marker[3],marker[4],marker[5],marker[6],marker[7],
 
- 	       marker[8],marker[9],marker[10],marker[11],marker[12],marker[13],marker[14],marker[15]);
 
- 	return 0;
 
-     }
 
-     mstream_getw(s,&entry->body.zebra_message.size);
 
-     
 
-     int expected = entry->body.zebra_message.size - sizeof(marker) - sizeof(u_int16_t);
 
-     
 
-     mstream_t copy = mstream_copy(s, expected);
 
-     
 
-     entry->body.zebra_message.cut_bytes = expected - mstream_can_read(©);
 
-     switch(mstream_getc (©, &entry->body.zebra_message.type)) {
 
- 	case BGP_MSG_OPEN:
 
- 	    return process_zebra_bgp_message_open(©, entry, asn_len);
 
- 	case BGP_MSG_UPDATE:
 
- 	    return process_zebra_bgp_message_update(©, entry, asn_len);
 
- 	case BGP_MSG_NOTIFY:
 
- 	    return process_zebra_bgp_message_notify(©, entry);
 
- 	case BGP_MSG_KEEPALIVE:
 
- 	    /* Nothing to do */
 
- 	    return 1;
 
- 	case BGP_MSG_ROUTE_REFRESH_01:
 
- 	    /* Not implemented yet */
 
- 	    warn("bgp_message: MSG_ROUTE_REFRESH_01 not implemented yet");
 
- 	    return 0;
 
- 	case BGP_MSG_ROUTE_REFRESH:
 
- 	    /* Not implemented yet */
 
- 	    warn("bgp_message: MSG_ROUTE_REFRESH not implemented yet");
 
- 	    return 0;
 
- 	default:
 
- 	    warn("bgp_message: unknown BGP message type %d",
 
- 		   entry->body.zebra_message.type);
 
- 	    return 0;
 
-     }
 
- }
 
- int process_zebra_bgp_message_notify(struct mstream *s, BGPDUMP_ENTRY *entry) {
 
-     mstream_getc(s, &entry->body.zebra_message.error_code);
 
-     mstream_getc(s, &entry->body.zebra_message.sub_error_code);
 
-     entry->body.zebra_message.notify_len = entry->body.zebra_message.size - 21;
 
-     if(entry->body.zebra_message.notify_len > 0) {
 
- 	if((entry->body.zebra_message.notify_data = malloc(entry->body.zebra_message.notify_len)) == NULL) {
 
-             err("%s: out of memory", __func__);
 
-             return 0;
 
-         }
 
- 	mstream_get(s, entry->body.zebra_message.notify_data, entry->body.zebra_message.notify_len);
 
-     }
 
-     return 1;
 
- }
 
- int process_zebra_bgp_message_open(struct mstream *s, BGPDUMP_ENTRY *entry, u_int8_t asn_len) {
 
-     mstream_getc(s, &entry->body.zebra_message.version);
 
-     // my_as in open is always 16bits, regardless of MRT subtype
 
-     entry->body.zebra_message.my_as = read_asn(s, ASN16_LEN);
 
-     mstream_getw(s, &entry->body.zebra_message.hold_time);
 
-     entry->body.zebra_message.bgp_id = mstream_get_ipv4(s);
 
-     mstream_getc(s, &entry->body.zebra_message.opt_len);
 
-     if(entry->body.zebra_message.opt_len) {
 
- 	if((entry->body.zebra_message.opt_data = malloc(entry->body.zebra_message.opt_len)) == NULL) {
 
- 	    err("%s: out of memory", __func__);
 
- 	    return 0;
 
- 	}
 
- 	mstream_get(s, entry->body.zebra_message.opt_data, entry->body.zebra_message.opt_len);
 
-     }
 
-     return 1;
 
- }
 
- int process_zebra_bgp_message_update(struct mstream *s, BGPDUMP_ENTRY *entry, u_int8_t asn_len) {
 
-     entry->body.zebra_message.incomplete.orig_len = 0;
 
-     mstream_t withdraw_stream = mstream_copy(s, mstream_getw(s, NULL));
 
-     entry->body.zebra_message.withdraw_count = read_prefix_list(&withdraw_stream, AFI_IP,
 
-                          entry->body.zebra_message.withdraw,
 
- 			 &entry->body.zebra_message.incomplete,
 
-              is_addpath(entry));
 
-     if((entry->attr = process_attributes(s, asn_len, &entry->body.zebra_message.incomplete, is_addpath(entry))) == NULL)
 
-         return 0;
 
-     entry->body.zebra_message.announce_count = read_prefix_list(s, AFI_IP, 
 
-                          entry->body.zebra_message.announce,
 
- 			 &entry->body.zebra_message.incomplete,
 
-              is_addpath(entry));
 
-     return 1;
 
- }
 
- int process_zebra_bgp_entry(struct mstream *s, BGPDUMP_ENTRY *entry) {
 
-     warn("process_zebra_bgp_entry: record type not implemented yet");
 
-     return 0;
 
- }
 
- int process_zebra_bgp_snapshot(struct mstream *s, BGPDUMP_ENTRY *entry) {
 
-     warn("process_zebra_bgp_snapshot: record type not implemented yet");
 
-     return 0;
 
- }
 
- static attributes_t *attr_init(struct mstream *s, int len) {
 
-     attributes_t *attr = malloc(sizeof(struct attr));
 
-     
 
-     if(attr == NULL) {
 
-         err("%s: out of memory", __func__);
 
-         return NULL;
 
-     }
 
-     if((attr->data=malloc(len)) == NULL) {
 
-         free(attr);
 
-         err("%s: out of memory", __func__);
 
-         return NULL;
 
-     }
 
-     memcpy(attr->data, &s->start[s->position], len);
 
-     
 
-     attr->len = len;
 
-     attr->flag			= 0;
 
-     attr->origin		= -1;
 
-     attr->nexthop.s_addr	= INADDR_NONE;
 
-     attr->med			= -1;
 
-     attr->local_pref		= -1;
 
-     attr->aggregator_as		= -1;
 
-     attr->aggregator_addr.s_addr = INADDR_NONE;
 
-     attr->weight		= -1;
 
-     attr->originator_id.s_addr	= -1;
 
-     attr->cluster		= NULL;
 
-     attr->aspath			= NULL;
 
-     attr->community		= NULL;
 
-     attr->lcommunity     = NULL;
 
-     attr->transit		= NULL;
 
-     attr->mp_info		= calloc(1, sizeof(struct mp_info));
 
-     if(attr->mp_info == NULL) {
 
-         free(attr->data);
 
-         free(attr);
 
-         err("%s: out of memory", __func__);
 
-         return NULL;
 
-     }
 
-     attr->unknown_num = 0;
 
-     attr->unknown = NULL;
 
-     attr->new_aspath		= NULL;
 
-     attr->old_aspath		= NULL;
 
-     attr->new_aggregator_as	= -1;
 
-     attr->new_aggregator_addr.s_addr = INADDR_NONE;
 
-     
 
-     return attr;
 
- }
 
- static void process_unknown_attr(struct mstream *s, attributes_t *attr, int flag, int type, int len) {
 
-     /* Unknown attribute. Save as is */
 
-     attr->unknown_num++;
 
-     if((attr->unknown = realloc(attr->unknown, attr->unknown_num * sizeof(struct unknown_attr))) == NULL) {
 
-         err("%s: out of memory", __func__);
 
-         exit(1); /* XXX */
 
-     }
 
-     /* Pointer to the unknown attribute we want to fill in */
 
-     struct unknown_attr unknown = {
 
-         .flag = flag,
 
-         .type = type,
 
-         .len = len,
 
-         .raw = malloc(len)
 
-     };
 
-     if(unknown.raw == NULL) {
 
-         err("%s: out of memory", __func__);
 
-         exit(1); /* XXX */
 
-     }
 
-     attr->unknown[attr->unknown_num - 1] = unknown;
 
-     
 
-     mstream_get(s, unknown.raw, len);
 
- }
 
- static void process_one_attr(struct mstream *outer_stream, attributes_t *attr, u_int8_t asn_len, struct zebra_incomplete *incomplete, int is_addp) {
 
-     int flag = mstream_getc(outer_stream, NULL);
 
-     int type = mstream_getc(outer_stream, NULL);
 
-     int len;
 
-     
 
-     if(flag & BGP_ATTR_FLAG_EXTLEN)
 
-         len = mstream_getw(outer_stream,NULL);
 
-     else
 
-         len = mstream_getc(outer_stream,NULL);    
 
-     //info("flag:%-2i type:%-2i length:%i", flag, type, len);
 
-     mstream_t ms = mstream_copy(outer_stream, len), *s = &ms;
 
-     if(mstream_can_read(s) != len) {
 
-         warn("ERROR attribute is truncated: expected=%u remaining=%u\n", len, mstream_can_read(s));
 
-         return;
 
-     }
 
-     
 
-     /* Take note of all attributes, including unknown ones */
 
-     if(type <= sizeof(attr->flag) * 8)
 
-         attr->flag |= ATTR_FLAG_BIT(type);
 
-         
 
-     switch(type) {
 
-         case BGP_ATTR_MP_REACH_NLRI:
 
-             process_mp_announce(s, attr->mp_info, incomplete, is_addp);
 
-             break;
 
-         case BGP_ATTR_MP_UNREACH_NLRI:
 
-             process_mp_withdraw(s, attr->mp_info, incomplete, is_addp);
 
-             break;
 
-         case BGP_ATTR_ORIGIN:
 
-             assert(attr->origin == -1);
 
-             attr->origin = mstream_getc(s, NULL);
 
-             break;
 
-         case BGP_ATTR_AS_PATH:
 
-             assert(! attr->aspath);
 
-             attr->aspath = create_aspath(len, asn_len);
 
-             mstream_get(s, attr->aspath->data, len);
 
-             process_attr_aspath_string(attr->aspath);
 
-             break;
 
-         case BGP_ATTR_NEXT_HOP:
 
-             assert(INADDR_NONE == attr->nexthop.s_addr);
 
-             attr->nexthop = mstream_get_ipv4(s);
 
-             break;
 
-         case BGP_ATTR_MULTI_EXIT_DISC:
 
-             assert(-1 == attr->med);
 
-             mstream_getl(s,&attr->med);
 
-             break;
 
-         case BGP_ATTR_LOCAL_PREF:
 
-             assert(-1 == attr->local_pref);
 
-             mstream_getl(s,&attr->local_pref);
 
-             break;
 
-         case BGP_ATTR_ATOMIC_AGGREGATE:
 
-             break;
 
-         case BGP_ATTR_AGGREGATOR:
 
-             assert(-1 == attr->new_aggregator_as);
 
-             attr->aggregator_as = read_asn(s, asn_len);
 
-             attr->aggregator_addr = mstream_get_ipv4(s);
 
-             break;
 
-         case BGP_ATTR_COMMUNITIES:
 
-             assert(! attr->community);
 
-             if((attr->community		= malloc(sizeof(struct community))) == NULL) {
 
-                 err("%s: out of memory", __func__);
 
-                 exit(1); /* XXX */
 
-             }
 
-             
 
-             attr->community->size	= len / 4;
 
-             if((attr->community->val	= malloc(len)) == NULL) {
 
-                 err("%s: out of memory", __func__);
 
-                 exit(1); /* XXX */
 
-             }
 
-             mstream_get(s,attr->community->val,len);
 
-             attr->community->str	= NULL;
 
-             process_attr_community_string(attr->community);
 
-             break;
 
-          case BGP_ATTR_LARGE_COMMUNITIES:
 
-             assert(! attr->lcommunity);
 
-             if((attr->lcommunity     = malloc(sizeof(struct lcommunity))) == NULL) {
 
-                 err("%s: out of memory", __func__);
 
-                 exit(1); /* XXX */
 
-             }
 
-             
 
-             attr->lcommunity->size   = len / 12;
 
-             if((attr->lcommunity->val    = malloc(len)) == NULL) {
 
-                 err("%s: out of memory", __func__);
 
-                 exit(1); /* XXX */
 
-             }
 
-             mstream_get(s,attr->lcommunity->val,len);
 
-             attr->lcommunity->str    = NULL;
 
-             process_attr_lcommunity_string(attr->lcommunity);
 
-             break;
 
-         case BGP_ATTR_NEW_AS_PATH:
 
-             assert(! attr->new_aspath);
 
-             attr->new_aspath = create_aspath(len, ASN32_LEN);
 
-             mstream_get(s,attr->new_aspath->data, len);
 
-             process_attr_aspath_string(attr->new_aspath);
 
-             /* AS_CONFED_SEQUENCE and AS_CONFED_SET segments invalid in NEW_AS_PATH */
 
-             check_new_aspath(attr->new_aspath);
 
-             break;
 
-         case BGP_ATTR_NEW_AGGREGATOR:
 
-             assert(-1 == attr->new_aggregator_as);
 
-             attr->new_aggregator_as = read_asn(s, ASN32_LEN);
 
-             attr->new_aggregator_addr = mstream_get_ipv4(s);
 
-             break;
 
-         case BGP_ATTR_ORIGINATOR_ID:
 
-             assert(INADDR_NONE == attr->originator_id.s_addr);
 
-             attr->originator_id = mstream_get_ipv4(s);
 
-             break;
 
-         case BGP_ATTR_CLUSTER_LIST:
 
-             assert(! attr->cluster);
 
-             if((attr->cluster= malloc(sizeof(struct cluster_list))) == NULL) {
 
-                 err("%s: out of memory", __func__);
 
-                 exit(1); /* XXX */
 
-             }
 
-             attr->cluster->length	= len/4;
 
-             if((attr->cluster->list = calloc((attr->cluster->length), sizeof(struct in_addr))) == NULL) {
 
-                 err("%s: out of memory", __func__);
 
-                 exit(1); /* XXX */
 
-             }
 
-             
 
-             int i; // cluster index
 
-             for (i = 0; i < attr->cluster->length; i++)
 
-                 attr->cluster->list[i] = mstream_get_ipv4(s);
 
-             break;
 
-         default:
 
-             process_unknown_attr(s, attr, flag, type, len);
 
-     }    
 
- }
 
- attributes_t *process_attributes(struct mstream *s, u_int8_t asn_len, struct zebra_incomplete *incomplete, int is_addp) {
 
-     int	total = mstream_getw(s, NULL);
 
-     
 
-     attributes_t *attr = attr_init(s, total);
 
-     if(attr == NULL)
 
-         return NULL;
 
-     mstream_t copy = mstream_copy(s, total);
 
-     if(mstream_can_read(©) != total)
 
-         warn("entry is truncated: expected=%u remaining=%u", total, mstream_can_read(©));
 
-     
 
-     while(mstream_can_read(©))
 
-         process_one_attr(©, attr, asn_len, incomplete, is_addp);
 
-     
 
-     // Once all attributes have been read, take care of ASN32 transition
 
-     process_asn32_trans(attr, asn_len);
 
-     
 
-     return attr;
 
- }
 
- struct aspath *create_aspath(u_int16_t len, u_int8_t asn_len) {
 
-   struct aspath *aspath = malloc(sizeof(struct aspath));
 
-   if(aspath) {
 
-     aspath->asn_len	= asn_len;
 
-     aspath->length	= len;
 
-     aspath->count	= 0;
 
-     aspath->str		= NULL;
 
-     if(len > 0) {
 
-        if((aspath->data	= malloc(len)) == NULL) {
 
-          err("%s: out of memory", __func__);
 
-          free(aspath);
 
-          return NULL;
 
-        }
 
-     } else
 
-        aspath->data	= NULL;
 
-   } else
 
-       err("%s: out of memory", __func__);
 
-   return aspath;
 
- }
 
- void aspath_error(struct aspath *as) {
 
-   as->count = 0;
 
-   if(as->str) {
 
-     free(as->str);
 
-     as->str = NULL;
 
-   }
 
-   as->str = malloc(strlen(ASPATH_STR_ERROR) + 1);
 
-   if (as->str != NULL)
 
-     strcpy(as->str, ASPATH_STR_ERROR);
 
- }
 
- void process_attr_aspath_string(struct aspath *as) {
 
-     const int MAX_ASPATH_LEN = 8000;  
 
-     if((as->str = malloc(MAX_ASPATH_LEN)) == NULL) {
 
-         err("%s: out of memory", __func__);
 
-         exit(1); /* XXX */
 
-     }
 
-     /* Set default values */
 
-     int space = 0;
 
-     u_char type = AS_SEQUENCE;
 
-     int pos = 0;
 
-     
 
-     /* Set initial pointer. */
 
-     caddr_t pnt = as->data;
 
-     caddr_t end = pnt + as->length;
 
-     struct assegment *segment = NULL;
 
-     
 
-     while (pnt < end) {
 
-       int i;
 
-       /* For fetch value. */
 
-       segment = (struct assegment *) pnt;
 
-       /* Check AS type validity. */
 
-       if ((segment->type != AS_SET) &&
 
- 	  (segment->type != AS_SEQUENCE) &&
 
- 	  (segment->type != AS_CONFED_SET) &&
 
- 	  (segment->type != AS_CONFED_SEQUENCE))
 
- 	{
 
- 	  aspath_error(as);
 
- 	  return;
 
- 	}
 
-       /* Check AS length. */
 
-       if ((pnt + (segment->length * as->asn_len) + AS_HEADER_SIZE) > end)
 
- 	{
 
- 	  aspath_error(as);
 
- 	  return;
 
- 	}
 
-       /* If segment type is changed, print previous type's end
 
-          character. */
 
-       if (type != AS_SEQUENCE)
 
- 	as->str[pos++] = aspath_delimiter_char (type, AS_SEG_END);
 
-       if (space)
 
- 	as->str[pos++] = ' ';
 
-       if (segment->type != AS_SEQUENCE)
 
- 	as->str[pos++] = aspath_delimiter_char (segment->type, AS_SEG_START);
 
-       space = 0;
 
-       /* Increment as->count - NOT ignoring CONFED_SETS/SEQUENCES any more.
 
-          I doubt anybody was relying on this behaviour anyway. */
 
-       switch(segment->type) {
 
- 	case AS_SEQUENCE:
 
- 	case AS_CONFED_SEQUENCE:
 
- 	  as->count += segment->length;
 
- 	break;
 
- 	case AS_SET:
 
- 	case AS_CONFED_SET:
 
- 	  as->count += 1;
 
- 	break;
 
-       }
 
-       for (i = 0; i < segment->length; i++)
 
- 	{
 
- 	  as_t asn;
 
- 	  if (space)
 
- 	    {
 
- 	      if (segment->type == AS_SET
 
- 		  || segment->type == AS_CONFED_SET)
 
- 		as->str[pos++] = ',';
 
- 	      else
 
- 		as->str[pos++] = ' ';
 
- 	    }
 
- 	  else
 
- 	    space = 1;
 
- 	  int asn_pos = i * as->asn_len;
 
-           switch(as->asn_len) {
 
-                 case ASN16_LEN:
 
-                     asn = ntohs (*(u_int16_t *) (segment->data + asn_pos));
 
-                     break;
 
-                 case ASN32_LEN:
 
-                     asn = ntohl (*(u_int32_t *) (segment->data + asn_pos));
 
-                     break;
 
-                 default:
 
-                     assert("invalid asn_len" && false);
 
-           }
 
-           pos += int2str(asn, as->str + pos);
 
-           if(pos > MAX_ASPATH_LEN - 100) {
 
-               strcpy(as->str + pos, "...");
 
-               return;
 
-           };
 
- 	}
 
-       type = segment->type;
 
-       pnt += (segment->length * as->asn_len) + AS_HEADER_SIZE;
 
-     }
 
-   if (segment && segment->type != AS_SEQUENCE)
 
-     as->str[pos++] = aspath_delimiter_char (segment->type, AS_SEG_END);
 
-   as->str[pos] = '\0';
 
- }
 
- char aspath_delimiter_char (u_char type, u_char which) {
 
-   int i;
 
-   struct
 
-   {
 
-     int type;
 
-     char start;
 
-     char end;
 
-   } aspath_delim_char [] =
 
-     {
 
-       { AS_SET,             '{', '}' },
 
-       { AS_SEQUENCE,        ' ', ' ' },
 
-       { AS_CONFED_SET,      '[', ']' },
 
-       { AS_CONFED_SEQUENCE, '(', ')' },
 
-       { 0, '\0', '\0' }
 
-     };
 
-   for (i = 0; aspath_delim_char[i].type != 0; i++)
 
-     {
 
-       if (aspath_delim_char[i].type == type)
 
- 	{
 
- 	  if (which == AS_SEG_START)
 
- 	    return aspath_delim_char[i].start;
 
- 	  else if (which == AS_SEG_END)
 
- 	    return aspath_delim_char[i].end;
 
- 	}
 
-     }
 
-   return ' ';
 
- }
 
- void process_attr_community_string(struct community *com) {
 
-   char buf[BUFSIZ];
 
-   int i;
 
-   u_int32_t comval;
 
-   u_int16_t as;
 
-   u_int16_t val;
 
-   memset (buf, 0, BUFSIZ);
 
-   for (i = 0; i < com->size; i++)
 
-     {
 
-       memcpy (&comval, com_nthval (com, i), sizeof (u_int32_t));
 
-       comval = ntohl (comval);
 
-       switch (comval)
 
- 	{
 
- 	case COMMUNITY_NO_EXPORT:
 
- 	  strlcat (buf, " no-export", BUFSIZ);
 
- 	  break;
 
- 	case COMMUNITY_NO_ADVERTISE:
 
- 	  strlcat (buf, " no-advertise", BUFSIZ);
 
- 	  break;
 
- 	case COMMUNITY_LOCAL_AS:
 
- 	  strlcat (buf, " local-AS", BUFSIZ);
 
- 	  break;
 
- 	default:
 
- 	  as = (comval >> 16) & 0xFFFF;
 
- 	  val = comval & 0xFFFF;
 
- 	  snprintf (buf + strlen (buf), BUFSIZ - strlen (buf),
 
- 		    " %d:%d", as, val);
 
- 	  break;
 
- 	}
 
-     }
 
-     if((com->str = malloc(strlen(buf)+1)) != NULL) {
 
-         strcpy(com->str, buf);
 
-     } else {
 
-         err("%s: out of memory", __func__);
 
-         exit(1); /* XXX */
 
-     }
 
- }
 
- void process_attr_lcommunity_string(struct lcommunity *lcom) {
 
-   char buf[BUFSIZ];
 
-   u_int32_t i;
 
-   u_int32_t global;
 
-   u_int32_t local1;
 
-   u_int32_t local2;
 
-   memset (buf, 0, BUFSIZ);
 
-   for (i = 0; i < lcom->size; i++)
 
-     {
 
-         memcpy (&global, lcom->val + (i * 3), sizeof (u_int32_t));
 
-         memcpy (&local1, lcom->val + (i * 3) + 1, sizeof (u_int32_t));
 
-         memcpy (&local2, lcom->val + (i * 3) + 2, sizeof (u_int32_t));
 
-         global = ntohl (global);
 
-         local1 = ntohl (local1);
 
-         local2 = ntohl (local2);
 
-         snprintf (buf + strlen (buf), BUFSIZ - strlen (buf),
 
-             " %u:%u:%u", global, local1, local2);
 
-     }
 
-     if((lcom->str = malloc(strlen(buf)+1)) != NULL) {
 
-         strcpy(lcom->str, buf);
 
-     } else {
 
-         err("%s: out of memory", __func__);
 
-         exit(1); /* XXX */
 
-     }
 
- }
 
- static struct mp_nlri *get_nexthop(struct mstream *s) {
 
-     struct mp_nlri *nlri = calloc(1, sizeof(struct mp_nlri));
 
-     
 
-     if(nlri == NULL) {
 
-         err("%s: out of memory", __func__);
 
-         return NULL;
 
-     }
 
-     nlri->nexthop_len = mstream_getc(s, NULL);
 
-     
 
-     // sometimes nexthop_len is 0 - not sure what this means (see IS-626)
 
-     // if(mp_nlri->nexthop_len == 0)
 
-     //    return len;
 
-     // the AFI is irrelevant to the nexthop, since RFC5549
 
-     // so this code now just checks for an expected length:
 
-     // 4 = IPv4
 
-     // 16 = IPv6
 
-     // 32 = IPv6 + link-local
 
-     if(nlri->nexthop_len == 4) {
 
-         nlri->nexthop.v4_addr = mstream_get_ipv4(s);
 
-     } else if(nlri->nexthop_len == 16) {
 
-         mstream_get(s, &nlri->nexthop.v6_addr, 16);
 
-     } else if(nlri->nexthop_len == 32) {
 
-         mstream_get(s, &nlri->nexthop.v6_addr, 16);
 
-         mstream_get(s, &nlri->nexthop_local.v6_addr.s6_addr, 16);
 
-     }
 
-     else {
 
-         warn("process_mp_announce: unknown MP nexthop length %d", nlri->nexthop_len);
 
-     }
 
-     return nlri;
 
- }
 
- void process_mp_announce(struct mstream *s, struct mp_info *info, struct zebra_incomplete *incomplete, int is_addp) {
 
-     u_int16_t afi;
 
-     u_int8_t safi;
 
-     // look for MRT abbreviated MP_NLRI packets
 
-     if(s->start[s->position] != 0) {
 
-         assert(info->announce[AFI_IP6][SAFI_UNICAST] == NULL);
 
-         info->announce[AFI_IP6][SAFI_UNICAST] = get_nexthop(s);
 
-         return;
 
-     }
 
-     
 
-     mstream_getw(s, &afi);
 
-     mstream_getc(s, &safi);
 
-         
 
-     if(afi > BGPDUMP_MAX_AFI || safi > BGPDUMP_MAX_SAFI) {
 
-             warn("process_mp_announce: unknown protocol(AFI=%d, SAFI=%d)!", afi, safi);
 
-             return;
 
-     }
 
-     if(info->announce[afi][safi] != NULL) {
 
-             warn("process_mp_announce: two MP_NLRI for the same protocol(%d, %d)!", afi, safi);
 
-             return;
 
-     }
 
-     info->announce[afi][safi] = get_nexthop(s);
 
-     // SNPA is defunct and num_snpa should always be 0
 
-     u_int8_t num_snpa;
 
-     if(mstream_getc(s, &num_snpa))
 
-         warn("process_mp_announce: MP_NLRI contains SNPAs, skipping");
 
-     for(; num_snpa > 0; --num_snpa) {
 
-         mstream_get(s, NULL, mstream_getc(s, NULL));
 
-     }
 
-     info->announce[afi][safi]->prefix_count = read_prefix_list(s, afi, info->announce[afi][safi]->nlri, incomplete, is_addp);
 
- }
 
- void process_mp_withdraw(struct mstream *s, struct mp_info *info, struct zebra_incomplete *incomplete, int is_addp) {
 
- 	u_int16_t afi;
 
- 	u_int8_t safi;
 
- 	struct mp_nlri *mp_nlri;
 
- 	mstream_getw(s, &afi);
 
- 	mstream_getc(s, &safi);
 
- 	/* Do we know about this address family? */
 
- 	if(afi > BGPDUMP_MAX_AFI || safi > BGPDUMP_MAX_SAFI) {
 
- 		warn("process_mp_withdraw: unknown AFI,SAFI %d,%d!", afi, safi);
 
- 		return;
 
- 	}
 
- 	/* If there are 2 NLRI's for the same protocol, fail but don't burn and die */
 
- 	if(info->withdraw[afi][safi] != NULL) {
 
- 		warn("process_mp_withdraw: update contains more than one MP_NLRI with AFI,SAFI %d,%d!", afi, safi);
 
- 		return;
 
- 	}
 
- 	/* Allocate structure */
 
- 	if((mp_nlri = malloc(sizeof(struct mp_nlri))) == NULL) {
 
- 	    err("%s: out of memory", __func__);
 
- 	    exit(1); /* XXX */
 
- 	}
 
- 	memset(mp_nlri, 0, sizeof(struct mp_nlri));
 
- 	info->withdraw[afi][safi] = mp_nlri;
 
- 	mp_nlri->prefix_count = read_prefix_list(s, afi, mp_nlri->nlri, incomplete, is_addp);
 
- }
 
- static int read_prefix_list(struct mstream *s, u_int16_t afi, struct prefix *prefixes, struct zebra_incomplete *incomplete, int is_addp) {
 
-     int count = 0;
 
-     
 
-     while(mstream_can_read(s)) {
 
-         /* If this is an Additional Paths enabled NLRI, then there is a
 
-          * 4 octet identifier preceeding each prefix entry */
 
-         pathid_t path_id = 0;
 
-         if (is_addp)
 
-             path_id = mstream_getl(s, NULL);
 
-         u_int8_t p_len = mstream_getc(s,NULL); // length in bits
 
-         u_int8_t p_bytes = (p_len + 7) / 8;
 
-         
 
-         /* Truncated prefix list? */
 
-         if(mstream_can_read(s) < p_bytes) {
 
-             if(! incomplete)
 
-                 break;
 
-             
 
-             /* Put prefix in incomplete structure */
 
-             incomplete->afi = afi;
 
-             incomplete->orig_len = p_len;
 
-             incomplete->prefix = (struct prefix) {
 
-                 .len = mstream_can_read(s) * 8,
 
-                 .path_id = path_id
 
-             };
 
-             mstream_get(s, &incomplete->prefix.address, p_bytes);
 
-             break;
 
-         }
 
-         
 
-         struct prefix *prefix = prefixes + count;
 
-         
 
-         if(count++ > MAX_PREFIXES)
 
-             continue;
 
-         *prefix = (struct prefix) { .len = p_len, .path_id = path_id };
 
-         mstream_get(s, &prefix->address, p_bytes);
 
-     }
 
-     
 
-     if(count > MAX_PREFIXES) {
 
-         err("too many prefixes (%i > %i)", count, MAX_PREFIXES);
 
-         return MAX_PREFIXES;
 
-     }
 
-     
 
-     return count;
 
- }
 
- static as_t read_asn(struct mstream *s, u_int8_t len) {
 
-   assert(len == ASN32_LEN || len == ASN16_LEN);
 
-   switch(len) {
 
-     case ASN32_LEN:
 
-       return mstream_getl(s, NULL);
 
-     case ASN16_LEN:
 
-       return mstream_getw(s, NULL);
 
-     default:
 
-       /* Not reached. Avoid compiler warning */
 
-       return 0;
 
-   }
 
- }
 
- int check_new_aspath(struct aspath *aspath) {
 
-   struct assegment *segment;
 
-   for(segment = (struct assegment *) aspath->data;
 
-       segment < (struct assegment *) (aspath->data + aspath->length);
 
-       segment = (struct assegment *) ((char *) segment + sizeof(*segment) + segment->length * ASN32_LEN)) {
 
-     if(segment->type == AS_CONFED_SEQUENCE || segment->type == AS_CONFED_SET) {
 
-       warn("check_new_aspath: invalid segment of type AS_CONFED_%s in NEW_AS_PATH",
 
- 	     segment->type == AS_CONFED_SET ? "SET" : "SEQUENCE");
 
-       return 0;
 
-     }
 
-   }
 
-   return 1;
 
- }
 
- void process_asn32_trans(attributes_t *attr, u_int8_t asn_len) {
 
-   if(asn_len == ASN32_LEN) {
 
-     /* These attributes "SHOULD NOT" be used with ASN32. */
 
-     if(attr->flag & ATTR_FLAG_BIT(BGP_ATTR_NEW_AS_PATH))
 
-       warn("process_asn32_trans: ASN32 message contains NEW_AS_PATH attribute");
 
-     if(attr->flag & ATTR_FLAG_BIT(BGP_ATTR_NEW_AGGREGATOR))
 
-       warn("process_asn32_trans: ASN32 message contains NEW_AGGREGATOR attribute");
 
-     /* Don't compute anything, just leave AS_PATH and AGGREGATOR as they are */
 
-     return;
 
-   }
 
-   /* Process NEW_AGGREGATOR */
 
-   if(attr->flag & ATTR_FLAG_BIT(BGP_ATTR_AGGREGATOR) &&
 
-      attr->flag & ATTR_FLAG_BIT(BGP_ATTR_NEW_AGGREGATOR)) {
 
-       /* Both AGGREGATOR and NEW_AGGREGATOR present, merge */
 
-       if(attr->aggregator_as != AS_TRAN) {
 
- 	/* Don't do anything */
 
- 	return;
 
-       } else {
 
- 	attr->old_aggregator_as = attr->aggregator_as;
 
- 	attr->old_aggregator_addr = attr->aggregator_addr;
 
- 	attr->aggregator_as = attr->new_aggregator_as;
 
- 	attr->aggregator_addr = attr->new_aggregator_addr;
 
-       }
 
-   }
 
-   /* Process NEW_AS_PATH */
 
-   if(! (attr->flag & ATTR_FLAG_BIT(BGP_ATTR_NEW_AS_PATH)))
 
-     return;
 
-   // attr->aspath may be NULL, at least in case of MP_UNREACH_NLRI
 
-   if(attr->aspath == NULL) return;
 
-   if(attr->aspath->count < attr->new_aspath->count) {
 
-     return;
 
-   }
 
-   /* Merge paths */
 
-   attr->old_aspath = attr->aspath;
 
-   attr->aspath = asn32_merge_paths(attr->old_aspath, attr->new_aspath);
 
-   if(attr->aspath) {
 
-     process_attr_aspath_string(attr->aspath);
 
-   }
 
- }
 
- struct aspath *asn32_merge_paths(struct aspath *path, struct aspath *newpath) {
 
-   void *tmp;
 
-   struct aspath *mergedpath = create_aspath(0, ASN32_LEN);
 
-   if(mergedpath == NULL)
 
-       return NULL;
 
-   struct assegment *segment, *mergedsegment;
 
-   int newlen;
 
-   /* Keep copying segments from AS_PATH until our path is as long as AS_PATH - NEW_AS_PATH. */
 
-   segment = (struct assegment *) (path->data);
 
-   while(mergedpath->count < path->count - newpath->count) {
 
-     /* Make room */
 
-     newlen = mergedpath->length + sizeof(struct assegment) + segment->length * ASN32_LEN;
 
-     tmp = realloc(mergedpath->data, newlen);
 
-     if(tmp == NULL) {
 
-         free(mergedpath->data);
 
-         free(mergedpath);
 
-         err("%s: out of memory", __func__);
 
-         return NULL;
 
-     }
 
-     mergedpath->data = tmp;
 
-     /* Create a new AS-path segment */
 
-     mergedsegment = (struct assegment *) (mergedpath->data + mergedpath->length);
 
-     /* Copy segment over. AS_PATH contains 16-bit ASes, so expand */
 
-     mergedsegment->type = segment->type;
 
-     mergedsegment->length = segment->length;
 
-     asn32_expand_16_to_32(mergedsegment->data, segment->data, segment->length);
 
-     /* Update length */
 
-     mergedpath->length = newlen;
 
-     if(segment->type == AS_SET || segment->type == AS_CONFED_SET) {
 
-       mergedpath->count += 1;
 
-     } else {
 
-       mergedpath->count += segment->length;
 
-       /* Did we copy too many ASes over? */
 
-       if(mergedpath->count > path->count - newpath->count) {
 
- 	mergedsegment->length -= mergedpath->count - (path->count - newpath->count);
 
- 	mergedpath->length -= (mergedpath->count - (path->count - newpath->count)) * ASN32_LEN;
 
-       }
 
-     }
 
-   }
 
-   /* Append NEW_AS_PATH to merged path */
 
-   tmp = realloc(mergedpath->data, mergedpath->length + newpath->length);
 
-   if(tmp == NULL) {
 
-       free(mergedpath->data);
 
-       free(mergedpath);
 
-       err("%s: out of memory", __func__);
 
-       return NULL;
 
-   }
 
-   mergedpath->data = tmp;
 
-   memcpy(mergedpath->data + mergedpath->length, newpath->data, newpath->length);
 
-   mergedpath->length += newpath->length;
 
-   return mergedpath;
 
- }
 
- void asn32_expand_16_to_32(char *dst, char *src, int len) {
 
-   u_int32_t *dstarray = (u_int32_t *) dst;
 
-   u_int16_t *srcarray = (u_int16_t *) src;
 
-   int i;
 
-   for(i = 0; i < len; i++) {
 
-     dstarray[i] = htonl(ntohs(srcarray[i]));
 
-   }
 
- }
 
- #ifndef HAVE_STRLCAT
 
- size_t strlcat(char *dst, const char *src, size_t size) {
 
-   if (strlen (dst) + strlen (src) >= size)
 
-     return -1;
 
-   strcat (dst, src);
 
-   return (strlen(dst));
 
- }
 
- #endif
 
 
  |