| /* |
| * Copyright (c) 1994, 1995, 1996 |
| * The Regents of the University of California. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that: (1) source code distributions |
| * retain the above copyright notice and this paragraph in its entirety, (2) |
| * distributions including binary code include the above copyright notice and |
| * this paragraph in its entirety in the documentation or other materials |
| * provided with the distribution, and (3) all advertising materials mentioning |
| * features or use of this software display the following acknowledgement: |
| * ``This product includes software developed by the University of California, |
| * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of |
| * the University nor the names of its contributors may 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. |
| * |
| * Contributed by Brad Parker ([email protected]). |
| */ |
| |
| /* \summary: Novell IPX printer */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include <config.h> |
| #endif |
| |
| #include "netdissect-stdinc.h" |
| |
| #include <stdio.h> |
| |
| #include "netdissect.h" |
| #include "addrtoname.h" |
| #include "extract.h" |
| |
| /* well-known sockets */ |
| #define IPX_SKT_NCP 0x0451 |
| #define IPX_SKT_SAP 0x0452 |
| #define IPX_SKT_RIP 0x0453 |
| #define IPX_SKT_NETBIOS 0x0455 |
| #define IPX_SKT_DIAGNOSTICS 0x0456 |
| #define IPX_SKT_NWLINK_DGM 0x0553 /* NWLink datagram, may contain SMB */ |
| #define IPX_SKT_EIGRP 0x85be /* Cisco EIGRP over IPX */ |
| |
| /* IPX transport header */ |
| struct ipxHdr { |
| nd_uint16_t cksum; /* Checksum */ |
| nd_uint16_t length; /* Length, in bytes, including header */ |
| nd_uint8_t tCtl; /* Transport Control (i.e. hop count) */ |
| nd_uint8_t pType; /* Packet Type (i.e. level 2 protocol) */ |
| nd_uint32_t dstNet; /* destination net */ |
| nd_byte dstNode[6]; /* destination node */ |
| nd_uint16_t dstSkt; /* destination socket */ |
| nd_uint32_t srcNet; /* source net */ |
| nd_byte srcNode[6]; /* source node */ |
| nd_uint16_t srcSkt; /* source socket */ |
| }; |
| |
| #define ipxSize 30 |
| |
| static const char *ipxaddr_string(uint32_t, const u_char *); |
| static void ipx_decode(netdissect_options *, const struct ipxHdr *, const u_char *, u_int); |
| static void ipx_sap_print(netdissect_options *, const u_char *, u_int); |
| static void ipx_rip_print(netdissect_options *, const u_char *, u_int); |
| |
| /* |
| * Print IPX datagram packets. |
| */ |
| void |
| ipx_print(netdissect_options *ndo, const u_char *p, u_int length) |
| { |
| const struct ipxHdr *ipx = (const struct ipxHdr *)p; |
| |
| ndo->ndo_protocol = "ipx"; |
| if (!ndo->ndo_eflag) |
| ND_PRINT("IPX "); |
| |
| ND_TCHECK_2(ipx->srcSkt); |
| ND_PRINT("%s.%04x > ", |
| ipxaddr_string(EXTRACT_BE_U_4(ipx->srcNet), ipx->srcNode), |
| EXTRACT_BE_U_2(ipx->srcSkt)); |
| |
| ND_PRINT("%s.%04x: ", |
| ipxaddr_string(EXTRACT_BE_U_4(ipx->dstNet), ipx->dstNode), |
| EXTRACT_BE_U_2(ipx->dstSkt)); |
| |
| /* take length from ipx header */ |
| ND_TCHECK_2(ipx->length); |
| length = EXTRACT_BE_U_2(ipx->length); |
| |
| ipx_decode(ndo, ipx, p + ipxSize, length - ipxSize); |
| return; |
| trunc: |
| nd_print_trunc(ndo); |
| } |
| |
| static const char * |
| ipxaddr_string(uint32_t net, const u_char *node) |
| { |
| static char line[256]; |
| |
| nd_snprintf(line, sizeof(line), "%08x.%02x:%02x:%02x:%02x:%02x:%02x", |
| net, EXTRACT_U_1(node), EXTRACT_U_1(node + 1), |
| EXTRACT_U_1(node + 2), EXTRACT_U_1(node + 3), |
| EXTRACT_U_1(node + 4), EXTRACT_U_1(node + 5)); |
| |
| return line; |
| } |
| |
| static void |
| ipx_decode(netdissect_options *ndo, const struct ipxHdr *ipx, const u_char *datap, u_int length) |
| { |
| u_short dstSkt; |
| |
| dstSkt = EXTRACT_BE_U_2(ipx->dstSkt); |
| switch (dstSkt) { |
| case IPX_SKT_NCP: |
| ND_PRINT("ipx-ncp %u", length); |
| break; |
| case IPX_SKT_SAP: |
| ipx_sap_print(ndo, datap, length); |
| break; |
| case IPX_SKT_RIP: |
| ipx_rip_print(ndo, datap, length); |
| break; |
| case IPX_SKT_NETBIOS: |
| ND_PRINT("ipx-netbios %u", length); |
| #ifdef ENABLE_SMB |
| ipx_netbios_print(ndo, datap, length); |
| #endif |
| break; |
| case IPX_SKT_DIAGNOSTICS: |
| ND_PRINT("ipx-diags %u", length); |
| break; |
| case IPX_SKT_NWLINK_DGM: |
| ND_PRINT("ipx-nwlink-dgm %u", length); |
| #ifdef ENABLE_SMB |
| ipx_netbios_print(ndo, datap, length); |
| #endif |
| break; |
| case IPX_SKT_EIGRP: |
| eigrp_print(ndo, datap, length); |
| break; |
| default: |
| ND_PRINT("ipx-#%x %u", dstSkt, length); |
| break; |
| } |
| } |
| |
| static void |
| ipx_sap_print(netdissect_options *ndo, const u_char *ipx, u_int length) |
| { |
| int command, i; |
| |
| ND_TCHECK_2(ipx); |
| command = EXTRACT_BE_U_2(ipx); |
| ipx += 2; |
| length -= 2; |
| |
| switch (command) { |
| case 1: |
| case 3: |
| if (command == 1) |
| ND_PRINT("ipx-sap-req"); |
| else |
| ND_PRINT("ipx-sap-nearest-req"); |
| |
| ND_TCHECK_2(ipx); |
| ND_PRINT(" %s", ipxsap_string(ndo, htons(EXTRACT_BE_U_2(ipx)))); |
| break; |
| |
| case 2: |
| case 4: |
| if (command == 2) |
| ND_PRINT("ipx-sap-resp"); |
| else |
| ND_PRINT("ipx-sap-nearest-resp"); |
| |
| for (i = 0; i < 8 && length != 0; i++) { |
| ND_TCHECK_2(ipx); |
| if (length < 2) |
| goto trunc; |
| ND_PRINT(" %s '", ipxsap_string(ndo, htons(EXTRACT_BE_U_2(ipx)))); |
| ipx += 2; |
| length -= 2; |
| if (length < 48) { |
| ND_PRINT("'"); |
| goto trunc; |
| } |
| if (nd_printzp(ndo, ipx, 48, ndo->ndo_snapend)) { |
| ND_PRINT("'"); |
| goto trunc; |
| } |
| ND_PRINT("'"); |
| ipx += 48; |
| length -= 48; |
| /* |
| * 10 bytes of IPX address. |
| */ |
| ND_TCHECK_LEN(ipx, 10); |
| if (length < 10) |
| goto trunc; |
| ND_PRINT(" addr %s", |
| ipxaddr_string(EXTRACT_BE_U_4(ipx), ipx + 4)); |
| ipx += 10; |
| length -= 10; |
| /* |
| * 2 bytes of socket and 2 bytes of number of intermediate |
| * networks. |
| */ |
| ND_TCHECK_4(ipx); |
| if (length < 4) |
| goto trunc; |
| ipx += 4; |
| length -= 4; |
| } |
| break; |
| default: |
| ND_PRINT("ipx-sap-?%x", command); |
| break; |
| } |
| return; |
| trunc: |
| nd_print_trunc(ndo); |
| } |
| |
| static void |
| ipx_rip_print(netdissect_options *ndo, const u_char *ipx, u_int length) |
| { |
| int command, i; |
| |
| ND_TCHECK_2(ipx); |
| command = EXTRACT_BE_U_2(ipx); |
| ipx += 2; |
| length -= 2; |
| |
| switch (command) { |
| case 1: |
| ND_PRINT("ipx-rip-req"); |
| if (length != 0) { |
| if (length < 8) |
| goto trunc; |
| ND_TCHECK_8(ipx); |
| ND_PRINT(" %08x/%u.%u", EXTRACT_BE_U_4(ipx), |
| EXTRACT_BE_U_2(ipx + 4), EXTRACT_BE_U_2(ipx + 6)); |
| } |
| break; |
| case 2: |
| ND_PRINT("ipx-rip-resp"); |
| for (i = 0; i < 50 && length != 0; i++) { |
| if (length < 8) |
| goto trunc; |
| ND_TCHECK_8(ipx); |
| ND_PRINT(" %08x/%u.%u", EXTRACT_BE_U_4(ipx), |
| EXTRACT_BE_U_2(ipx + 4), EXTRACT_BE_U_2(ipx + 6)); |
| |
| ipx += 8; |
| length -= 8; |
| } |
| break; |
| default: |
| ND_PRINT("ipx-rip-?%x", command); |
| break; |
| } |
| return; |
| trunc: |
| nd_print_trunc(ndo); |
| } |