123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250 |
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif
- #include <netdissect-stdinc.h>
- #include <string.h>
- #include "netdissect.h"
- #include "extract.h"
- #include "addrtoname.h"
- #include "ether.h"
- #define TOKEN_HDRLEN 14
- #define TOKEN_RING_MAC_LEN 6
- #define ROUTING_SEGMENT_MAX 16
- #define IS_SOURCE_ROUTED(trp) ((trp)->token_shost[0] & 0x80)
- #define FRAME_TYPE(trp) (((trp)->token_fc & 0xC0) >> 6)
- #define TOKEN_FC_LLC 1
- #define BROADCAST(trp) ((EXTRACT_16BITS(&(trp)->token_rcf) & 0xE000) >> 13)
- #define RIF_LENGTH(trp) ((EXTRACT_16BITS(&(trp)->token_rcf) & 0x1f00) >> 8)
- #define DIRECTION(trp) ((EXTRACT_16BITS(&(trp)->token_rcf) & 0x0080) >> 7)
- #define LARGEST_FRAME(trp) ((EXTRACT_16BITS(&(trp)->token_rcf) & 0x0070) >> 4)
- #define RING_NUMBER(trp, x) ((EXTRACT_16BITS(&(trp)->token_rseg[x]) & 0xfff0) >> 4)
- #define BRIDGE_NUMBER(trp, x) ((EXTRACT_16BITS(&(trp)->token_rseg[x]) & 0x000f))
- #define SEGMENT_COUNT(trp) ((int)((RIF_LENGTH(trp) - 2) / 2))
- struct token_header {
- uint8_t token_ac;
- uint8_t token_fc;
- uint8_t token_dhost[TOKEN_RING_MAC_LEN];
- uint8_t token_shost[TOKEN_RING_MAC_LEN];
- uint16_t token_rcf;
- uint16_t token_rseg[ROUTING_SEGMENT_MAX];
- };
- static const char tstr[] = "[|token-ring]";
- static inline void
- extract_token_addrs(const struct token_header *trp, char *fsrc, char *fdst)
- {
- memcpy(fdst, (const char *)trp->token_dhost, 6);
- memcpy(fsrc, (const char *)trp->token_shost, 6);
- }
- static inline void
- token_hdr_print(netdissect_options *ndo,
- register const struct token_header *trp, register u_int length,
- register const u_char *fsrc, register const u_char *fdst)
- {
- const char *srcname, *dstname;
- srcname = etheraddr_string(ndo, fsrc);
- dstname = etheraddr_string(ndo, fdst);
- if (!ndo->ndo_qflag)
- ND_PRINT((ndo, "%02x %02x ",
- trp->token_ac,
- trp->token_fc));
- ND_PRINT((ndo, "%s > %s, length %u: ",
- srcname, dstname,
- length));
- }
- static const char *broadcast_indicator[] = {
- "Non-Broadcast", "Non-Broadcast",
- "Non-Broadcast", "Non-Broadcast",
- "All-routes", "All-routes",
- "Single-route", "Single-route"
- };
- static const char *direction[] = {
- "Forward", "Backward"
- };
- static const char *largest_frame[] = {
- "516",
- "1500",
- "2052",
- "4472",
- "8144",
- "11407",
- "17800",
- "??"
- };
- u_int
- token_print(netdissect_options *ndo, const u_char *p, u_int length, u_int caplen)
- {
- const struct token_header *trp;
- int llc_hdrlen;
- struct ether_header ehdr;
- struct lladdr_info src, dst;
- u_int route_len = 0, hdr_len = TOKEN_HDRLEN;
- int seg;
- trp = (const struct token_header *)p;
- if (caplen < TOKEN_HDRLEN) {
- ND_PRINT((ndo, "%s", tstr));
- return hdr_len;
- }
-
- extract_token_addrs(trp, (char*)ESRC(&ehdr), (char*)EDST(&ehdr));
-
- if (IS_SOURCE_ROUTED(trp)) {
-
- *ESRC(&ehdr) &= 0x7f;
- if (ndo->ndo_eflag)
- token_hdr_print(ndo, trp, length, ESRC(&ehdr), EDST(&ehdr));
- if (caplen < TOKEN_HDRLEN + 2) {
- ND_PRINT((ndo, "%s", tstr));
- return hdr_len;
- }
- route_len = RIF_LENGTH(trp);
- hdr_len += route_len;
- if (caplen < hdr_len) {
- ND_PRINT((ndo, "%s", tstr));
- return hdr_len;
- }
- if (ndo->ndo_vflag) {
- ND_PRINT((ndo, "%s ", broadcast_indicator[BROADCAST(trp)]));
- ND_PRINT((ndo, "%s", direction[DIRECTION(trp)]));
- for (seg = 0; seg < SEGMENT_COUNT(trp); seg++)
- ND_PRINT((ndo, " [%d:%d]", RING_NUMBER(trp, seg),
- BRIDGE_NUMBER(trp, seg)));
- } else {
- ND_PRINT((ndo, "rt = %x", EXTRACT_16BITS(&trp->token_rcf)));
- for (seg = 0; seg < SEGMENT_COUNT(trp); seg++)
- ND_PRINT((ndo, ":%x", EXTRACT_16BITS(&trp->token_rseg[seg])));
- }
- ND_PRINT((ndo, " (%s) ", largest_frame[LARGEST_FRAME(trp)]));
- } else {
- if (ndo->ndo_eflag)
- token_hdr_print(ndo, trp, length, ESRC(&ehdr), EDST(&ehdr));
- }
- src.addr = ESRC(&ehdr);
- src.addr_string = etheraddr_string;
- dst.addr = EDST(&ehdr);
- dst.addr_string = etheraddr_string;
-
- length -= hdr_len;
- p += hdr_len;
- caplen -= hdr_len;
-
- if (FRAME_TYPE(trp) == TOKEN_FC_LLC) {
-
- llc_hdrlen = llc_print(ndo, p, length, caplen, &src, &dst);
- if (llc_hdrlen < 0) {
-
- if (!ndo->ndo_suppress_default_print)
- ND_DEFAULTPRINT(p, caplen);
- llc_hdrlen = -llc_hdrlen;
- }
- hdr_len += llc_hdrlen;
- } else {
-
-
- if (!ndo->ndo_eflag)
- token_hdr_print(ndo, trp, length + TOKEN_HDRLEN + route_len,
- ESRC(&ehdr), EDST(&ehdr));
- if (!ndo->ndo_suppress_default_print)
- ND_DEFAULTPRINT(p, caplen);
- }
- return (hdr_len);
- }
- u_int
- token_if_print(netdissect_options *ndo, const struct pcap_pkthdr *h, const u_char *p)
- {
- return (token_print(ndo, p, h->len, h->caplen));
- }
|