123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555 |
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif
- #include <netdissect-stdinc.h>
- #include "netdissect.h"
- #include "extract.h"
- #include "addrtoname.h"
- #include "atm.h"
- #include "llc.h"
- #define VCI_PPC 0x05
- #define VCI_BCC 0x02
- #define VCI_OAMF4SC 0x03
- #define VCI_OAMF4EC 0x04
- #define VCI_METAC 0x01
- #define VCI_ILMIC 0x10
- #define CALL_PROCEED 0x02
- #define CONNECT 0x07
- #define CONNECT_ACK 0x0f
- #define SETUP 0x05
- #define RELEASE 0x4d
- #define RELEASE_DONE 0x5a
- #define RESTART 0x46
- #define RESTART_ACK 0x4e
- #define STATUS 0x7d
- #define STATUS_ENQ 0x75
- #define ADD_PARTY 0x80
- #define ADD_PARTY_ACK 0x81
- #define ADD_PARTY_REJ 0x82
- #define DROP_PARTY 0x83
- #define DROP_PARTY_ACK 0x84
- #define CAUSE 0x08
- #define ENDPT_REF 0x54
- #define AAL_PARA 0x58
- #define TRAFF_DESCRIP 0x59
- #define CONNECT_ID 0x5a
- #define QOS_PARA 0x5c
- #define B_HIGHER 0x5d
- #define B_BEARER 0x5e
- #define B_LOWER 0x5f
- #define CALLING_PARTY 0x6c
- #define CALLED_PARTY 0x70
- #define Q2931 0x09
- #define PROTO_POS 0
- #define CALL_REF_POS 2
- #define MSG_TYPE_POS 5
- #define MSG_LEN_POS 7
- #define IE_BEGIN_POS 9
- #define TYPE_POS 0
- #define LEN_POS 2
- #define FIELD_BEGIN_POS 4
- static const char tstr[] = "[|atm]";
- #define OAM_CRC10_MASK 0x3ff
- #define OAM_PAYLOAD_LEN 48
- #define OAM_FUNCTION_SPECIFIC_LEN 45
- #define OAM_CELLTYPE_FUNCTYPE_LEN 1
- static const struct tok oam_f_values[] = {
- { VCI_OAMF4SC, "OAM F4 (segment)" },
- { VCI_OAMF4EC, "OAM F4 (end)" },
- { 0, NULL }
- };
- static const struct tok atm_pty_values[] = {
- { 0x0, "user data, uncongested, SDU 0" },
- { 0x1, "user data, uncongested, SDU 1" },
- { 0x2, "user data, congested, SDU 0" },
- { 0x3, "user data, congested, SDU 1" },
- { 0x4, "VCC OAM F5 flow segment" },
- { 0x5, "VCC OAM F5 flow end-to-end" },
- { 0x6, "Traffic Control and resource Mgmt" },
- { 0, NULL }
- };
- #define OAM_CELLTYPE_FM 0x1
- #define OAM_CELLTYPE_PM 0x2
- #define OAM_CELLTYPE_AD 0x8
- #define OAM_CELLTYPE_SM 0xf
- static const struct tok oam_celltype_values[] = {
- { OAM_CELLTYPE_FM, "Fault Management" },
- { OAM_CELLTYPE_PM, "Performance Management" },
- { OAM_CELLTYPE_AD, "activate/deactivate" },
- { OAM_CELLTYPE_SM, "System Management" },
- { 0, NULL }
- };
- #define OAM_FM_FUNCTYPE_AIS 0x0
- #define OAM_FM_FUNCTYPE_RDI 0x1
- #define OAM_FM_FUNCTYPE_CONTCHECK 0x4
- #define OAM_FM_FUNCTYPE_LOOPBACK 0x8
- static const struct tok oam_fm_functype_values[] = {
- { OAM_FM_FUNCTYPE_AIS, "AIS" },
- { OAM_FM_FUNCTYPE_RDI, "RDI" },
- { OAM_FM_FUNCTYPE_CONTCHECK, "Continuity Check" },
- { OAM_FM_FUNCTYPE_LOOPBACK, "Loopback" },
- { 0, NULL }
- };
- static const struct tok oam_pm_functype_values[] = {
- { 0x0, "Forward Monitoring" },
- { 0x1, "Backward Reporting" },
- { 0x2, "Monitoring and Reporting" },
- { 0, NULL }
- };
- static const struct tok oam_ad_functype_values[] = {
- { 0x0, "Performance Monitoring" },
- { 0x1, "Continuity Check" },
- { 0, NULL }
- };
- #define OAM_FM_LOOPBACK_INDICATOR_MASK 0x1
- static const struct tok oam_fm_loopback_indicator_values[] = {
- { 0x0, "Reply" },
- { 0x1, "Request" },
- { 0, NULL }
- };
- static const struct tok *oam_functype_values[16] = {
- NULL,
- oam_fm_functype_values,
- oam_pm_functype_values,
- NULL,
- NULL,
- NULL,
- NULL,
- NULL,
- oam_ad_functype_values,
- NULL,
- NULL,
- NULL,
- NULL,
- NULL,
- NULL,
- NULL
- };
- static u_int
- atm_llc_print(netdissect_options *ndo,
- const u_char *p, int length, int caplen)
- {
- int llc_hdrlen;
- llc_hdrlen = llc_print(ndo, p, length, caplen, NULL, NULL);
- if (llc_hdrlen < 0) {
-
- if (!ndo->ndo_suppress_default_print)
- ND_DEFAULTPRINT(p, caplen);
- llc_hdrlen = -llc_hdrlen;
- }
- return (llc_hdrlen);
- }
- #define LLC_UI_HDR(sap) ((sap)<<16 | (sap<<8) | 0x03)
- u_int
- atm_if_print(netdissect_options *ndo,
- const struct pcap_pkthdr *h, const u_char *p)
- {
- u_int caplen = h->caplen;
- u_int length = h->len;
- uint32_t llchdr;
- u_int hdrlen = 0;
- if (caplen < 1 || length < 1) {
- ND_PRINT((ndo, "%s", tstr));
- return (caplen);
- }
-
- if (*p == LLC_UI) {
- if (ndo->ndo_eflag)
- ND_PRINT((ndo, "CNLPID "));
- isoclns_print(ndo, p + 1, length - 1);
- return hdrlen;
- }
-
- if (caplen < 3 || length < 3) {
- ND_PRINT((ndo, "%s", tstr));
- return (caplen);
- }
-
- llchdr = EXTRACT_24BITS(p);
- if (llchdr != LLC_UI_HDR(LLCSAP_SNAP) &&
- llchdr != LLC_UI_HDR(LLCSAP_ISONS) &&
- llchdr != LLC_UI_HDR(LLCSAP_IP)) {
-
- if (caplen < 20 || length < 20) {
- ND_PRINT((ndo, "%s", tstr));
- return (caplen);
- }
- if (ndo->ndo_eflag)
- ND_PRINT((ndo, "%08x%08x %08x%08x ",
- EXTRACT_32BITS(p),
- EXTRACT_32BITS(p+4),
- EXTRACT_32BITS(p+8),
- EXTRACT_32BITS(p+12)));
- p += 20;
- length -= 20;
- caplen -= 20;
- hdrlen += 20;
- }
- hdrlen += atm_llc_print(ndo, p, length, caplen);
- return (hdrlen);
- }
- static const struct tok msgtype2str[] = {
- { CALL_PROCEED, "Call_proceeding" },
- { CONNECT, "Connect" },
- { CONNECT_ACK, "Connect_ack" },
- { SETUP, "Setup" },
- { RELEASE, "Release" },
- { RELEASE_DONE, "Release_complete" },
- { RESTART, "Restart" },
- { RESTART_ACK, "Restart_ack" },
- { STATUS, "Status" },
- { STATUS_ENQ, "Status_enquiry" },
- { ADD_PARTY, "Add_party" },
- { ADD_PARTY_ACK, "Add_party_ack" },
- { ADD_PARTY_REJ, "Add_party_reject" },
- { DROP_PARTY, "Drop_party" },
- { DROP_PARTY_ACK, "Drop_party_ack" },
- { 0, NULL }
- };
- static void
- sig_print(netdissect_options *ndo,
- const u_char *p)
- {
- uint32_t call_ref;
- ND_TCHECK(p[PROTO_POS]);
- if (p[PROTO_POS] == Q2931) {
-
- ND_PRINT((ndo, "Q.2931"));
- ND_TCHECK(p[MSG_TYPE_POS]);
- ND_PRINT((ndo, ":%s ",
- tok2str(msgtype2str, "msgtype#%d", p[MSG_TYPE_POS])));
-
- call_ref = EXTRACT_24BITS(&p[CALL_REF_POS]);
- ND_PRINT((ndo, "CALL_REF:0x%06x", call_ref));
- } else {
-
- ND_PRINT((ndo, "SSCOP, proto %d ", p[PROTO_POS]));
- }
- return;
- trunc:
- ND_PRINT((ndo, " %s", tstr));
- }
- void
- atm_print(netdissect_options *ndo,
- u_int vpi, u_int vci, u_int traftype, const u_char *p, u_int length,
- u_int caplen)
- {
- if (ndo->ndo_eflag)
- ND_PRINT((ndo, "VPI:%u VCI:%u ", vpi, vci));
- if (vpi == 0) {
- switch (vci) {
- case VCI_PPC:
- sig_print(ndo, p);
- return;
- case VCI_BCC:
- ND_PRINT((ndo, "broadcast sig: "));
- return;
- case VCI_OAMF4SC:
- case VCI_OAMF4EC:
- oam_print(ndo, p, length, ATM_OAM_HEC);
- return;
- case VCI_METAC:
- ND_PRINT((ndo, "meta: "));
- return;
- case VCI_ILMIC:
- ND_PRINT((ndo, "ilmi: "));
- snmp_print(ndo, p, length);
- return;
- }
- }
- switch (traftype) {
- case ATM_LLC:
- default:
-
- atm_llc_print(ndo, p, length, caplen);
- break;
- case ATM_LANE:
- lane_print(ndo, p, length, caplen);
- break;
- }
- }
- struct oam_fm_loopback_t {
- uint8_t loopback_indicator;
- uint8_t correlation_tag[4];
- uint8_t loopback_id[12];
- uint8_t source_id[12];
- uint8_t unused[16];
- };
- struct oam_fm_ais_rdi_t {
- uint8_t failure_type;
- uint8_t failure_location[16];
- uint8_t unused[28];
- };
- void
- oam_print (netdissect_options *ndo,
- const u_char *p, u_int length, u_int hec)
- {
- uint32_t cell_header;
- uint16_t vpi, vci, cksum, cksum_shouldbe, idx;
- uint8_t cell_type, func_type, payload, clp;
- union {
- const struct oam_fm_loopback_t *oam_fm_loopback;
- const struct oam_fm_ais_rdi_t *oam_fm_ais_rdi;
- } oam_ptr;
- ND_TCHECK(*(p+ATM_HDR_LEN_NOHEC+hec));
- cell_header = EXTRACT_32BITS(p+hec);
- cell_type = ((*(p+ATM_HDR_LEN_NOHEC+hec))>>4) & 0x0f;
- func_type = (*(p+ATM_HDR_LEN_NOHEC+hec)) & 0x0f;
- vpi = (cell_header>>20)&0xff;
- vci = (cell_header>>4)&0xffff;
- payload = (cell_header>>1)&0x7;
- clp = cell_header&0x1;
- ND_PRINT((ndo, "%s, vpi %u, vci %u, payload [ %s ], clp %u, length %u",
- tok2str(oam_f_values, "OAM F5", vci),
- vpi, vci,
- tok2str(atm_pty_values, "Unknown", payload),
- clp, length));
- if (!ndo->ndo_vflag) {
- return;
- }
- ND_PRINT((ndo, "\n\tcell-type %s (%u)",
- tok2str(oam_celltype_values, "unknown", cell_type),
- cell_type));
- if (oam_functype_values[cell_type] == NULL)
- ND_PRINT((ndo, ", func-type unknown (%u)", func_type));
- else
- ND_PRINT((ndo, ", func-type %s (%u)",
- tok2str(oam_functype_values[cell_type],"none",func_type),
- func_type));
- p += ATM_HDR_LEN_NOHEC + hec;
- switch (cell_type << 4 | func_type) {
- case (OAM_CELLTYPE_FM << 4 | OAM_FM_FUNCTYPE_LOOPBACK):
- oam_ptr.oam_fm_loopback = (const struct oam_fm_loopback_t *)(p + OAM_CELLTYPE_FUNCTYPE_LEN);
- ND_TCHECK(*oam_ptr.oam_fm_loopback);
- ND_PRINT((ndo, "\n\tLoopback-Indicator %s, Correlation-Tag 0x%08x",
- tok2str(oam_fm_loopback_indicator_values,
- "Unknown",
- oam_ptr.oam_fm_loopback->loopback_indicator & OAM_FM_LOOPBACK_INDICATOR_MASK),
- EXTRACT_32BITS(&oam_ptr.oam_fm_loopback->correlation_tag)));
- ND_PRINT((ndo, "\n\tLocation-ID "));
- for (idx = 0; idx < sizeof(oam_ptr.oam_fm_loopback->loopback_id); idx++) {
- if (idx % 2) {
- ND_PRINT((ndo, "%04x ", EXTRACT_16BITS(&oam_ptr.oam_fm_loopback->loopback_id[idx])));
- }
- }
- ND_PRINT((ndo, "\n\tSource-ID "));
- for (idx = 0; idx < sizeof(oam_ptr.oam_fm_loopback->source_id); idx++) {
- if (idx % 2) {
- ND_PRINT((ndo, "%04x ", EXTRACT_16BITS(&oam_ptr.oam_fm_loopback->source_id[idx])));
- }
- }
- break;
- case (OAM_CELLTYPE_FM << 4 | OAM_FM_FUNCTYPE_AIS):
- case (OAM_CELLTYPE_FM << 4 | OAM_FM_FUNCTYPE_RDI):
- oam_ptr.oam_fm_ais_rdi = (const struct oam_fm_ais_rdi_t *)(p + OAM_CELLTYPE_FUNCTYPE_LEN);
- ND_TCHECK(*oam_ptr.oam_fm_ais_rdi);
- ND_PRINT((ndo, "\n\tFailure-type 0x%02x", oam_ptr.oam_fm_ais_rdi->failure_type));
- ND_PRINT((ndo, "\n\tLocation-ID "));
- for (idx = 0; idx < sizeof(oam_ptr.oam_fm_ais_rdi->failure_location); idx++) {
- if (idx % 2) {
- ND_PRINT((ndo, "%04x ", EXTRACT_16BITS(&oam_ptr.oam_fm_ais_rdi->failure_location[idx])));
- }
- }
- break;
- case (OAM_CELLTYPE_FM << 4 | OAM_FM_FUNCTYPE_CONTCHECK):
-
- break;
- default:
- break;
- }
-
- ND_TCHECK2(*(p + OAM_CELLTYPE_FUNCTYPE_LEN + OAM_FUNCTION_SPECIFIC_LEN), 2);
- cksum = EXTRACT_16BITS(p + OAM_CELLTYPE_FUNCTYPE_LEN + OAM_FUNCTION_SPECIFIC_LEN)
- & OAM_CRC10_MASK;
- cksum_shouldbe = verify_crc10_cksum(0, p, OAM_PAYLOAD_LEN);
- ND_PRINT((ndo, "\n\tcksum 0x%03x (%scorrect)",
- cksum,
- cksum_shouldbe == 0 ? "" : "in"));
- return;
- trunc:
- ND_PRINT((ndo, "[|oam]"));
- return;
- }
|