123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480 |
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif
- #include <netdissect-stdinc.h>
- #include <string.h>
- #include "netdissect.h"
- #include "addrtoname.h"
- #include "ether.h"
- #include "ethertype.h"
- #include "extract.h"
- static const char tstr[] = "[|ARP]";
- struct arp_pkthdr {
- u_short ar_hrd;
- #define ARPHRD_ETHER 1
- #define ARPHRD_IEEE802 6
- #define ARPHRD_ARCNET 7
- #define ARPHRD_FRELAY 15
- #define ARPHRD_ATM2225 19
- #define ARPHRD_STRIP 23
- #define ARPHRD_IEEE1394 24
- u_short ar_pro;
- u_char ar_hln;
- u_char ar_pln;
- u_short ar_op;
- #define ARPOP_REQUEST 1
- #define ARPOP_REPLY 2
- #define ARPOP_REVREQUEST 3
- #define ARPOP_REVREPLY 4
- #define ARPOP_INVREQUEST 8
- #define ARPOP_INVREPLY 9
- #define ARPOP_NAK 10
- #ifdef COMMENT_ONLY
- u_char ar_sha[];
- u_char ar_spa[];
- u_char ar_tha[];
- u_char ar_tpa[];
- #endif
- #define ar_sha(ap) (((const u_char *)((ap)+1))+ 0)
- #define ar_spa(ap) (((const u_char *)((ap)+1))+ (ap)->ar_hln)
- #define ar_tha(ap) (((const u_char *)((ap)+1))+ (ap)->ar_hln+(ap)->ar_pln)
- #define ar_tpa(ap) (((const u_char *)((ap)+1))+2*(ap)->ar_hln+(ap)->ar_pln)
- };
- #define ARP_HDRLEN 8
- #define HRD(ap) EXTRACT_16BITS(&(ap)->ar_hrd)
- #define HRD_LEN(ap) ((ap)->ar_hln)
- #define PROTO_LEN(ap) ((ap)->ar_pln)
- #define OP(ap) EXTRACT_16BITS(&(ap)->ar_op)
- #define PRO(ap) EXTRACT_16BITS(&(ap)->ar_pro)
- #define SHA(ap) (ar_sha(ap))
- #define SPA(ap) (ar_spa(ap))
- #define THA(ap) (ar_tha(ap))
- #define TPA(ap) (ar_tpa(ap))
- static const struct tok arpop_values[] = {
- { ARPOP_REQUEST, "Request" },
- { ARPOP_REPLY, "Reply" },
- { ARPOP_REVREQUEST, "Reverse Request" },
- { ARPOP_REVREPLY, "Reverse Reply" },
- { ARPOP_INVREQUEST, "Inverse Request" },
- { ARPOP_INVREPLY, "Inverse Reply" },
- { ARPOP_NAK, "NACK Reply" },
- { 0, NULL }
- };
- static const struct tok arphrd_values[] = {
- { ARPHRD_ETHER, "Ethernet" },
- { ARPHRD_IEEE802, "TokenRing" },
- { ARPHRD_ARCNET, "ArcNet" },
- { ARPHRD_FRELAY, "FrameRelay" },
- { ARPHRD_STRIP, "Strip" },
- { ARPHRD_IEEE1394, "IEEE 1394" },
- { ARPHRD_ATM2225, "ATM" },
- { 0, NULL }
- };
- struct atmarp_pkthdr {
- u_short aar_hrd;
- u_short aar_pro;
- u_char aar_shtl;
- u_char aar_sstl;
- #define ATMARP_IS_E164 0x40
- #define ATMARP_LEN_MASK 0x3F
- u_short aar_op;
- u_char aar_spln;
- u_char aar_thtl;
- u_char aar_tstl;
- u_char aar_tpln;
- #ifdef COMMENT_ONLY
- u_char aar_sha[];
- u_char aar_ssa[];
- u_char aar_spa[];
- u_char aar_tha[];
- u_char aar_tsa[];
- u_char aar_tpa[];
- #endif
- #define ATMHRD(ap) EXTRACT_16BITS(&(ap)->aar_hrd)
- #define ATMSHRD_LEN(ap) ((ap)->aar_shtl & ATMARP_LEN_MASK)
- #define ATMSSLN(ap) ((ap)->aar_sstl & ATMARP_LEN_MASK)
- #define ATMSPROTO_LEN(ap) ((ap)->aar_spln)
- #define ATMOP(ap) EXTRACT_16BITS(&(ap)->aar_op)
- #define ATMPRO(ap) EXTRACT_16BITS(&(ap)->aar_pro)
- #define ATMTHRD_LEN(ap) ((ap)->aar_thtl & ATMARP_LEN_MASK)
- #define ATMTSLN(ap) ((ap)->aar_tstl & ATMARP_LEN_MASK)
- #define ATMTPROTO_LEN(ap) ((ap)->aar_tpln)
- #define aar_sha(ap) ((const u_char *)((ap)+1))
- #define aar_ssa(ap) (aar_sha(ap) + ATMSHRD_LEN(ap))
- #define aar_spa(ap) (aar_ssa(ap) + ATMSSLN(ap))
- #define aar_tha(ap) (aar_spa(ap) + ATMSPROTO_LEN(ap))
- #define aar_tsa(ap) (aar_tha(ap) + ATMTHRD_LEN(ap))
- #define aar_tpa(ap) (aar_tsa(ap) + ATMTSLN(ap))
- };
- #define ATMSHA(ap) (aar_sha(ap))
- #define ATMSSA(ap) (aar_ssa(ap))
- #define ATMSPA(ap) (aar_spa(ap))
- #define ATMTHA(ap) (aar_tha(ap))
- #define ATMTSA(ap) (aar_tsa(ap))
- #define ATMTPA(ap) (aar_tpa(ap))
- static int
- isnonzero(const u_char *a, size_t len)
- {
- while (len > 0) {
- if (*a != 0)
- return (1);
- a++;
- len--;
- }
- return (0);
- }
- static void
- tpaddr_print_ip(netdissect_options *ndo,
- const struct arp_pkthdr *ap, u_short pro)
- {
- if (pro != ETHERTYPE_IP && pro != ETHERTYPE_TRAIL)
- ND_PRINT((ndo, "<wrong proto type>"));
- else if (PROTO_LEN(ap) != 4)
- ND_PRINT((ndo, "<wrong len>"));
- else
- ND_PRINT((ndo, "%s", ipaddr_string(ndo, TPA(ap))));
- }
- static void
- spaddr_print_ip(netdissect_options *ndo,
- const struct arp_pkthdr *ap, u_short pro)
- {
- if (pro != ETHERTYPE_IP && pro != ETHERTYPE_TRAIL)
- ND_PRINT((ndo, "<wrong proto type>"));
- else if (PROTO_LEN(ap) != 4)
- ND_PRINT((ndo, "<wrong len>"));
- else
- ND_PRINT((ndo, "%s", ipaddr_string(ndo, SPA(ap))));
- }
- static void
- atmarp_addr_print(netdissect_options *ndo,
- const u_char *ha, u_int ha_len, const u_char *srca,
- u_int srca_len)
- {
- if (ha_len == 0)
- ND_PRINT((ndo, "<No address>"));
- else {
- ND_PRINT((ndo, "%s", linkaddr_string(ndo, ha, LINKADDR_ATM, ha_len)));
- if (srca_len != 0)
- ND_PRINT((ndo, ",%s",
- linkaddr_string(ndo, srca, LINKADDR_ATM, srca_len)));
- }
- }
- static void
- atmarp_tpaddr_print(netdissect_options *ndo,
- const struct atmarp_pkthdr *ap, u_short pro)
- {
- if (pro != ETHERTYPE_IP && pro != ETHERTYPE_TRAIL)
- ND_PRINT((ndo, "<wrong proto type>"));
- else if (ATMTPROTO_LEN(ap) != 4)
- ND_PRINT((ndo, "<wrong tplen>"));
- else
- ND_PRINT((ndo, "%s", ipaddr_string(ndo, ATMTPA(ap))));
- }
- static void
- atmarp_spaddr_print(netdissect_options *ndo,
- const struct atmarp_pkthdr *ap, u_short pro)
- {
- if (pro != ETHERTYPE_IP && pro != ETHERTYPE_TRAIL)
- ND_PRINT((ndo, "<wrong proto type>"));
- else if (ATMSPROTO_LEN(ap) != 4)
- ND_PRINT((ndo, "<wrong splen>"));
- else
- ND_PRINT((ndo, "%s", ipaddr_string(ndo, ATMSPA(ap))));
- }
- static void
- atmarp_print(netdissect_options *ndo,
- const u_char *bp, u_int length, u_int caplen)
- {
- const struct atmarp_pkthdr *ap;
- u_short pro, hrd, op;
- ap = (const struct atmarp_pkthdr *)bp;
- ND_TCHECK(*ap);
- hrd = ATMHRD(ap);
- pro = ATMPRO(ap);
- op = ATMOP(ap);
- if (!ND_TTEST2(*aar_tpa(ap), ATMTPROTO_LEN(ap))) {
- ND_PRINT((ndo, "%s", tstr));
- ND_DEFAULTPRINT((const u_char *)ap, length);
- return;
- }
- if (!ndo->ndo_eflag) {
- ND_PRINT((ndo, "ARP, "));
- }
- if ((pro != ETHERTYPE_IP && pro != ETHERTYPE_TRAIL) ||
- ATMSPROTO_LEN(ap) != 4 ||
- ATMTPROTO_LEN(ap) != 4 ||
- ndo->ndo_vflag) {
- ND_PRINT((ndo, "%s, %s (len %u/%u)",
- tok2str(arphrd_values, "Unknown Hardware (%u)", hrd),
- tok2str(ethertype_values, "Unknown Protocol (0x%04x)", pro),
- ATMSPROTO_LEN(ap),
- ATMTPROTO_LEN(ap)));
-
- if (!ndo->ndo_vflag) {
- goto out;
- }
- }
-
- ND_PRINT((ndo, "%s%s ",
- ndo->ndo_vflag ? ", " : "",
- tok2str(arpop_values, "Unknown (%u)", op)));
- switch (op) {
- case ARPOP_REQUEST:
- ND_PRINT((ndo, "who-has "));
- atmarp_tpaddr_print(ndo, ap, pro);
- if (ATMTHRD_LEN(ap) != 0) {
- ND_PRINT((ndo, " ("));
- atmarp_addr_print(ndo, ATMTHA(ap), ATMTHRD_LEN(ap),
- ATMTSA(ap), ATMTSLN(ap));
- ND_PRINT((ndo, ")"));
- }
- ND_PRINT((ndo, " tell "));
- atmarp_spaddr_print(ndo, ap, pro);
- break;
- case ARPOP_REPLY:
- atmarp_spaddr_print(ndo, ap, pro);
- ND_PRINT((ndo, " is-at "));
- atmarp_addr_print(ndo, ATMSHA(ap), ATMSHRD_LEN(ap), ATMSSA(ap),
- ATMSSLN(ap));
- break;
- case ARPOP_INVREQUEST:
- ND_PRINT((ndo, "who-is "));
- atmarp_addr_print(ndo, ATMTHA(ap), ATMTHRD_LEN(ap), ATMTSA(ap),
- ATMTSLN(ap));
- ND_PRINT((ndo, " tell "));
- atmarp_addr_print(ndo, ATMSHA(ap), ATMSHRD_LEN(ap), ATMSSA(ap),
- ATMSSLN(ap));
- break;
- case ARPOP_INVREPLY:
- atmarp_addr_print(ndo, ATMSHA(ap), ATMSHRD_LEN(ap), ATMSSA(ap),
- ATMSSLN(ap));
- ND_PRINT((ndo, "at "));
- atmarp_spaddr_print(ndo, ap, pro);
- break;
- case ARPOP_NAK:
- ND_PRINT((ndo, "for "));
- atmarp_spaddr_print(ndo, ap, pro);
- break;
- default:
- ND_DEFAULTPRINT((const u_char *)ap, caplen);
- return;
- }
- out:
- ND_PRINT((ndo, ", length %u", length));
- return;
- trunc:
- ND_PRINT((ndo, "%s", tstr));
- }
- void
- arp_print(netdissect_options *ndo,
- const u_char *bp, u_int length, u_int caplen)
- {
- const struct arp_pkthdr *ap;
- u_short pro, hrd, op, linkaddr;
- ap = (const struct arp_pkthdr *)bp;
- ND_TCHECK(*ap);
- hrd = HRD(ap);
- pro = PRO(ap);
- op = OP(ap);
-
- switch(hrd) {
- case ARPHRD_ATM2225:
- atmarp_print(ndo, bp, length, caplen);
- return;
- case ARPHRD_FRELAY:
- linkaddr = LINKADDR_FRELAY;
- break;
- default:
- linkaddr = LINKADDR_ETHER;
- break;
- }
- if (!ND_TTEST2(*TPA(ap), PROTO_LEN(ap))) {
- ND_PRINT((ndo, "%s", tstr));
- ND_DEFAULTPRINT((const u_char *)ap, length);
- return;
- }
- if (!ndo->ndo_eflag) {
- ND_PRINT((ndo, "ARP, "));
- }
-
- if ((pro != ETHERTYPE_IP && pro != ETHERTYPE_TRAIL) ||
- PROTO_LEN(ap) != 4 ||
- HRD_LEN(ap) == 0 ||
- ndo->ndo_vflag) {
- ND_PRINT((ndo, "%s (len %u), %s (len %u)",
- tok2str(arphrd_values, "Unknown Hardware (%u)", hrd),
- HRD_LEN(ap),
- tok2str(ethertype_values, "Unknown Protocol (0x%04x)", pro),
- PROTO_LEN(ap)));
-
- if (!ndo->ndo_vflag) {
- goto out;
- }
- }
-
- ND_PRINT((ndo, "%s%s ",
- ndo->ndo_vflag ? ", " : "",
- tok2str(arpop_values, "Unknown (%u)", op)));
- switch (op) {
- case ARPOP_REQUEST:
- ND_PRINT((ndo, "who-has "));
- tpaddr_print_ip(ndo, ap, pro);
- if (isnonzero((const u_char *)THA(ap), HRD_LEN(ap)))
- ND_PRINT((ndo, " (%s)",
- linkaddr_string(ndo, THA(ap), linkaddr, HRD_LEN(ap))));
- ND_PRINT((ndo, " tell "));
- spaddr_print_ip(ndo, ap, pro);
- break;
- case ARPOP_REPLY:
- spaddr_print_ip(ndo, ap, pro);
- ND_PRINT((ndo, " is-at %s",
- linkaddr_string(ndo, SHA(ap), linkaddr, HRD_LEN(ap))));
- break;
- case ARPOP_REVREQUEST:
- ND_PRINT((ndo, "who-is %s tell %s",
- linkaddr_string(ndo, THA(ap), linkaddr, HRD_LEN(ap)),
- linkaddr_string(ndo, SHA(ap), linkaddr, HRD_LEN(ap))));
- break;
- case ARPOP_REVREPLY:
- ND_PRINT((ndo, "%s at ",
- linkaddr_string(ndo, THA(ap), linkaddr, HRD_LEN(ap))));
- tpaddr_print_ip(ndo, ap, pro);
- break;
- case ARPOP_INVREQUEST:
- ND_PRINT((ndo, "who-is %s tell %s",
- linkaddr_string(ndo, THA(ap), linkaddr, HRD_LEN(ap)),
- linkaddr_string(ndo, SHA(ap), linkaddr, HRD_LEN(ap))));
- break;
- case ARPOP_INVREPLY:
- ND_PRINT((ndo,"%s at ",
- linkaddr_string(ndo, SHA(ap), linkaddr, HRD_LEN(ap))));
- spaddr_print_ip(ndo, ap, pro);
- break;
- default:
- ND_DEFAULTPRINT((const u_char *)ap, caplen);
- return;
- }
- out:
- ND_PRINT((ndo, ", length %u", length));
- return;
- trunc:
- ND_PRINT((ndo, "%s", tstr));
- }
|