141c99275SPeter Avalos /*
241c99275SPeter Avalos * Copyright (c) 1994, 1995, 1996
341c99275SPeter Avalos * The Regents of the University of California. All rights reserved.
441c99275SPeter Avalos *
541c99275SPeter Avalos * Redistribution and use in source and binary forms, with or without
641c99275SPeter Avalos * modification, are permitted provided that: (1) source code distributions
741c99275SPeter Avalos * retain the above copyright notice and this paragraph in its entirety, (2)
841c99275SPeter Avalos * distributions including binary code include the above copyright notice and
941c99275SPeter Avalos * this paragraph in its entirety in the documentation or other materials
1041c99275SPeter Avalos * provided with the distribution, and (3) all advertising materials mentioning
1141c99275SPeter Avalos * features or use of this software display the following acknowledgement:
1241c99275SPeter Avalos * ``This product includes software developed by the University of California,
1341c99275SPeter Avalos * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
1441c99275SPeter Avalos * the University nor the names of its contributors may be used to endorse
1541c99275SPeter Avalos * or promote products derived from this software without specific prior
1641c99275SPeter Avalos * written permission.
1741c99275SPeter Avalos * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
1841c99275SPeter Avalos * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
1941c99275SPeter Avalos * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
2041c99275SPeter Avalos *
2141c99275SPeter Avalos * Contributed by Brad Parker (brad@fcr.com).
2241c99275SPeter Avalos */
2341c99275SPeter Avalos
24411677aeSAaron LI /* \summary: Novell IPX printer */
2541c99275SPeter Avalos
2641c99275SPeter Avalos #ifdef HAVE_CONFIG_H
27*ed775ee7SAntonio Huete Jimenez #include <config.h>
2841c99275SPeter Avalos #endif
2941c99275SPeter Avalos
30*ed775ee7SAntonio Huete Jimenez #include "netdissect-stdinc.h"
3141c99275SPeter Avalos
3241c99275SPeter Avalos #include <stdio.h>
3341c99275SPeter Avalos
34*ed775ee7SAntonio Huete Jimenez #define ND_LONGJMP_FROM_TCHECK
35411677aeSAaron LI #include "netdissect.h"
3641c99275SPeter Avalos #include "addrtoname.h"
3741c99275SPeter Avalos #include "extract.h"
3841c99275SPeter Avalos
39411677aeSAaron LI /* well-known sockets */
40411677aeSAaron LI #define IPX_SKT_NCP 0x0451
41411677aeSAaron LI #define IPX_SKT_SAP 0x0452
42411677aeSAaron LI #define IPX_SKT_RIP 0x0453
43411677aeSAaron LI #define IPX_SKT_NETBIOS 0x0455
44411677aeSAaron LI #define IPX_SKT_DIAGNOSTICS 0x0456
45411677aeSAaron LI #define IPX_SKT_NWLINK_DGM 0x0553 /* NWLink datagram, may contain SMB */
46411677aeSAaron LI #define IPX_SKT_EIGRP 0x85be /* Cisco EIGRP over IPX */
4741c99275SPeter Avalos
48411677aeSAaron LI /* IPX transport header */
49411677aeSAaron LI struct ipxHdr {
50*ed775ee7SAntonio Huete Jimenez nd_uint16_t cksum; /* Checksum */
51*ed775ee7SAntonio Huete Jimenez nd_uint16_t length; /* Length, in bytes, including header */
52*ed775ee7SAntonio Huete Jimenez nd_uint8_t tCtl; /* Transport Control (i.e. hop count) */
53*ed775ee7SAntonio Huete Jimenez nd_uint8_t pType; /* Packet Type (i.e. level 2 protocol) */
54*ed775ee7SAntonio Huete Jimenez nd_uint32_t dstNet; /* destination net */
55*ed775ee7SAntonio Huete Jimenez nd_mac_addr dstNode; /* destination node */
56*ed775ee7SAntonio Huete Jimenez nd_uint16_t dstSkt; /* destination socket */
57*ed775ee7SAntonio Huete Jimenez nd_uint32_t srcNet; /* source net */
58*ed775ee7SAntonio Huete Jimenez nd_mac_addr srcNode; /* source node */
59*ed775ee7SAntonio Huete Jimenez nd_uint16_t srcSkt; /* source socket */
60411677aeSAaron LI };
61411677aeSAaron LI
62411677aeSAaron LI #define ipxSize 30
63411677aeSAaron LI
64*ed775ee7SAntonio Huete Jimenez static const char *ipxaddr_string(netdissect_options *, uint32_t, const u_char *);
65411677aeSAaron LI static void ipx_decode(netdissect_options *, const struct ipxHdr *, const u_char *, u_int);
66*ed775ee7SAntonio Huete Jimenez static void ipx_sap_print(netdissect_options *, const u_char *, u_int);
67*ed775ee7SAntonio Huete Jimenez static void ipx_rip_print(netdissect_options *, const u_char *, u_int);
6841c99275SPeter Avalos
6941c99275SPeter Avalos /*
7041c99275SPeter Avalos * Print IPX datagram packets.
7141c99275SPeter Avalos */
7241c99275SPeter Avalos void
ipx_print(netdissect_options * ndo,const u_char * p,u_int length)73411677aeSAaron LI ipx_print(netdissect_options *ndo, const u_char *p, u_int length)
7441c99275SPeter Avalos {
7541c99275SPeter Avalos const struct ipxHdr *ipx = (const struct ipxHdr *)p;
7641c99275SPeter Avalos
77*ed775ee7SAntonio Huete Jimenez ndo->ndo_protocol = "ipx";
78411677aeSAaron LI if (!ndo->ndo_eflag)
79*ed775ee7SAntonio Huete Jimenez ND_PRINT("IPX ");
8041c99275SPeter Avalos
81*ed775ee7SAntonio Huete Jimenez ND_PRINT("%s.%04x > ",
82*ed775ee7SAntonio Huete Jimenez ipxaddr_string(ndo, GET_BE_U_4(ipx->srcNet), ipx->srcNode),
83*ed775ee7SAntonio Huete Jimenez GET_BE_U_2(ipx->srcSkt));
8441c99275SPeter Avalos
85*ed775ee7SAntonio Huete Jimenez ND_PRINT("%s.%04x: ",
86*ed775ee7SAntonio Huete Jimenez ipxaddr_string(ndo, GET_BE_U_4(ipx->dstNet), ipx->dstNode),
87*ed775ee7SAntonio Huete Jimenez GET_BE_U_2(ipx->dstSkt));
8841c99275SPeter Avalos
8941c99275SPeter Avalos /* take length from ipx header */
90*ed775ee7SAntonio Huete Jimenez length = GET_BE_U_2(ipx->length);
9141c99275SPeter Avalos
92*ed775ee7SAntonio Huete Jimenez if (length < ipxSize) {
93*ed775ee7SAntonio Huete Jimenez ND_PRINT("[length %u < %u]", length, ipxSize);
94*ed775ee7SAntonio Huete Jimenez nd_print_invalid(ndo);
9541c99275SPeter Avalos return;
96*ed775ee7SAntonio Huete Jimenez }
97*ed775ee7SAntonio Huete Jimenez ipx_decode(ndo, ipx, p + ipxSize, length - ipxSize);
9841c99275SPeter Avalos }
9941c99275SPeter Avalos
10041c99275SPeter Avalos static const char *
ipxaddr_string(netdissect_options * ndo,uint32_t net,const u_char * node)101*ed775ee7SAntonio Huete Jimenez ipxaddr_string(netdissect_options *ndo, uint32_t net, const u_char *node)
10241c99275SPeter Avalos {
10341c99275SPeter Avalos static char line[256];
10441c99275SPeter Avalos
10541c99275SPeter Avalos snprintf(line, sizeof(line), "%08x.%02x:%02x:%02x:%02x:%02x:%02x",
106*ed775ee7SAntonio Huete Jimenez net, GET_U_1(node), GET_U_1(node + 1),
107*ed775ee7SAntonio Huete Jimenez GET_U_1(node + 2), GET_U_1(node + 3),
108*ed775ee7SAntonio Huete Jimenez GET_U_1(node + 4), GET_U_1(node + 5));
10941c99275SPeter Avalos
11041c99275SPeter Avalos return line;
11141c99275SPeter Avalos }
11241c99275SPeter Avalos
113411677aeSAaron LI static void
ipx_decode(netdissect_options * ndo,const struct ipxHdr * ipx,const u_char * datap,u_int length)114411677aeSAaron LI ipx_decode(netdissect_options *ndo, const struct ipxHdr *ipx, const u_char *datap, u_int length)
11541c99275SPeter Avalos {
116*ed775ee7SAntonio Huete Jimenez u_short dstSkt;
11741c99275SPeter Avalos
118*ed775ee7SAntonio Huete Jimenez dstSkt = GET_BE_U_2(ipx->dstSkt);
11941c99275SPeter Avalos switch (dstSkt) {
12041c99275SPeter Avalos case IPX_SKT_NCP:
121*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-ncp %u", length);
12241c99275SPeter Avalos break;
12341c99275SPeter Avalos case IPX_SKT_SAP:
124*ed775ee7SAntonio Huete Jimenez ipx_sap_print(ndo, datap, length);
12541c99275SPeter Avalos break;
12641c99275SPeter Avalos case IPX_SKT_RIP:
127*ed775ee7SAntonio Huete Jimenez ipx_rip_print(ndo, datap, length);
12841c99275SPeter Avalos break;
12941c99275SPeter Avalos case IPX_SKT_NETBIOS:
130*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-netbios %u", length);
131411677aeSAaron LI #ifdef ENABLE_SMB
132411677aeSAaron LI ipx_netbios_print(ndo, datap, length);
13341c99275SPeter Avalos #endif
13441c99275SPeter Avalos break;
13541c99275SPeter Avalos case IPX_SKT_DIAGNOSTICS:
136*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-diags %u", length);
13741c99275SPeter Avalos break;
13841c99275SPeter Avalos case IPX_SKT_NWLINK_DGM:
139*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-nwlink-dgm %u", length);
140411677aeSAaron LI #ifdef ENABLE_SMB
141411677aeSAaron LI ipx_netbios_print(ndo, datap, length);
14241c99275SPeter Avalos #endif
14341c99275SPeter Avalos break;
14441c99275SPeter Avalos case IPX_SKT_EIGRP:
145411677aeSAaron LI eigrp_print(ndo, datap, length);
14641c99275SPeter Avalos break;
14741c99275SPeter Avalos default:
148*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-#%x %u", dstSkt, length);
14941c99275SPeter Avalos break;
15041c99275SPeter Avalos }
15141c99275SPeter Avalos }
15241c99275SPeter Avalos
153411677aeSAaron LI static void
ipx_sap_print(netdissect_options * ndo,const u_char * ipx,u_int length)154*ed775ee7SAntonio Huete Jimenez ipx_sap_print(netdissect_options *ndo, const u_char *ipx, u_int length)
15541c99275SPeter Avalos {
15641c99275SPeter Avalos int command, i;
15741c99275SPeter Avalos
158*ed775ee7SAntonio Huete Jimenez command = GET_BE_U_2(ipx);
159*ed775ee7SAntonio Huete Jimenez ipx += 2;
16041c99275SPeter Avalos length -= 2;
16141c99275SPeter Avalos
16241c99275SPeter Avalos switch (command) {
16341c99275SPeter Avalos case 1:
16441c99275SPeter Avalos case 3:
16541c99275SPeter Avalos if (command == 1)
166*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-sap-req");
16741c99275SPeter Avalos else
168*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-sap-nearest-req");
16941c99275SPeter Avalos
170*ed775ee7SAntonio Huete Jimenez ND_PRINT(" %s", ipxsap_string(ndo, htons(GET_BE_U_2(ipx))));
17141c99275SPeter Avalos break;
17241c99275SPeter Avalos
17341c99275SPeter Avalos case 2:
17441c99275SPeter Avalos case 4:
17541c99275SPeter Avalos if (command == 2)
176*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-sap-resp");
17741c99275SPeter Avalos else
178*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-sap-nearest-resp");
17941c99275SPeter Avalos
180*ed775ee7SAntonio Huete Jimenez for (i = 0; i < 8 && length != 0; i++) {
181*ed775ee7SAntonio Huete Jimenez ND_TCHECK_2(ipx);
182*ed775ee7SAntonio Huete Jimenez if (length < 2)
183*ed775ee7SAntonio Huete Jimenez goto invalid;
184*ed775ee7SAntonio Huete Jimenez ND_PRINT(" %s '", ipxsap_string(ndo, htons(GET_BE_U_2(ipx))));
185*ed775ee7SAntonio Huete Jimenez ipx += 2;
186*ed775ee7SAntonio Huete Jimenez length -= 2;
187*ed775ee7SAntonio Huete Jimenez if (length < 48) {
188*ed775ee7SAntonio Huete Jimenez ND_PRINT("'");
189*ed775ee7SAntonio Huete Jimenez goto invalid;
19041c99275SPeter Avalos }
191*ed775ee7SAntonio Huete Jimenez nd_printjnp(ndo, ipx, 48);
192*ed775ee7SAntonio Huete Jimenez ND_PRINT("'");
193*ed775ee7SAntonio Huete Jimenez ipx += 48;
194*ed775ee7SAntonio Huete Jimenez length -= 48;
195*ed775ee7SAntonio Huete Jimenez /*
196*ed775ee7SAntonio Huete Jimenez * 10 bytes of IPX address.
197*ed775ee7SAntonio Huete Jimenez */
198*ed775ee7SAntonio Huete Jimenez ND_TCHECK_LEN(ipx, 10);
199*ed775ee7SAntonio Huete Jimenez if (length < 10)
200*ed775ee7SAntonio Huete Jimenez goto invalid;
201*ed775ee7SAntonio Huete Jimenez ND_PRINT(" addr %s",
202*ed775ee7SAntonio Huete Jimenez ipxaddr_string(ndo, GET_BE_U_4(ipx), ipx + 4));
203*ed775ee7SAntonio Huete Jimenez ipx += 10;
204*ed775ee7SAntonio Huete Jimenez length -= 10;
205*ed775ee7SAntonio Huete Jimenez /*
206*ed775ee7SAntonio Huete Jimenez * 2 bytes of socket and 2 bytes of number of intermediate
207*ed775ee7SAntonio Huete Jimenez * networks.
208*ed775ee7SAntonio Huete Jimenez */
209*ed775ee7SAntonio Huete Jimenez ND_TCHECK_4(ipx);
210*ed775ee7SAntonio Huete Jimenez if (length < 4)
211*ed775ee7SAntonio Huete Jimenez goto invalid;
212*ed775ee7SAntonio Huete Jimenez ipx += 4;
213*ed775ee7SAntonio Huete Jimenez length -= 4;
21441c99275SPeter Avalos }
21541c99275SPeter Avalos break;
21641c99275SPeter Avalos default:
217*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-sap-?%x", command);
21841c99275SPeter Avalos break;
21941c99275SPeter Avalos }
22041c99275SPeter Avalos return;
221*ed775ee7SAntonio Huete Jimenez
222*ed775ee7SAntonio Huete Jimenez invalid:
223*ed775ee7SAntonio Huete Jimenez nd_print_invalid(ndo);
22441c99275SPeter Avalos }
22541c99275SPeter Avalos
226411677aeSAaron LI static void
ipx_rip_print(netdissect_options * ndo,const u_char * ipx,u_int length)227*ed775ee7SAntonio Huete Jimenez ipx_rip_print(netdissect_options *ndo, const u_char *ipx, u_int length)
22841c99275SPeter Avalos {
22941c99275SPeter Avalos int command, i;
23041c99275SPeter Avalos
231*ed775ee7SAntonio Huete Jimenez command = GET_BE_U_2(ipx);
232*ed775ee7SAntonio Huete Jimenez ipx += 2;
23341c99275SPeter Avalos length -= 2;
23441c99275SPeter Avalos
23541c99275SPeter Avalos switch (command) {
23641c99275SPeter Avalos case 1:
237*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-rip-req");
238*ed775ee7SAntonio Huete Jimenez if (length != 0) {
239*ed775ee7SAntonio Huete Jimenez if (length < 8)
240*ed775ee7SAntonio Huete Jimenez goto invalid;
241*ed775ee7SAntonio Huete Jimenez ND_PRINT(" %08x/%u.%u", GET_BE_U_4(ipx),
242*ed775ee7SAntonio Huete Jimenez GET_BE_U_2(ipx + 4), GET_BE_U_2(ipx + 6));
24341c99275SPeter Avalos }
24441c99275SPeter Avalos break;
24541c99275SPeter Avalos case 2:
246*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-rip-resp");
247*ed775ee7SAntonio Huete Jimenez for (i = 0; i < 50 && length != 0; i++) {
248*ed775ee7SAntonio Huete Jimenez if (length < 8)
249*ed775ee7SAntonio Huete Jimenez goto invalid;
250*ed775ee7SAntonio Huete Jimenez ND_PRINT(" %08x/%u.%u", GET_BE_U_4(ipx),
251*ed775ee7SAntonio Huete Jimenez GET_BE_U_2(ipx + 4), GET_BE_U_2(ipx + 6));
25241c99275SPeter Avalos
253*ed775ee7SAntonio Huete Jimenez ipx += 8;
25441c99275SPeter Avalos length -= 8;
25541c99275SPeter Avalos }
25641c99275SPeter Avalos break;
25741c99275SPeter Avalos default:
258*ed775ee7SAntonio Huete Jimenez ND_PRINT("ipx-rip-?%x", command);
25941c99275SPeter Avalos break;
26041c99275SPeter Avalos }
26141c99275SPeter Avalos return;
262*ed775ee7SAntonio Huete Jimenez
263*ed775ee7SAntonio Huete Jimenez invalid:
264*ed775ee7SAntonio Huete Jimenez nd_print_invalid(ndo);
26541c99275SPeter Avalos }
266