| 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'ssource code and therefore might contain declarations/code from GNUZebra, Copyright (C) 1999 Kunihiro Ishiguro. Zebra is a free routingsoftware, distributed under the GNU General Public License. A copy ofthis 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_STRLCATstatic    size_t strlcat(char *dst, const char *src, size_t size);#endifchar *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_STRLCATsize_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
 |