1999-10-08 07:47:09 +08:00
|
|
|
/*
|
|
|
|
* 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 (brad@fcr.com).
|
|
|
|
*/
|
|
|
|
|
2016-08-14 21:42:19 +08:00
|
|
|
/* \summary: Novell IPX printer */
|
|
|
|
|
1999-11-21 17:36:43 +08:00
|
|
|
#ifdef HAVE_CONFIG_H
|
2018-01-22 04:27:08 +08:00
|
|
|
#include <config.h>
|
1999-10-08 07:47:09 +08:00
|
|
|
#endif
|
|
|
|
|
2018-01-20 22:59:49 +08:00
|
|
|
#include "netdissect-stdinc.h"
|
1999-10-08 07:47:09 +08:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2020-10-13 03:01:05 +08:00
|
|
|
#define ND_LONGJMP_FROM_TCHECK
|
2015-09-06 05:35:58 +08:00
|
|
|
#include "netdissect.h"
|
1999-10-08 07:47:09 +08:00
|
|
|
#include "addrtoname.h"
|
|
|
|
#include "extract.h"
|
|
|
|
|
2013-12-31 07:52:13 +08:00
|
|
|
/* 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 {
|
2017-12-31 07:17:03 +08:00
|
|
|
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) */
|
2018-01-31 02:03:49 +08:00
|
|
|
nd_uint32_t dstNet; /* destination net */
|
2020-10-18 03:44:27 +08:00
|
|
|
nd_mac_addr dstNode; /* destination node */
|
2017-12-31 07:17:03 +08:00
|
|
|
nd_uint16_t dstSkt; /* destination socket */
|
2018-01-31 02:03:49 +08:00
|
|
|
nd_uint32_t srcNet; /* source net */
|
2020-10-18 03:44:27 +08:00
|
|
|
nd_mac_addr srcNode; /* source node */
|
2017-12-31 07:17:03 +08:00
|
|
|
nd_uint16_t srcSkt; /* source socket */
|
2013-12-31 07:52:13 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
#define ipxSize 30
|
1999-10-08 07:47:09 +08:00
|
|
|
|
2018-06-15 21:44:51 +08:00
|
|
|
static const char *ipxaddr_string(netdissect_options *, uint32_t, const u_char *);
|
2014-03-13 21:21:55 +08:00
|
|
|
static void ipx_decode(netdissect_options *, const struct ipxHdr *, const u_char *, u_int);
|
2017-12-04 14:19:12 +08:00
|
|
|
static void ipx_sap_print(netdissect_options *, const u_char *, u_int);
|
|
|
|
static void ipx_rip_print(netdissect_options *, const u_char *, u_int);
|
1999-10-08 07:47:09 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Print IPX datagram packets.
|
|
|
|
*/
|
|
|
|
void
|
2014-03-13 21:21:55 +08:00
|
|
|
ipx_print(netdissect_options *ndo, const u_char *p, u_int length)
|
1999-10-08 07:47:09 +08:00
|
|
|
{
|
|
|
|
const struct ipxHdr *ipx = (const struct ipxHdr *)p;
|
|
|
|
|
2018-03-14 23:54:17 +08:00
|
|
|
ndo->ndo_protocol = "ipx";
|
2014-03-13 21:21:55 +08:00
|
|
|
if (!ndo->ndo_eflag)
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("IPX ");
|
2004-05-27 03:57:57 +08:00
|
|
|
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("%s.%04x > ",
|
2018-06-16 23:23:21 +08:00
|
|
|
ipxaddr_string(ndo, GET_BE_U_4(ipx->srcNet), ipx->srcNode),
|
|
|
|
GET_BE_U_2(ipx->srcSkt));
|
1999-10-08 07:47:09 +08:00
|
|
|
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("%s.%04x: ",
|
2018-06-16 23:23:21 +08:00
|
|
|
ipxaddr_string(ndo, GET_BE_U_4(ipx->dstNet), ipx->dstNode),
|
|
|
|
GET_BE_U_2(ipx->dstSkt));
|
1999-10-08 07:47:09 +08:00
|
|
|
|
|
|
|
/* take length from ipx header */
|
2018-06-16 23:23:21 +08:00
|
|
|
length = GET_BE_U_2(ipx->length);
|
1999-10-08 07:47:09 +08:00
|
|
|
|
2019-03-12 05:07:06 +08:00
|
|
|
if (length < ipxSize) {
|
|
|
|
ND_PRINT("[length %u < %u]", length, ipxSize);
|
|
|
|
nd_print_invalid(ndo);
|
|
|
|
return;
|
|
|
|
}
|
2015-04-27 08:24:42 +08:00
|
|
|
ipx_decode(ndo, ipx, p + ipxSize, length - ipxSize);
|
1999-10-08 07:47:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static const char *
|
2018-06-15 21:44:51 +08:00
|
|
|
ipxaddr_string(netdissect_options *ndo, uint32_t net, const u_char *node)
|
1999-10-08 07:47:09 +08:00
|
|
|
{
|
|
|
|
static char line[256];
|
|
|
|
|
2019-08-09 13:39:54 +08:00
|
|
|
snprintf(line, sizeof(line), "%08x.%02x:%02x:%02x:%02x:%02x:%02x",
|
2018-06-16 23:23:21 +08:00
|
|
|
net, GET_U_1(node), GET_U_1(node + 1),
|
|
|
|
GET_U_1(node + 2), GET_U_1(node + 3),
|
|
|
|
GET_U_1(node + 4), GET_U_1(node + 5));
|
1999-10-08 07:47:09 +08:00
|
|
|
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
|
2014-03-13 21:21:55 +08:00
|
|
|
static void
|
|
|
|
ipx_decode(netdissect_options *ndo, const struct ipxHdr *ipx, const u_char *datap, u_int length)
|
1999-10-08 07:47:09 +08:00
|
|
|
{
|
2017-12-14 02:17:47 +08:00
|
|
|
u_short dstSkt;
|
1999-10-08 07:47:09 +08:00
|
|
|
|
2018-06-16 23:23:21 +08:00
|
|
|
dstSkt = GET_BE_U_2(ipx->dstSkt);
|
1999-10-08 07:47:09 +08:00
|
|
|
switch (dstSkt) {
|
|
|
|
case IPX_SKT_NCP:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-ncp %u", length);
|
1999-10-08 07:47:09 +08:00
|
|
|
break;
|
|
|
|
case IPX_SKT_SAP:
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx_sap_print(ndo, datap, length);
|
1999-10-08 07:47:09 +08:00
|
|
|
break;
|
|
|
|
case IPX_SKT_RIP:
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx_rip_print(ndo, datap, length);
|
1999-10-08 07:47:09 +08:00
|
|
|
break;
|
|
|
|
case IPX_SKT_NETBIOS:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-netbios %u", length);
|
2015-10-08 22:36:23 +08:00
|
|
|
#ifdef ENABLE_SMB
|
2014-04-02 12:14:19 +08:00
|
|
|
ipx_netbios_print(ndo, datap, length);
|
2001-10-04 17:18:45 +08:00
|
|
|
#endif
|
1999-10-08 07:47:09 +08:00
|
|
|
break;
|
|
|
|
case IPX_SKT_DIAGNOSTICS:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-diags %u", length);
|
1999-10-08 07:47:09 +08:00
|
|
|
break;
|
2001-01-15 11:23:58 +08:00
|
|
|
case IPX_SKT_NWLINK_DGM:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-nwlink-dgm %u", length);
|
2015-10-08 22:36:23 +08:00
|
|
|
#ifdef ENABLE_SMB
|
2014-04-02 12:14:19 +08:00
|
|
|
ipx_netbios_print(ndo, datap, length);
|
2001-10-04 17:18:45 +08:00
|
|
|
#endif
|
2001-01-15 11:23:58 +08:00
|
|
|
break;
|
2001-01-15 08:43:59 +08:00
|
|
|
case IPX_SKT_EIGRP:
|
2014-03-16 03:25:06 +08:00
|
|
|
eigrp_print(ndo, datap, length);
|
2001-01-15 08:43:59 +08:00
|
|
|
break;
|
1999-10-08 07:47:09 +08:00
|
|
|
default:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-#%x %u", dstSkt, length);
|
1999-10-08 07:47:09 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-13 21:21:55 +08:00
|
|
|
static void
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx_sap_print(netdissect_options *ndo, const u_char *ipx, u_int length)
|
1999-10-08 07:47:09 +08:00
|
|
|
{
|
|
|
|
int command, i;
|
|
|
|
|
2018-06-16 23:23:21 +08:00
|
|
|
command = GET_BE_U_2(ipx);
|
Update the ND_LCHECK*() macros to ND_ICHECK*() macros
ICHECK like Invalid-Check.
Reminder: If the checked expression is true an error message is printed
and a 'goto invalid' is executed.
This change adds the parameter 'operator'.
Before this change, '<' comparison was hard coded.
We can do now:
ND_ICHECK_U(length, <, HEADER_LEN);
ND_ICHECK_U(length, ==, 24);
ND_ICHECK_U(length, !=, 8);
ND_ICHECK_ZU(length, <, sizeof(struct my_struct));
ND_ICHECKMSG_U("message length", msg_tlen, <, 4);
...
(Any comparison operator)
Remark: The change of names from ND_LCHECK*() to ND_ICHECK*() is
because something else than a length(L) can be checked.
Moreover:
Place the 'message' parameter at the beginning of ND_ICHECKMSG_U()
and ND_ICHECKMSG_ZU() paramaters lists.
2021-12-29 21:38:08 +08:00
|
|
|
ND_ICHECK_U(length, <, 2);
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx += 2;
|
1999-10-08 07:47:09 +08:00
|
|
|
length -= 2;
|
|
|
|
|
|
|
|
switch (command) {
|
|
|
|
case 1:
|
|
|
|
case 3:
|
|
|
|
if (command == 1)
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-sap-req");
|
1999-10-08 07:47:09 +08:00
|
|
|
else
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-sap-nearest-req");
|
1999-10-08 07:47:09 +08:00
|
|
|
|
2018-06-16 23:23:21 +08:00
|
|
|
ND_PRINT(" %s", ipxsap_string(ndo, htons(GET_BE_U_2(ipx))));
|
1999-10-08 07:47:09 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
case 4:
|
|
|
|
if (command == 2)
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-sap-resp");
|
1999-10-08 07:47:09 +08:00
|
|
|
else
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-sap-nearest-resp");
|
1999-10-08 07:47:09 +08:00
|
|
|
|
2017-12-04 14:19:12 +08:00
|
|
|
for (i = 0; i < 8 && length != 0; i++) {
|
2022-07-12 22:03:07 +08:00
|
|
|
ND_ICHECK_U(length, <, 2);
|
2018-06-16 23:23:21 +08:00
|
|
|
ND_PRINT(" %s '", ipxsap_string(ndo, htons(GET_BE_U_2(ipx))));
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx += 2;
|
|
|
|
length -= 2;
|
|
|
|
if (length < 48) {
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("'");
|
2020-10-13 03:01:05 +08:00
|
|
|
goto invalid;
|
2017-12-04 14:19:12 +08:00
|
|
|
}
|
2020-12-11 03:40:15 +08:00
|
|
|
nd_printjnp(ndo, ipx, 48);
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("'");
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx += 48;
|
|
|
|
length -= 48;
|
|
|
|
/*
|
|
|
|
* 10 bytes of IPX address.
|
|
|
|
*/
|
2022-07-12 22:03:07 +08:00
|
|
|
ND_ICHECK_U(length, <, 10);
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT(" addr %s",
|
2018-06-16 23:23:21 +08:00
|
|
|
ipxaddr_string(ndo, GET_BE_U_4(ipx), ipx + 4));
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx += 10;
|
|
|
|
length -= 10;
|
|
|
|
/*
|
|
|
|
* 2 bytes of socket and 2 bytes of number of intermediate
|
|
|
|
* networks.
|
|
|
|
*/
|
2022-07-12 22:03:07 +08:00
|
|
|
ND_ICHECK_U(length, <, 4);
|
2018-02-12 21:24:07 +08:00
|
|
|
ND_TCHECK_4(ipx);
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx += 4;
|
|
|
|
length -= 4;
|
1999-10-08 07:47:09 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-sap-?%x", command);
|
1999-10-08 07:47:09 +08:00
|
|
|
break;
|
|
|
|
}
|
2001-01-15 10:23:25 +08:00
|
|
|
return;
|
2020-10-13 03:01:05 +08:00
|
|
|
|
|
|
|
invalid:
|
|
|
|
nd_print_invalid(ndo);
|
1999-10-08 07:47:09 +08:00
|
|
|
}
|
|
|
|
|
2014-03-13 21:21:55 +08:00
|
|
|
static void
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx_rip_print(netdissect_options *ndo, const u_char *ipx, u_int length)
|
1999-10-08 07:47:09 +08:00
|
|
|
{
|
|
|
|
int command, i;
|
|
|
|
|
2018-06-16 23:23:21 +08:00
|
|
|
command = GET_BE_U_2(ipx);
|
Update the ND_LCHECK*() macros to ND_ICHECK*() macros
ICHECK like Invalid-Check.
Reminder: If the checked expression is true an error message is printed
and a 'goto invalid' is executed.
This change adds the parameter 'operator'.
Before this change, '<' comparison was hard coded.
We can do now:
ND_ICHECK_U(length, <, HEADER_LEN);
ND_ICHECK_U(length, ==, 24);
ND_ICHECK_U(length, !=, 8);
ND_ICHECK_ZU(length, <, sizeof(struct my_struct));
ND_ICHECKMSG_U("message length", msg_tlen, <, 4);
...
(Any comparison operator)
Remark: The change of names from ND_LCHECK*() to ND_ICHECK*() is
because something else than a length(L) can be checked.
Moreover:
Place the 'message' parameter at the beginning of ND_ICHECKMSG_U()
and ND_ICHECKMSG_ZU() paramaters lists.
2021-12-29 21:38:08 +08:00
|
|
|
ND_ICHECK_U(length, <, 2);
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx += 2;
|
1999-10-08 07:47:09 +08:00
|
|
|
length -= 2;
|
|
|
|
|
|
|
|
switch (command) {
|
|
|
|
case 1:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-rip-req");
|
2017-12-04 14:19:12 +08:00
|
|
|
if (length != 0) {
|
2022-07-12 22:03:07 +08:00
|
|
|
ND_ICHECK_U(length, <, 8);
|
2018-06-16 23:23:21 +08:00
|
|
|
ND_PRINT(" %08x/%u.%u", GET_BE_U_4(ipx),
|
|
|
|
GET_BE_U_2(ipx + 4), GET_BE_U_2(ipx + 6));
|
1999-10-08 07:47:09 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-rip-resp");
|
2017-12-04 14:19:12 +08:00
|
|
|
for (i = 0; i < 50 && length != 0; i++) {
|
2022-07-12 22:03:07 +08:00
|
|
|
ND_ICHECK_U(length, <, 8);
|
2018-06-16 23:23:21 +08:00
|
|
|
ND_PRINT(" %08x/%u.%u", GET_BE_U_4(ipx),
|
|
|
|
GET_BE_U_2(ipx + 4), GET_BE_U_2(ipx + 6));
|
1999-10-08 07:47:09 +08:00
|
|
|
|
2017-12-04 14:19:12 +08:00
|
|
|
ipx += 8;
|
1999-10-08 07:47:09 +08:00
|
|
|
length -= 8;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2018-01-07 18:47:30 +08:00
|
|
|
ND_PRINT("ipx-rip-?%x", command);
|
2001-01-15 10:23:25 +08:00
|
|
|
break;
|
1999-10-08 07:47:09 +08:00
|
|
|
}
|
2001-01-15 10:23:25 +08:00
|
|
|
return;
|
2020-10-13 03:01:05 +08:00
|
|
|
|
|
|
|
invalid:
|
|
|
|
nd_print_invalid(ndo);
|
1999-10-08 07:47:09 +08:00
|
|
|
}
|