mirror of
https://github.com/the-tcpdump-group/tcpdump.git
synced 2024-11-23 18:14:29 +08:00
375 lines
9.3 KiB
C
375 lines
9.3 KiB
C
/*
|
|
* Copyright (c) 1991, 1992, 1993, 1994, 1995, 1996
|
|
* The Regents of the University of California. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms are permitted
|
|
* provided that the above copyright notice and this paragraph are
|
|
* duplicated in all such forms and that any documentation,
|
|
* advertising materials, and other materials related to such
|
|
* distribution and use acknowledge that the software was developed
|
|
* by the University of California, Lawrence Berkeley Laboratory,
|
|
* Berkeley, CA. The name of the University may not be used to
|
|
* endorse or promote products derived from this software without
|
|
* specific prior written permission.
|
|
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
|
*
|
|
* Initial contribution from Jeff Honig (jch@MITCHELL.CIT.CORNELL.EDU).
|
|
*/
|
|
|
|
/* \summary: Exterior Gateway Protocol (EGP) printer */
|
|
|
|
/* specification: RFC 827 */
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include "netdissect-stdinc.h"
|
|
|
|
#define ND_LONGJMP_FROM_TCHECK
|
|
#include "netdissect.h"
|
|
#include "addrtoname.h"
|
|
#include "extract.h"
|
|
|
|
struct egp_packet {
|
|
nd_uint8_t egp_version;
|
|
#define EGP_VERSION 2
|
|
nd_uint8_t egp_type;
|
|
#define EGPT_ACQUIRE 3
|
|
#define EGPT_REACH 5
|
|
#define EGPT_POLL 2
|
|
#define EGPT_UPDATE 1
|
|
#define EGPT_ERROR 8
|
|
nd_uint8_t egp_code;
|
|
#define EGPC_REQUEST 0
|
|
#define EGPC_CONFIRM 1
|
|
#define EGPC_REFUSE 2
|
|
#define EGPC_CEASE 3
|
|
#define EGPC_CEASEACK 4
|
|
#define EGPC_HELLO 0
|
|
#define EGPC_HEARDU 1
|
|
nd_uint8_t egp_status;
|
|
#define EGPS_UNSPEC 0
|
|
#define EGPS_ACTIVE 1
|
|
#define EGPS_PASSIVE 2
|
|
#define EGPS_NORES 3
|
|
#define EGPS_ADMIN 4
|
|
#define EGPS_GODOWN 5
|
|
#define EGPS_PARAM 6
|
|
#define EGPS_PROTO 7
|
|
#define EGPS_INDET 0
|
|
#define EGPS_UP 1
|
|
#define EGPS_DOWN 2
|
|
#define EGPS_UNSOL 0x80
|
|
nd_uint16_t egp_checksum;
|
|
nd_uint16_t egp_as;
|
|
nd_uint16_t egp_sequence;
|
|
union {
|
|
nd_uint16_t egpu_hello;
|
|
nd_uint8_t egpu_gws[2];
|
|
nd_uint16_t egpu_reason;
|
|
#define EGPR_UNSPEC 0
|
|
#define EGPR_BADHEAD 1
|
|
#define EGPR_BADDATA 2
|
|
#define EGPR_NOREACH 3
|
|
#define EGPR_XSPOLL 4
|
|
#define EGPR_NORESP 5
|
|
#define EGPR_UVERSION 6
|
|
} egp_handg;
|
|
#define egp_hello egp_handg.egpu_hello
|
|
#define egp_intgw egp_handg.egpu_gws[0]
|
|
#define egp_extgw egp_handg.egpu_gws[1]
|
|
#define egp_reason egp_handg.egpu_reason
|
|
union {
|
|
nd_uint16_t egpu_poll;
|
|
nd_ipv4 egpu_sourcenet;
|
|
} egp_pands;
|
|
#define egp_poll egp_pands.egpu_poll
|
|
#define egp_sourcenet egp_pands.egpu_sourcenet
|
|
};
|
|
|
|
static const struct tok egp_type_str[] = {
|
|
{ EGPT_ACQUIRE, "acquire" },
|
|
{ EGPT_REACH, "reach" },
|
|
{ EGPT_POLL, "poll" },
|
|
{ EGPT_UPDATE, "update" },
|
|
{ EGPT_ERROR, "error" },
|
|
{ 0, NULL }
|
|
};
|
|
|
|
static const struct tok egp_acquire_codes_str[] = {
|
|
{ EGPC_REQUEST, "request" },
|
|
{ EGPC_CONFIRM, "confirm" },
|
|
{ EGPC_REFUSE, "refuse" },
|
|
{ EGPC_CEASE, "cease" },
|
|
{ EGPC_CEASEACK, "cease_ack" },
|
|
{ 0, NULL }
|
|
};
|
|
|
|
static const struct tok egp_acquire_status_str[] = {
|
|
{ EGPS_UNSPEC, "unspecified" },
|
|
{ EGPS_ACTIVE, "active_mode" },
|
|
{ EGPS_PASSIVE, "passive_mode" },
|
|
{ EGPS_NORES, "insufficient_resources" },
|
|
{ EGPS_ADMIN, "administratively_prohibited" },
|
|
{ EGPS_GODOWN, "going_down" },
|
|
{ EGPS_PARAM, "parameter_violation" },
|
|
{ EGPS_PROTO, "protocol_violation" },
|
|
{ 0, NULL }
|
|
};
|
|
|
|
static const struct tok egp_reach_codes_str[] = {
|
|
{ EGPC_HELLO, "hello" },
|
|
{ EGPC_HEARDU, "i-h-u" },
|
|
{ 0, NULL }
|
|
};
|
|
|
|
static const struct tok egp_status_updown_str[] = {
|
|
{ EGPS_INDET, "indeterminate" },
|
|
{ EGPS_UP, "up" },
|
|
{ EGPS_DOWN, "down" },
|
|
{ 0, NULL }
|
|
};
|
|
|
|
static const struct tok egp_reasons_str[] = {
|
|
{ EGPR_UNSPEC, "unspecified" },
|
|
{ EGPR_BADHEAD, "bad_EGP_header_format" },
|
|
{ EGPR_BADDATA, "bad_EGP_data_field_format" },
|
|
{ EGPR_NOREACH, "reachability_info_unavailable" },
|
|
{ EGPR_XSPOLL, "excessive_polling_rate" },
|
|
{ EGPR_NORESP, "no_response" },
|
|
{ EGPR_UVERSION, "unsupported_version" },
|
|
{ 0, NULL }
|
|
};
|
|
|
|
static void
|
|
egpnr_print(netdissect_options *ndo,
|
|
const struct egp_packet *egp, u_int length)
|
|
{
|
|
const uint8_t *cp;
|
|
uint32_t addr;
|
|
uint32_t net;
|
|
u_int netlen;
|
|
u_int gateways, distances, networks;
|
|
u_int intgw, extgw, t_gateways;
|
|
const char *comma;
|
|
|
|
addr = GET_IPV4_TO_NETWORK_ORDER(egp->egp_sourcenet);
|
|
if (IN_CLASSA(addr)) {
|
|
net = addr & IN_CLASSA_NET;
|
|
netlen = 1;
|
|
} else if (IN_CLASSB(addr)) {
|
|
net = addr & IN_CLASSB_NET;
|
|
netlen = 2;
|
|
} else if (IN_CLASSC(addr)) {
|
|
net = addr & IN_CLASSC_NET;
|
|
netlen = 3;
|
|
} else {
|
|
net = 0;
|
|
netlen = 0;
|
|
}
|
|
cp = (const uint8_t *)(egp + 1);
|
|
length -= sizeof(*egp);
|
|
|
|
intgw = GET_U_1(egp->egp_intgw);
|
|
extgw = GET_U_1(egp->egp_extgw);
|
|
t_gateways = intgw + extgw;
|
|
for (gateways = 0; gateways < t_gateways; ++gateways) {
|
|
/* Pickup host part of gateway address */
|
|
addr = 0;
|
|
if (length < 4 - netlen)
|
|
goto invalid;
|
|
ND_TCHECK_LEN(cp, 4 - netlen);
|
|
switch (netlen) {
|
|
|
|
case 1:
|
|
addr = GET_U_1(cp);
|
|
cp++;
|
|
/* fall through */
|
|
case 2:
|
|
addr = (addr << 8) | GET_U_1(cp);
|
|
cp++;
|
|
/* fall through */
|
|
case 3:
|
|
addr = (addr << 8) | GET_U_1(cp);
|
|
cp++;
|
|
break;
|
|
}
|
|
addr |= net;
|
|
length -= 4 - netlen;
|
|
if (length < 1)
|
|
goto invalid;
|
|
distances = GET_U_1(cp);
|
|
cp++;
|
|
length--;
|
|
ND_PRINT(" %s %s ",
|
|
gateways < intgw ? "int" : "ext",
|
|
ipaddr_string(ndo, (const u_char *)&addr)); /* local buffer, not packet data; don't use GET_IPADDR_STRING() */
|
|
|
|
comma = "";
|
|
ND_PRINT("(");
|
|
while (distances != 0) {
|
|
if (length < 2)
|
|
goto invalid;
|
|
ND_PRINT("%sd%u:", comma, GET_U_1(cp));
|
|
cp++;
|
|
comma = ", ";
|
|
networks = GET_U_1(cp);
|
|
cp++;
|
|
length -= 2;
|
|
while (networks != 0) {
|
|
/* Pickup network number */
|
|
if (length < 1)
|
|
goto invalid;
|
|
addr = ((uint32_t) GET_U_1(cp)) << 24;
|
|
cp++;
|
|
length--;
|
|
if (IN_CLASSB(addr)) {
|
|
if (length < 1)
|
|
goto invalid;
|
|
addr |= ((uint32_t) GET_U_1(cp)) << 16;
|
|
cp++;
|
|
length--;
|
|
} else if (!IN_CLASSA(addr)) {
|
|
if (length < 2)
|
|
goto invalid;
|
|
addr |= ((uint32_t) GET_U_1(cp)) << 16;
|
|
cp++;
|
|
addr |= ((uint32_t) GET_U_1(cp)) << 8;
|
|
cp++;
|
|
length -= 2;
|
|
}
|
|
ND_PRINT(" %s", ipaddr_string(ndo, (const u_char *)&addr)); /* local buffer, not packet data; don't use GET_IPADDR_STRING() */
|
|
networks--;
|
|
}
|
|
distances--;
|
|
}
|
|
ND_PRINT(")");
|
|
}
|
|
return;
|
|
invalid:
|
|
nd_print_invalid(ndo);
|
|
}
|
|
|
|
void
|
|
egp_print(netdissect_options *ndo,
|
|
const uint8_t *bp, u_int length)
|
|
{
|
|
const struct egp_packet *egp;
|
|
u_int version;
|
|
u_int type;
|
|
u_int code;
|
|
u_int status;
|
|
|
|
ndo->ndo_protocol = "egp";
|
|
egp = (const struct egp_packet *)bp;
|
|
ND_ICHECKMSG_ZU("packet length", length, <, sizeof(*egp));
|
|
ND_TCHECK_SIZE(egp);
|
|
|
|
version = GET_U_1(egp->egp_version);
|
|
if (!ndo->ndo_vflag) {
|
|
ND_PRINT("EGPv%u, AS %u, seq %u, length %u",
|
|
version,
|
|
GET_BE_U_2(egp->egp_as),
|
|
GET_BE_U_2(egp->egp_sequence),
|
|
length);
|
|
return;
|
|
} else
|
|
ND_PRINT("EGPv%u, length %u",
|
|
version,
|
|
length);
|
|
|
|
if (version != EGP_VERSION) {
|
|
ND_PRINT("[version %u]", version);
|
|
return;
|
|
}
|
|
|
|
type = GET_U_1(egp->egp_type);
|
|
ND_PRINT(" %s", tok2str(egp_type_str, "[type %u]", type));
|
|
code = GET_U_1(egp->egp_code);
|
|
status = GET_U_1(egp->egp_status);
|
|
|
|
switch (type) {
|
|
case EGPT_ACQUIRE:
|
|
ND_PRINT(" %s", tok2str(egp_acquire_codes_str, "[code %u]", code));
|
|
switch (code) {
|
|
case EGPC_REQUEST:
|
|
case EGPC_CONFIRM:
|
|
switch (status) {
|
|
case EGPS_UNSPEC:
|
|
case EGPS_ACTIVE:
|
|
case EGPS_PASSIVE:
|
|
ND_PRINT(" %s", tok2str(egp_acquire_status_str, "%u", status));
|
|
break;
|
|
|
|
default:
|
|
ND_PRINT(" [status %u]", status);
|
|
break;
|
|
}
|
|
ND_PRINT(" hello:%u poll:%u",
|
|
GET_BE_U_2(egp->egp_hello),
|
|
GET_BE_U_2(egp->egp_poll));
|
|
break;
|
|
|
|
case EGPC_REFUSE:
|
|
case EGPC_CEASE:
|
|
case EGPC_CEASEACK:
|
|
switch (status ) {
|
|
case EGPS_UNSPEC:
|
|
case EGPS_NORES:
|
|
case EGPS_ADMIN:
|
|
case EGPS_GODOWN:
|
|
case EGPS_PARAM:
|
|
case EGPS_PROTO:
|
|
ND_PRINT(" %s", tok2str(egp_acquire_status_str, "%u", status));
|
|
break;
|
|
|
|
default:
|
|
ND_PRINT("[status %u]", status);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case EGPT_REACH:
|
|
ND_PRINT(" %s", tok2str(egp_reach_codes_str, "[reach code %u]", code));
|
|
switch (code) {
|
|
case EGPC_HELLO:
|
|
case EGPC_HEARDU:
|
|
ND_PRINT(" state:%s", tok2str(egp_status_updown_str, "%u", status));
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case EGPT_POLL:
|
|
ND_PRINT(" state:%s", tok2str(egp_status_updown_str, "%u", status));
|
|
ND_PRINT(" net:%s", GET_IPADDR_STRING(egp->egp_sourcenet));
|
|
break;
|
|
|
|
case EGPT_UPDATE:
|
|
if (status & EGPS_UNSOL) {
|
|
status &= ~EGPS_UNSOL;
|
|
ND_PRINT(" unsolicited");
|
|
}
|
|
ND_PRINT(" state:%s", tok2str(egp_status_updown_str, "%u", status));
|
|
ND_PRINT(" %s int %u ext %u",
|
|
GET_IPADDR_STRING(egp->egp_sourcenet),
|
|
GET_U_1(egp->egp_intgw),
|
|
GET_U_1(egp->egp_extgw));
|
|
if (ndo->ndo_vflag)
|
|
egpnr_print(ndo, egp, length);
|
|
break;
|
|
|
|
case EGPT_ERROR:
|
|
ND_PRINT(" state:%s", tok2str(egp_status_updown_str, "%u", status));
|
|
ND_PRINT(" %s", tok2str(egp_reasons_str, "[reason %u]", GET_BE_U_2(egp->egp_reason)));
|
|
break;
|
|
}
|
|
return;
|
|
invalid:
|
|
nd_print_invalid(ndo);
|
|
}
|