14edb46e9SPaul Traina /* 24644f044SBill Fenner * Copyright (c) 1992, 1993, 1994, 1995, 1996, 1997 34edb46e9SPaul Traina * The Regents of the University of California. All rights reserved. 44edb46e9SPaul Traina * 54edb46e9SPaul Traina * Redistribution and use in source and binary forms, with or without 64edb46e9SPaul Traina * modification, are permitted provided that: (1) source code distributions 74edb46e9SPaul Traina * retain the above copyright notice and this paragraph in its entirety, (2) 84edb46e9SPaul Traina * distributions including binary code include the above copyright notice and 94edb46e9SPaul Traina * this paragraph in its entirety in the documentation or other materials 104edb46e9SPaul Traina * provided with the distribution, and (3) all advertising materials mentioning 114edb46e9SPaul Traina * features or use of this software display the following acknowledgement: 124edb46e9SPaul Traina * ``This product includes software developed by the University of California, 134edb46e9SPaul Traina * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of 144edb46e9SPaul Traina * the University nor the names of its contributors may be used to endorse 154edb46e9SPaul Traina * or promote products derived from this software without specific prior 164edb46e9SPaul Traina * written permission. 174edb46e9SPaul Traina * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED 184edb46e9SPaul Traina * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF 194edb46e9SPaul Traina * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. 204edb46e9SPaul Traina */ 214edb46e9SPaul Traina 223340d773SGleb Smirnoff /* \summary: DECnet printer */ 233340d773SGleb Smirnoff 24b0453382SBill Fenner #ifdef HAVE_CONFIG_H 25*ee67461eSJoseph Mingrone #include <config.h> 264edb46e9SPaul Traina #endif 274edb46e9SPaul Traina 28*ee67461eSJoseph Mingrone #include "netdissect-stdinc.h" 294edb46e9SPaul Traina 304edb46e9SPaul Traina #include <stdio.h> 314edb46e9SPaul Traina #include <stdlib.h> 324edb46e9SPaul Traina 33*ee67461eSJoseph Mingrone #define ND_LONGJMP_FROM_TCHECK 343340d773SGleb Smirnoff #include "netdissect.h" 35*ee67461eSJoseph Mingrone #include "extract.h" 364edb46e9SPaul Traina #include "addrtoname.h" 374edb46e9SPaul Traina 383c602fabSXin LI 393340d773SGleb Smirnoff #ifndef _WIN32 40*ee67461eSJoseph Mingrone typedef nd_uint8_t byte; /* single byte field */ 413c602fabSXin LI #else 423c602fabSXin LI /* 433c602fabSXin LI * the keyword 'byte' generates conflicts in Windows 443c602fabSXin LI */ 45*ee67461eSJoseph Mingrone typedef nd_uint8_t Byte; /* single byte field */ 463c602fabSXin LI #define byte Byte 473340d773SGleb Smirnoff #endif /* _WIN32 */ 48*ee67461eSJoseph Mingrone typedef nd_uint16_t word; /* 2 byte field */ 49*ee67461eSJoseph Mingrone typedef nd_uint32_t longword; /* 4 bytes field */ 503c602fabSXin LI 513c602fabSXin LI /* 523c602fabSXin LI * Definitions for DECNET Phase IV protocol headers 533c602fabSXin LI */ 54*ee67461eSJoseph Mingrone typedef union { 55*ee67461eSJoseph Mingrone nd_mac_addr dne_addr; /* full Ethernet address */ 563c602fabSXin LI struct { 57*ee67461eSJoseph Mingrone nd_byte dne_hiord[4]; /* DECnet HIORD prefix */ 58*ee67461eSJoseph Mingrone nd_byte dne_nodeaddr[2]; /* DECnet node address */ 593c602fabSXin LI } dne_remote; 60*ee67461eSJoseph Mingrone } etheraddr; /* Ethernet address */ 613c602fabSXin LI 623c602fabSXin LI #define HIORD 0x000400aa /* high 32-bits of address (swapped) */ 633c602fabSXin LI 643c602fabSXin LI #define AREAMASK 0176000 /* mask for area field */ 653c602fabSXin LI #define AREASHIFT 10 /* bit-offset for area field */ 663c602fabSXin LI #define NODEMASK 01777 /* mask for node address field */ 673c602fabSXin LI 683c602fabSXin LI /* 693c602fabSXin LI * Define long and short header formats. 703c602fabSXin LI */ 713c602fabSXin LI struct shorthdr 723c602fabSXin LI { 733c602fabSXin LI byte sh_flags; /* route flags */ 743c602fabSXin LI word sh_dst; /* destination node address */ 753c602fabSXin LI word sh_src; /* source node address */ 763c602fabSXin LI byte sh_visits; /* visit count */ 773c602fabSXin LI }; 783c602fabSXin LI 793c602fabSXin LI struct longhdr 803c602fabSXin LI { 813c602fabSXin LI byte lg_flags; /* route flags */ 823c602fabSXin LI byte lg_darea; /* destination area (reserved) */ 833c602fabSXin LI byte lg_dsarea; /* destination subarea (reserved) */ 843c602fabSXin LI etheraddr lg_dst; /* destination id */ 853c602fabSXin LI byte lg_sarea; /* source area (reserved) */ 863c602fabSXin LI byte lg_ssarea; /* source subarea (reserved) */ 873c602fabSXin LI etheraddr lg_src; /* source id */ 883c602fabSXin LI byte lg_nextl2; /* next level 2 router (reserved) */ 893c602fabSXin LI byte lg_visits; /* visit count */ 903c602fabSXin LI byte lg_service; /* service class (reserved) */ 913c602fabSXin LI byte lg_pt; /* protocol type (reserved) */ 923c602fabSXin LI }; 933c602fabSXin LI 943c602fabSXin LI union routehdr 953c602fabSXin LI { 963c602fabSXin LI struct shorthdr rh_short; /* short route header */ 973c602fabSXin LI struct longhdr rh_long; /* long route header */ 983c602fabSXin LI }; 993c602fabSXin LI 1003c602fabSXin LI /* 1013c602fabSXin LI * Define the values of various fields in the protocol messages. 1023c602fabSXin LI * 1033c602fabSXin LI * 1. Data packet formats. 1043c602fabSXin LI */ 1053c602fabSXin LI #define RMF_MASK 7 /* mask for message type */ 1063c602fabSXin LI #define RMF_SHORT 2 /* short message format */ 1073c602fabSXin LI #define RMF_LONG 6 /* long message format */ 1083c602fabSXin LI #ifndef RMF_RQR 1093c602fabSXin LI #define RMF_RQR 010 /* request return to sender */ 1103c602fabSXin LI #define RMF_RTS 020 /* returning to sender */ 1113c602fabSXin LI #define RMF_IE 040 /* intra-ethernet packet */ 1123c602fabSXin LI #endif /* RMR_RQR */ 1133c602fabSXin LI #define RMF_FVER 0100 /* future version flag */ 1143c602fabSXin LI #define RMF_PAD 0200 /* pad field */ 1153c602fabSXin LI #define RMF_PADMASK 0177 /* pad field mask */ 1163c602fabSXin LI 1173c602fabSXin LI #define VIS_MASK 077 /* visit field mask */ 1183c602fabSXin LI 1193c602fabSXin LI /* 1203c602fabSXin LI * 2. Control packet formats. 1213c602fabSXin LI */ 1223c602fabSXin LI #define RMF_CTLMASK 017 /* mask for message type */ 1233c602fabSXin LI #define RMF_CTLMSG 01 /* control message indicator */ 1243c602fabSXin LI #define RMF_INIT 01 /* initialization message */ 1253c602fabSXin LI #define RMF_VER 03 /* verification message */ 1263c602fabSXin LI #define RMF_TEST 05 /* hello and test message */ 1273c602fabSXin LI #define RMF_L1ROUT 07 /* level 1 routing message */ 1283c602fabSXin LI #define RMF_L2ROUT 011 /* level 2 routing message */ 1293c602fabSXin LI #define RMF_RHELLO 013 /* router hello message */ 1303c602fabSXin LI #define RMF_EHELLO 015 /* endnode hello message */ 1313c602fabSXin LI 1323c602fabSXin LI #define TI_L2ROUT 01 /* level 2 router */ 1333c602fabSXin LI #define TI_L1ROUT 02 /* level 1 router */ 1343c602fabSXin LI #define TI_ENDNODE 03 /* endnode */ 1353c602fabSXin LI #define TI_VERIF 04 /* verification required */ 1363c602fabSXin LI #define TI_BLOCK 010 /* blocking requested */ 1373c602fabSXin LI 1383c602fabSXin LI #define VE_VERS 2 /* version number (2) */ 1393c602fabSXin LI #define VE_ECO 0 /* ECO number */ 1403c602fabSXin LI #define VE_UECO 0 /* user ECO number (0) */ 1413c602fabSXin LI 1423c602fabSXin LI #define P3_VERS 1 /* phase III version number (1) */ 1433c602fabSXin LI #define P3_ECO 3 /* ECO number (3) */ 1443c602fabSXin LI #define P3_UECO 0 /* user ECO number (0) */ 1453c602fabSXin LI 1463c602fabSXin LI #define II_L2ROUT 01 /* level 2 router */ 1473c602fabSXin LI #define II_L1ROUT 02 /* level 1 router */ 1483c602fabSXin LI #define II_ENDNODE 03 /* endnode */ 1493c602fabSXin LI #define II_VERIF 04 /* verification required */ 1503c602fabSXin LI #define II_NOMCAST 040 /* no multicast traffic accepted */ 1513c602fabSXin LI #define II_BLOCK 0100 /* blocking requested */ 1523c602fabSXin LI #define II_TYPEMASK 03 /* mask for node type */ 1533c602fabSXin LI 1543c602fabSXin LI #define TESTDATA 0252 /* test data bytes */ 1553c602fabSXin LI #define TESTLEN 1 /* length of transmitted test data */ 1563c602fabSXin LI 1573c602fabSXin LI /* 1583c602fabSXin LI * Define control message formats. 1593c602fabSXin LI */ 1603c602fabSXin LI struct initmsg /* initialization message */ 1613c602fabSXin LI { 1623c602fabSXin LI byte in_flags; /* route flags */ 1633c602fabSXin LI word in_src; /* source node address */ 1643c602fabSXin LI byte in_info; /* routing layer information */ 1653c602fabSXin LI word in_blksize; /* maximum data link block size */ 1663c602fabSXin LI byte in_vers; /* version number */ 1673c602fabSXin LI byte in_eco; /* ECO number */ 1683c602fabSXin LI byte in_ueco; /* user ECO number */ 1693c602fabSXin LI word in_hello; /* hello timer */ 1703c602fabSXin LI byte in_rsvd; /* reserved image field */ 1713c602fabSXin LI }; 1723c602fabSXin LI 1733c602fabSXin LI struct verifmsg /* verification message */ 1743c602fabSXin LI { 1753c602fabSXin LI byte ve_flags; /* route flags */ 1763c602fabSXin LI word ve_src; /* source node address */ 1773c602fabSXin LI byte ve_fcnval; /* function value image field */ 1783c602fabSXin LI }; 1793c602fabSXin LI 1803c602fabSXin LI struct testmsg /* hello and test message */ 1813c602fabSXin LI { 1823c602fabSXin LI byte te_flags; /* route flags */ 1833c602fabSXin LI word te_src; /* source node address */ 1843c602fabSXin LI byte te_data; /* test data image field */ 1853c602fabSXin LI }; 1863c602fabSXin LI 1873c602fabSXin LI struct l1rout /* level 1 routing message */ 1883c602fabSXin LI { 1893c602fabSXin LI byte r1_flags; /* route flags */ 1903c602fabSXin LI word r1_src; /* source node address */ 1913c602fabSXin LI byte r1_rsvd; /* reserved field */ 1923c602fabSXin LI }; 1933c602fabSXin LI 1943c602fabSXin LI struct l2rout /* level 2 routing message */ 1953c602fabSXin LI { 1963c602fabSXin LI byte r2_flags; /* route flags */ 1973c602fabSXin LI word r2_src; /* source node address */ 1983c602fabSXin LI byte r2_rsvd; /* reserved field */ 1993c602fabSXin LI }; 2003c602fabSXin LI 2013c602fabSXin LI struct rhellomsg /* router hello message */ 2023c602fabSXin LI { 2033c602fabSXin LI byte rh_flags; /* route flags */ 2043c602fabSXin LI byte rh_vers; /* version number */ 2053c602fabSXin LI byte rh_eco; /* ECO number */ 2063c602fabSXin LI byte rh_ueco; /* user ECO number */ 2073c602fabSXin LI etheraddr rh_src; /* source id */ 2083c602fabSXin LI byte rh_info; /* routing layer information */ 2093c602fabSXin LI word rh_blksize; /* maximum data link block size */ 2103c602fabSXin LI byte rh_priority; /* router's priority */ 2113c602fabSXin LI byte rh_area; /* reserved */ 2123c602fabSXin LI word rh_hello; /* hello timer */ 2133c602fabSXin LI byte rh_mpd; /* reserved */ 2143c602fabSXin LI }; 2153c602fabSXin LI 2163c602fabSXin LI struct ehellomsg /* endnode hello message */ 2173c602fabSXin LI { 2183c602fabSXin LI byte eh_flags; /* route flags */ 2193c602fabSXin LI byte eh_vers; /* version number */ 2203c602fabSXin LI byte eh_eco; /* ECO number */ 2213c602fabSXin LI byte eh_ueco; /* user ECO number */ 2223c602fabSXin LI etheraddr eh_src; /* source id */ 2233c602fabSXin LI byte eh_info; /* routing layer information */ 2243c602fabSXin LI word eh_blksize; /* maximum data link block size */ 2253c602fabSXin LI byte eh_area; /* area (reserved) */ 2263c602fabSXin LI byte eh_seed[8]; /* verification seed */ 2273c602fabSXin LI etheraddr eh_router; /* designated router */ 2283c602fabSXin LI word eh_hello; /* hello timer */ 2293c602fabSXin LI byte eh_mpd; /* (reserved) */ 2303c602fabSXin LI byte eh_data; /* test data image field */ 2313c602fabSXin LI }; 2323c602fabSXin LI 2333c602fabSXin LI union controlmsg 2343c602fabSXin LI { 2353c602fabSXin LI struct initmsg cm_init; /* initialization message */ 2363c602fabSXin LI struct verifmsg cm_ver; /* verification message */ 2373c602fabSXin LI struct testmsg cm_test; /* hello and test message */ 2383c602fabSXin LI struct l1rout cm_l1rou; /* level 1 routing message */ 2393c602fabSXin LI struct l2rout cm_l2rout; /* level 2 routing message */ 2403c602fabSXin LI struct rhellomsg cm_rhello; /* router hello message */ 2413c602fabSXin LI struct ehellomsg cm_ehello; /* endnode hello message */ 2423c602fabSXin LI }; 2433c602fabSXin LI 2443c602fabSXin LI /* Macros for decoding routing-info fields */ 2453c602fabSXin LI #define RI_COST(x) ((x)&0777) 2463c602fabSXin LI #define RI_HOPS(x) (((x)>>10)&037) 2478bdc5a62SPatrick Kelsey 2483c602fabSXin LI /* 2493c602fabSXin LI * NSP protocol fields and values. 2503c602fabSXin LI */ 2513c602fabSXin LI 2523c602fabSXin LI #define NSP_TYPEMASK 014 /* mask to isolate type code */ 2533c602fabSXin LI #define NSP_SUBMASK 0160 /* mask to isolate subtype code */ 2543c602fabSXin LI #define NSP_SUBSHFT 4 /* shift to move subtype code */ 2553c602fabSXin LI 2563c602fabSXin LI #define MFT_DATA 0 /* data message */ 2573c602fabSXin LI #define MFT_ACK 04 /* acknowledgement message */ 2583c602fabSXin LI #define MFT_CTL 010 /* control message */ 2593c602fabSXin LI 2603c602fabSXin LI #define MFS_ILS 020 /* data or I/LS indicator */ 2613c602fabSXin LI #define MFS_BOM 040 /* beginning of message (data) */ 2623c602fabSXin LI #define MFS_MOM 0 /* middle of message (data) */ 2633c602fabSXin LI #define MFS_EOM 0100 /* end of message (data) */ 2643c602fabSXin LI #define MFS_INT 040 /* interrupt message */ 2653c602fabSXin LI 2663c602fabSXin LI #define MFS_DACK 0 /* data acknowledgement */ 2673c602fabSXin LI #define MFS_IACK 020 /* I/LS acknowledgement */ 2683c602fabSXin LI #define MFS_CACK 040 /* connect acknowledgement */ 2693c602fabSXin LI 2703c602fabSXin LI #define MFS_NOP 0 /* no operation */ 2713c602fabSXin LI #define MFS_CI 020 /* connect initiate */ 2723c602fabSXin LI #define MFS_CC 040 /* connect confirm */ 2733c602fabSXin LI #define MFS_DI 060 /* disconnect initiate */ 2743c602fabSXin LI #define MFS_DC 0100 /* disconnect confirm */ 2753c602fabSXin LI #define MFS_RCI 0140 /* retransmitted connect initiate */ 2763c602fabSXin LI 2773c602fabSXin LI #define SGQ_ACK 0100000 /* ack */ 2783c602fabSXin LI #define SGQ_NAK 0110000 /* negative ack */ 2793c602fabSXin LI #define SGQ_OACK 0120000 /* other channel ack */ 2803c602fabSXin LI #define SGQ_ONAK 0130000 /* other channel negative ack */ 2813c602fabSXin LI #define SGQ_MASK 07777 /* mask to isolate seq # */ 2823c602fabSXin LI #define SGQ_OTHER 020000 /* other channel qualifier */ 2833c602fabSXin LI #define SGQ_DELAY 010000 /* ack delay flag */ 2843c602fabSXin LI 2853c602fabSXin LI #define SGQ_EOM 0100000 /* pseudo flag for end-of-message */ 2863c602fabSXin LI 2873c602fabSXin LI #define LSM_MASK 03 /* mask for modifier field */ 2883c602fabSXin LI #define LSM_NOCHANGE 0 /* no change */ 2893c602fabSXin LI #define LSM_DONOTSEND 1 /* do not send data */ 2903c602fabSXin LI #define LSM_SEND 2 /* send data */ 2913c602fabSXin LI 2923c602fabSXin LI #define LSI_MASK 014 /* mask for interpretation field */ 2933c602fabSXin LI #define LSI_DATA 0 /* data segment or message count */ 2943c602fabSXin LI #define LSI_INTR 4 /* interrupt request count */ 2953c602fabSXin LI #define LSI_INTM 0377 /* funny marker for int. message */ 2963c602fabSXin LI 2973c602fabSXin LI #define COS_MASK 014 /* mask for flow control field */ 2983c602fabSXin LI #define COS_NONE 0 /* no flow control */ 2993c602fabSXin LI #define COS_SEGMENT 04 /* segment flow control */ 3003c602fabSXin LI #define COS_MESSAGE 010 /* message flow control */ 3013c602fabSXin LI #define COS_DEFAULT 1 /* default value for field */ 3023c602fabSXin LI 3033c602fabSXin LI #define COI_MASK 3 /* mask for version field */ 3043c602fabSXin LI #define COI_32 0 /* version 3.2 */ 3053c602fabSXin LI #define COI_31 1 /* version 3.1 */ 3063c602fabSXin LI #define COI_40 2 /* version 4.0 */ 3073c602fabSXin LI #define COI_41 3 /* version 4.1 */ 3083c602fabSXin LI 3093c602fabSXin LI #define MNU_MASK 140 /* mask for session control version */ 3103c602fabSXin LI #define MNU_10 000 /* session V1.0 */ 3113c602fabSXin LI #define MNU_20 040 /* session V2.0 */ 3123c602fabSXin LI #define MNU_ACCESS 1 /* access control present */ 3133c602fabSXin LI #define MNU_USRDATA 2 /* user data field present */ 3143c602fabSXin LI #define MNU_INVKPROXY 4 /* invoke proxy field present */ 3153c602fabSXin LI #define MNU_UICPROXY 8 /* use uic-based proxy */ 3163c602fabSXin LI 3173c602fabSXin LI #define DC_NORESOURCES 1 /* no resource reason code */ 3183c602fabSXin LI #define DC_NOLINK 41 /* no link terminate reason code */ 3193c602fabSXin LI #define DC_COMPLETE 42 /* disconnect complete reason code */ 3203c602fabSXin LI 3213c602fabSXin LI #define DI_NOERROR 0 /* user disconnect */ 3223c602fabSXin LI #define DI_SHUT 3 /* node is shutting down */ 3233c602fabSXin LI #define DI_NOUSER 4 /* destination end user does not exist */ 3243c602fabSXin LI #define DI_INVDEST 5 /* invalid end user destination */ 3253c602fabSXin LI #define DI_REMRESRC 6 /* insufficient remote resources */ 3263c602fabSXin LI #define DI_TPA 8 /* third party abort */ 3273c602fabSXin LI #define DI_PROTOCOL 7 /* protocol error discovered */ 3283c602fabSXin LI #define DI_ABORT 9 /* user abort */ 3293c602fabSXin LI #define DI_LOCALRESRC 32 /* insufficient local resources */ 3303c602fabSXin LI #define DI_REMUSERRESRC 33 /* insufficient remote user resources */ 3313c602fabSXin LI #define DI_BADACCESS 34 /* bad access control information */ 3323c602fabSXin LI #define DI_BADACCNT 36 /* bad ACCOUNT information */ 3333c602fabSXin LI #define DI_CONNECTABORT 38 /* connect request cancelled */ 3343c602fabSXin LI #define DI_TIMEDOUT 38 /* remote node or user crashed */ 3353c602fabSXin LI #define DI_UNREACHABLE 39 /* local timers expired due to ... */ 3363c602fabSXin LI #define DI_BADIMAGE 43 /* bad image data in connect */ 3373c602fabSXin LI #define DI_SERVMISMATCH 54 /* cryptographic service mismatch */ 3383c602fabSXin LI 3393c602fabSXin LI #define UC_OBJREJECT 0 /* object rejected connect */ 3403c602fabSXin LI #define UC_USERDISCONNECT 0 /* user disconnect */ 3413c602fabSXin LI #define UC_RESOURCES 1 /* insufficient resources (local or remote) */ 3423c602fabSXin LI #define UC_NOSUCHNODE 2 /* unrecognized node name */ 3433c602fabSXin LI #define UC_REMOTESHUT 3 /* remote node shutting down */ 3443c602fabSXin LI #define UC_NOSUCHOBJ 4 /* unrecognized object */ 3453c602fabSXin LI #define UC_INVOBJFORMAT 5 /* invalid object name format */ 3463c602fabSXin LI #define UC_OBJTOOBUSY 6 /* object too busy */ 3473c602fabSXin LI #define UC_NETWORKABORT 8 /* network abort */ 3483c602fabSXin LI #define UC_USERABORT 9 /* user abort */ 3493c602fabSXin LI #define UC_INVNODEFORMAT 10 /* invalid node name format */ 3503c602fabSXin LI #define UC_LOCALSHUT 11 /* local node shutting down */ 3513c602fabSXin LI #define UC_ACCESSREJECT 34 /* invalid access control information */ 3523c602fabSXin LI #define UC_NORESPONSE 38 /* no response from object */ 3533c602fabSXin LI #define UC_UNREACHABLE 39 /* node unreachable */ 3543c602fabSXin LI 3553c602fabSXin LI /* 3563c602fabSXin LI * NSP message formats. 3573c602fabSXin LI */ 3583c602fabSXin LI struct nsphdr /* general nsp header */ 3593c602fabSXin LI { 3603c602fabSXin LI byte nh_flags; /* message flags */ 3613c602fabSXin LI word nh_dst; /* destination link address */ 3623c602fabSXin LI word nh_src; /* source link address */ 3633c602fabSXin LI }; 3643c602fabSXin LI 3653c602fabSXin LI struct seghdr /* data segment header */ 3663c602fabSXin LI { 3673c602fabSXin LI byte sh_flags; /* message flags */ 3683c602fabSXin LI word sh_dst; /* destination link address */ 3693c602fabSXin LI word sh_src; /* source link address */ 3703c602fabSXin LI word sh_seq[3]; /* sequence numbers */ 3713c602fabSXin LI }; 3723c602fabSXin LI 3733c602fabSXin LI struct minseghdr /* minimum data segment header */ 3743c602fabSXin LI { 3753c602fabSXin LI byte ms_flags; /* message flags */ 3763c602fabSXin LI word ms_dst; /* destination link address */ 3773c602fabSXin LI word ms_src; /* source link address */ 3783c602fabSXin LI word ms_seq; /* sequence number */ 3793c602fabSXin LI }; 3803c602fabSXin LI 3813c602fabSXin LI struct lsmsg /* link service message (after hdr) */ 3823c602fabSXin LI { 3833c602fabSXin LI byte ls_lsflags; /* link service flags */ 3843c602fabSXin LI byte ls_fcval; /* flow control value */ 3853c602fabSXin LI }; 3863c602fabSXin LI 3873c602fabSXin LI struct ackmsg /* acknowledgement message */ 3883c602fabSXin LI { 3893c602fabSXin LI byte ak_flags; /* message flags */ 3903c602fabSXin LI word ak_dst; /* destination link address */ 3913c602fabSXin LI word ak_src; /* source link address */ 3923c602fabSXin LI word ak_acknum[2]; /* acknowledgement numbers */ 3933c602fabSXin LI }; 3943c602fabSXin LI 3953c602fabSXin LI struct minackmsg /* minimum acknowledgement message */ 3963c602fabSXin LI { 3973c602fabSXin LI byte mk_flags; /* message flags */ 3983c602fabSXin LI word mk_dst; /* destination link address */ 3993c602fabSXin LI word mk_src; /* source link address */ 4003c602fabSXin LI word mk_acknum; /* acknowledgement number */ 4013c602fabSXin LI }; 4023c602fabSXin LI 4033c602fabSXin LI struct ciackmsg /* connect acknowledgement message */ 4043c602fabSXin LI { 4053c602fabSXin LI byte ck_flags; /* message flags */ 4063c602fabSXin LI word ck_dst; /* destination link address */ 4073c602fabSXin LI }; 4083c602fabSXin LI 4093c602fabSXin LI struct cimsg /* connect initiate message */ 4103c602fabSXin LI { 4113c602fabSXin LI byte ci_flags; /* message flags */ 4123c602fabSXin LI word ci_dst; /* destination link address (0) */ 4133c602fabSXin LI word ci_src; /* source link address */ 4143c602fabSXin LI byte ci_services; /* requested services */ 4153c602fabSXin LI byte ci_info; /* information */ 4163c602fabSXin LI word ci_segsize; /* maximum segment size */ 4173c602fabSXin LI }; 4183c602fabSXin LI 4193c602fabSXin LI struct ccmsg /* connect confirm message */ 4203c602fabSXin LI { 4213c602fabSXin LI byte cc_flags; /* message flags */ 4223c602fabSXin LI word cc_dst; /* destination link address */ 4233c602fabSXin LI word cc_src; /* source link address */ 4243c602fabSXin LI byte cc_services; /* requested services */ 4253c602fabSXin LI byte cc_info; /* information */ 4263c602fabSXin LI word cc_segsize; /* maximum segment size */ 4273c602fabSXin LI byte cc_optlen; /* optional data length */ 4283c602fabSXin LI }; 4293c602fabSXin LI 4303c602fabSXin LI struct cnmsg /* generic connect message */ 4313c602fabSXin LI { 4323c602fabSXin LI byte cn_flags; /* message flags */ 4333c602fabSXin LI word cn_dst; /* destination link address */ 4343c602fabSXin LI word cn_src; /* source link address */ 4353c602fabSXin LI byte cn_services; /* requested services */ 4363c602fabSXin LI byte cn_info; /* information */ 4373c602fabSXin LI word cn_segsize; /* maximum segment size */ 4383c602fabSXin LI }; 4393c602fabSXin LI 4403c602fabSXin LI struct dimsg /* disconnect initiate message */ 4413c602fabSXin LI { 4423c602fabSXin LI byte di_flags; /* message flags */ 4433c602fabSXin LI word di_dst; /* destination link address */ 4443c602fabSXin LI word di_src; /* source link address */ 4453c602fabSXin LI word di_reason; /* reason code */ 4463c602fabSXin LI byte di_optlen; /* optional data length */ 4473c602fabSXin LI }; 4483c602fabSXin LI 4493c602fabSXin LI struct dcmsg /* disconnect confirm message */ 4503c602fabSXin LI { 4513c602fabSXin LI byte dc_flags; /* message flags */ 4523c602fabSXin LI word dc_dst; /* destination link address */ 4533c602fabSXin LI word dc_src; /* source link address */ 4543c602fabSXin LI word dc_reason; /* reason code */ 4553c602fabSXin LI }; 4563c602fabSXin LI 4574edb46e9SPaul Traina /* Forwards */ 4583c602fabSXin LI static int print_decnet_ctlmsg(netdissect_options *, const union routehdr *, u_int, u_int); 459*ee67461eSJoseph Mingrone static void print_t_info(netdissect_options *, u_int); 460*ee67461eSJoseph Mingrone static void print_l1_routes(netdissect_options *, const u_char *, u_int); 461*ee67461eSJoseph Mingrone static void print_l2_routes(netdissect_options *, const u_char *, u_int); 462*ee67461eSJoseph Mingrone static void print_i_info(netdissect_options *, u_int); 463*ee67461eSJoseph Mingrone static void print_elist(const u_char *, u_int); 4643c602fabSXin LI static int print_nsp(netdissect_options *, const u_char *, u_int); 465*ee67461eSJoseph Mingrone static void print_reason(netdissect_options *, u_int); 4664edb46e9SPaul Traina 4674edb46e9SPaul Traina void 4683c602fabSXin LI decnet_print(netdissect_options *ndo, 469*ee67461eSJoseph Mingrone const u_char *ap, u_int length, 470*ee67461eSJoseph Mingrone u_int caplen) 4714edb46e9SPaul Traina { 472*ee67461eSJoseph Mingrone const union routehdr *rhp; 473*ee67461eSJoseph Mingrone u_int mflags; 474*ee67461eSJoseph Mingrone uint16_t dst, src; 475*ee67461eSJoseph Mingrone u_int hops; 476f4d0c64aSSam Leffler u_int nsplen, pktlen; 4774edb46e9SPaul Traina const u_char *nspp; 4784edb46e9SPaul Traina 479*ee67461eSJoseph Mingrone ndo->ndo_protocol = "decnet"; 4804edb46e9SPaul Traina if (length < sizeof(struct shorthdr)) { 481*ee67461eSJoseph Mingrone ND_PRINT(" (length %u < %zu)", length, sizeof(struct shorthdr)); 482*ee67461eSJoseph Mingrone goto invalid; 4834edb46e9SPaul Traina } 4844edb46e9SPaul Traina 485*ee67461eSJoseph Mingrone pktlen = GET_LE_U_2(ap); 486f4d0c64aSSam Leffler if (pktlen < sizeof(struct shorthdr)) { 487*ee67461eSJoseph Mingrone ND_PRINT(" (pktlen %u < %zu)", pktlen, sizeof(struct shorthdr)); 488*ee67461eSJoseph Mingrone goto invalid; 489f4d0c64aSSam Leffler } 490f4d0c64aSSam Leffler if (pktlen > length) { 491*ee67461eSJoseph Mingrone ND_PRINT(" (pktlen %u > %u)", pktlen, length); 492*ee67461eSJoseph Mingrone goto invalid; 493f4d0c64aSSam Leffler } 494f4d0c64aSSam Leffler length = pktlen; 4954edb46e9SPaul Traina 496*ee67461eSJoseph Mingrone rhp = (const union routehdr *)(ap + sizeof(short)); 497*ee67461eSJoseph Mingrone mflags = GET_U_1(rhp->rh_short.sh_flags); 4984edb46e9SPaul Traina 4994edb46e9SPaul Traina if (mflags & RMF_PAD) { 5004edb46e9SPaul Traina /* pad bytes of some sort in front of message */ 5014edb46e9SPaul Traina u_int padlen = mflags & RMF_PADMASK; 5023c602fabSXin LI if (ndo->ndo_vflag) 503*ee67461eSJoseph Mingrone ND_PRINT("[pad:%u] ", padlen); 504f4d0c64aSSam Leffler if (length < padlen + 2) { 505*ee67461eSJoseph Mingrone ND_PRINT(" (length %u < %u)", length, padlen + 2); 506*ee67461eSJoseph Mingrone goto invalid; 507f4d0c64aSSam Leffler } 508*ee67461eSJoseph Mingrone ND_TCHECK_LEN(ap + sizeof(short), padlen); 5094edb46e9SPaul Traina ap += padlen; 5104edb46e9SPaul Traina length -= padlen; 5114edb46e9SPaul Traina caplen -= padlen; 512*ee67461eSJoseph Mingrone rhp = (const union routehdr *)(ap + sizeof(short)); 513*ee67461eSJoseph Mingrone mflags = GET_U_1(rhp->rh_short.sh_flags); 5144edb46e9SPaul Traina } 5154edb46e9SPaul Traina 5164edb46e9SPaul Traina if (mflags & RMF_FVER) { 517*ee67461eSJoseph Mingrone ND_PRINT("future-version-decnet"); 518*ee67461eSJoseph Mingrone ND_DEFAULTPRINT(ap, ND_MIN(length, caplen)); 5194edb46e9SPaul Traina return; 5204edb46e9SPaul Traina } 5214edb46e9SPaul Traina 5224edb46e9SPaul Traina /* is it a control message? */ 5234edb46e9SPaul Traina if (mflags & RMF_CTLMSG) { 5243c602fabSXin LI if (!print_decnet_ctlmsg(ndo, rhp, length, caplen)) 525*ee67461eSJoseph Mingrone goto invalid; 5264edb46e9SPaul Traina return; 5274edb46e9SPaul Traina } 5284edb46e9SPaul Traina 5294edb46e9SPaul Traina switch (mflags & RMF_MASK) { 5304edb46e9SPaul Traina case RMF_LONG: 531f4d0c64aSSam Leffler if (length < sizeof(struct longhdr)) { 532*ee67461eSJoseph Mingrone ND_PRINT(" (length %u < %zu)", length, sizeof(struct longhdr)); 533*ee67461eSJoseph Mingrone goto invalid; 534f4d0c64aSSam Leffler } 535*ee67461eSJoseph Mingrone ND_TCHECK_SIZE(&rhp->rh_long); 5364edb46e9SPaul Traina dst = 537*ee67461eSJoseph Mingrone GET_LE_U_2(rhp->rh_long.lg_dst.dne_remote.dne_nodeaddr); 5384edb46e9SPaul Traina src = 539*ee67461eSJoseph Mingrone GET_LE_U_2(rhp->rh_long.lg_src.dne_remote.dne_nodeaddr); 540*ee67461eSJoseph Mingrone hops = GET_U_1(rhp->rh_long.lg_visits); 541*ee67461eSJoseph Mingrone nspp = ap + sizeof(short) + sizeof(struct longhdr); 542f4d0c64aSSam Leffler nsplen = length - sizeof(struct longhdr); 5434edb46e9SPaul Traina break; 5444edb46e9SPaul Traina case RMF_SHORT: 545*ee67461eSJoseph Mingrone dst = GET_LE_U_2(rhp->rh_short.sh_dst); 546*ee67461eSJoseph Mingrone src = GET_LE_U_2(rhp->rh_short.sh_src); 547*ee67461eSJoseph Mingrone hops = (GET_U_1(rhp->rh_short.sh_visits) & VIS_MASK)+1; 548*ee67461eSJoseph Mingrone nspp = ap + sizeof(short) + sizeof(struct shorthdr); 549f4d0c64aSSam Leffler nsplen = length - sizeof(struct shorthdr); 5504edb46e9SPaul Traina break; 5514edb46e9SPaul Traina default: 552*ee67461eSJoseph Mingrone ND_PRINT("unknown message flags under mask"); 553*ee67461eSJoseph Mingrone ND_DEFAULTPRINT((const u_char *)ap, ND_MIN(length, caplen)); 5544edb46e9SPaul Traina return; 5554edb46e9SPaul Traina } 5564edb46e9SPaul Traina 557*ee67461eSJoseph Mingrone ND_PRINT("%s > %s %u ", 558*ee67461eSJoseph Mingrone dnaddr_string(ndo, src), dnaddr_string(ndo, dst), pktlen); 5593c602fabSXin LI if (ndo->ndo_vflag) { 5604edb46e9SPaul Traina if (mflags & RMF_RQR) 561*ee67461eSJoseph Mingrone ND_PRINT("RQR "); 5624edb46e9SPaul Traina if (mflags & RMF_RTS) 563*ee67461eSJoseph Mingrone ND_PRINT("RTS "); 5644edb46e9SPaul Traina if (mflags & RMF_IE) 565*ee67461eSJoseph Mingrone ND_PRINT("IE "); 566*ee67461eSJoseph Mingrone ND_PRINT("%u hops ", hops); 5674edb46e9SPaul Traina } 5684edb46e9SPaul Traina 5693c602fabSXin LI if (!print_nsp(ndo, nspp, nsplen)) 570*ee67461eSJoseph Mingrone goto invalid; 571f4d0c64aSSam Leffler return; 572f4d0c64aSSam Leffler 573*ee67461eSJoseph Mingrone invalid: 574*ee67461eSJoseph Mingrone nd_print_invalid(ndo); 5754edb46e9SPaul Traina } 5764edb46e9SPaul Traina 577f4d0c64aSSam Leffler static int 5783c602fabSXin LI print_decnet_ctlmsg(netdissect_options *ndo, 579*ee67461eSJoseph Mingrone const union routehdr *rhp, u_int length, 580f4d0c64aSSam Leffler u_int caplen) 5814edb46e9SPaul Traina { 5820bff6a5aSEd Maste /* Our caller has already checked for mflags */ 583*ee67461eSJoseph Mingrone u_int mflags = GET_U_1(rhp->rh_short.sh_flags); 584*ee67461eSJoseph Mingrone const union controlmsg *cmp = (const union controlmsg *)rhp; 585*ee67461eSJoseph Mingrone uint16_t src, dst; 586*ee67461eSJoseph Mingrone u_int info, blksize, eco, ueco, hello, other, vers; 587*ee67461eSJoseph Mingrone u_int priority; 588*ee67461eSJoseph Mingrone const u_char *rhpx = (const u_char *)rhp; 5894edb46e9SPaul Traina 5904edb46e9SPaul Traina switch (mflags & RMF_CTLMASK) { 5914edb46e9SPaul Traina case RMF_INIT: 592*ee67461eSJoseph Mingrone ND_PRINT("init "); 593f4d0c64aSSam Leffler if (length < sizeof(struct initmsg)) 594*ee67461eSJoseph Mingrone goto invalid; 595*ee67461eSJoseph Mingrone ND_TCHECK_SIZE(&cmp->cm_init); 596*ee67461eSJoseph Mingrone src = GET_LE_U_2(cmp->cm_init.in_src); 597*ee67461eSJoseph Mingrone info = GET_U_1(cmp->cm_init.in_info); 598*ee67461eSJoseph Mingrone blksize = GET_LE_U_2(cmp->cm_init.in_blksize); 599*ee67461eSJoseph Mingrone vers = GET_U_1(cmp->cm_init.in_vers); 600*ee67461eSJoseph Mingrone eco = GET_U_1(cmp->cm_init.in_eco); 601*ee67461eSJoseph Mingrone ueco = GET_U_1(cmp->cm_init.in_ueco); 602*ee67461eSJoseph Mingrone hello = GET_LE_U_2(cmp->cm_init.in_hello); 6033c602fabSXin LI print_t_info(ndo, info); 604*ee67461eSJoseph Mingrone ND_PRINT("src %sblksize %u vers %u eco %u ueco %u hello %u", 6053c602fabSXin LI dnaddr_string(ndo, src), blksize, vers, eco, ueco, 606*ee67461eSJoseph Mingrone hello); 6074edb46e9SPaul Traina break; 6084edb46e9SPaul Traina case RMF_VER: 609*ee67461eSJoseph Mingrone ND_PRINT("verification "); 610f4d0c64aSSam Leffler if (length < sizeof(struct verifmsg)) 611*ee67461eSJoseph Mingrone goto invalid; 612*ee67461eSJoseph Mingrone src = GET_LE_U_2(cmp->cm_ver.ve_src); 613*ee67461eSJoseph Mingrone other = GET_U_1(cmp->cm_ver.ve_fcnval); 614*ee67461eSJoseph Mingrone ND_PRINT("src %s fcnval %o", dnaddr_string(ndo, src), other); 6154edb46e9SPaul Traina break; 6164edb46e9SPaul Traina case RMF_TEST: 617*ee67461eSJoseph Mingrone ND_PRINT("test "); 618f4d0c64aSSam Leffler if (length < sizeof(struct testmsg)) 619*ee67461eSJoseph Mingrone goto invalid; 620*ee67461eSJoseph Mingrone src = GET_LE_U_2(cmp->cm_test.te_src); 621*ee67461eSJoseph Mingrone other = GET_U_1(cmp->cm_test.te_data); 622*ee67461eSJoseph Mingrone ND_PRINT("src %s data %o", dnaddr_string(ndo, src), other); 6234edb46e9SPaul Traina break; 6244edb46e9SPaul Traina case RMF_L1ROUT: 625*ee67461eSJoseph Mingrone ND_PRINT("lev-1-routing "); 626f4d0c64aSSam Leffler if (length < sizeof(struct l1rout)) 627*ee67461eSJoseph Mingrone goto invalid; 628*ee67461eSJoseph Mingrone ND_TCHECK_SIZE(&cmp->cm_l1rou); 629*ee67461eSJoseph Mingrone src = GET_LE_U_2(cmp->cm_l1rou.r1_src); 630*ee67461eSJoseph Mingrone ND_PRINT("src %s ", dnaddr_string(ndo, src)); 631*ee67461eSJoseph Mingrone print_l1_routes(ndo, &(rhpx[sizeof(struct l1rout)]), 6324edb46e9SPaul Traina length - sizeof(struct l1rout)); 6334edb46e9SPaul Traina break; 6344edb46e9SPaul Traina case RMF_L2ROUT: 635*ee67461eSJoseph Mingrone ND_PRINT("lev-2-routing "); 636f4d0c64aSSam Leffler if (length < sizeof(struct l2rout)) 637*ee67461eSJoseph Mingrone goto invalid; 638*ee67461eSJoseph Mingrone ND_TCHECK_SIZE(&cmp->cm_l2rout); 639*ee67461eSJoseph Mingrone src = GET_LE_U_2(cmp->cm_l2rout.r2_src); 640*ee67461eSJoseph Mingrone ND_PRINT("src %s ", dnaddr_string(ndo, src)); 641*ee67461eSJoseph Mingrone print_l2_routes(ndo, &(rhpx[sizeof(struct l2rout)]), 6424edb46e9SPaul Traina length - sizeof(struct l2rout)); 6434edb46e9SPaul Traina break; 6444edb46e9SPaul Traina case RMF_RHELLO: 645*ee67461eSJoseph Mingrone ND_PRINT("router-hello "); 646f4d0c64aSSam Leffler if (length < sizeof(struct rhellomsg)) 647*ee67461eSJoseph Mingrone goto invalid; 648*ee67461eSJoseph Mingrone ND_TCHECK_SIZE(&cmp->cm_rhello); 649*ee67461eSJoseph Mingrone vers = GET_U_1(cmp->cm_rhello.rh_vers); 650*ee67461eSJoseph Mingrone eco = GET_U_1(cmp->cm_rhello.rh_eco); 651*ee67461eSJoseph Mingrone ueco = GET_U_1(cmp->cm_rhello.rh_ueco); 652*ee67461eSJoseph Mingrone src = 653*ee67461eSJoseph Mingrone GET_LE_U_2(cmp->cm_rhello.rh_src.dne_remote.dne_nodeaddr); 654*ee67461eSJoseph Mingrone info = GET_U_1(cmp->cm_rhello.rh_info); 655*ee67461eSJoseph Mingrone blksize = GET_LE_U_2(cmp->cm_rhello.rh_blksize); 656*ee67461eSJoseph Mingrone priority = GET_U_1(cmp->cm_rhello.rh_priority); 657*ee67461eSJoseph Mingrone hello = GET_LE_U_2(cmp->cm_rhello.rh_hello); 6583c602fabSXin LI print_i_info(ndo, info); 659*ee67461eSJoseph Mingrone ND_PRINT("vers %u eco %u ueco %u src %s blksize %u pri %u hello %u", 6603c602fabSXin LI vers, eco, ueco, dnaddr_string(ndo, src), 661*ee67461eSJoseph Mingrone blksize, priority, hello); 662*ee67461eSJoseph Mingrone print_elist(&(rhpx[sizeof(struct rhellomsg)]), 6634edb46e9SPaul Traina length - sizeof(struct rhellomsg)); 6644edb46e9SPaul Traina break; 6654edb46e9SPaul Traina case RMF_EHELLO: 666*ee67461eSJoseph Mingrone ND_PRINT("endnode-hello "); 667f4d0c64aSSam Leffler if (length < sizeof(struct ehellomsg)) 668*ee67461eSJoseph Mingrone goto invalid; 669*ee67461eSJoseph Mingrone vers = GET_U_1(cmp->cm_ehello.eh_vers); 670*ee67461eSJoseph Mingrone eco = GET_U_1(cmp->cm_ehello.eh_eco); 671*ee67461eSJoseph Mingrone ueco = GET_U_1(cmp->cm_ehello.eh_ueco); 672*ee67461eSJoseph Mingrone src = 673*ee67461eSJoseph Mingrone GET_LE_U_2(cmp->cm_ehello.eh_src.dne_remote.dne_nodeaddr); 674*ee67461eSJoseph Mingrone info = GET_U_1(cmp->cm_ehello.eh_info); 675*ee67461eSJoseph Mingrone blksize = GET_LE_U_2(cmp->cm_ehello.eh_blksize); 6764edb46e9SPaul Traina /*seed*/ 677*ee67461eSJoseph Mingrone dst = 678*ee67461eSJoseph Mingrone GET_LE_U_2(cmp->cm_ehello.eh_router.dne_remote.dne_nodeaddr); 679*ee67461eSJoseph Mingrone hello = GET_LE_U_2(cmp->cm_ehello.eh_hello); 680*ee67461eSJoseph Mingrone other = GET_U_1(cmp->cm_ehello.eh_data); 6813c602fabSXin LI print_i_info(ndo, info); 682*ee67461eSJoseph Mingrone ND_PRINT("vers %u eco %u ueco %u src %s blksize %u rtr %s hello %u data %o", 6833c602fabSXin LI vers, eco, ueco, dnaddr_string(ndo, src), 684*ee67461eSJoseph Mingrone blksize, dnaddr_string(ndo, dst), hello, other); 6854edb46e9SPaul Traina break; 6864edb46e9SPaul Traina 6874edb46e9SPaul Traina default: 688*ee67461eSJoseph Mingrone ND_PRINT("unknown control message"); 689*ee67461eSJoseph Mingrone ND_DEFAULTPRINT((const u_char *)rhp, ND_MIN(length, caplen)); 6904edb46e9SPaul Traina break; 6914edb46e9SPaul Traina } 692*ee67461eSJoseph Mingrone return (1); 693f4d0c64aSSam Leffler 694*ee67461eSJoseph Mingrone invalid: 695f4d0c64aSSam Leffler return (0); 6964edb46e9SPaul Traina } 6974edb46e9SPaul Traina 6984edb46e9SPaul Traina static void 6993c602fabSXin LI print_t_info(netdissect_options *ndo, 700*ee67461eSJoseph Mingrone u_int info) 7014edb46e9SPaul Traina { 702*ee67461eSJoseph Mingrone u_int ntype = info & 3; 7034edb46e9SPaul Traina switch (ntype) { 704*ee67461eSJoseph Mingrone case 0: ND_PRINT("reserved-ntype? "); break; 705*ee67461eSJoseph Mingrone case TI_L2ROUT: ND_PRINT("l2rout "); break; 706*ee67461eSJoseph Mingrone case TI_L1ROUT: ND_PRINT("l1rout "); break; 707*ee67461eSJoseph Mingrone case TI_ENDNODE: ND_PRINT("endnode "); break; 7084edb46e9SPaul Traina } 7094edb46e9SPaul Traina if (info & TI_VERIF) 710*ee67461eSJoseph Mingrone ND_PRINT("verif "); 7114edb46e9SPaul Traina if (info & TI_BLOCK) 712*ee67461eSJoseph Mingrone ND_PRINT("blo "); 7134edb46e9SPaul Traina } 7144edb46e9SPaul Traina 715*ee67461eSJoseph Mingrone static void 7163c602fabSXin LI print_l1_routes(netdissect_options *ndo, 717*ee67461eSJoseph Mingrone const u_char *rp, u_int len) 7184edb46e9SPaul Traina { 719*ee67461eSJoseph Mingrone u_int count; 720*ee67461eSJoseph Mingrone u_int id; 721*ee67461eSJoseph Mingrone u_int info; 7224edb46e9SPaul Traina 7234edb46e9SPaul Traina /* The last short is a checksum */ 7244edb46e9SPaul Traina while (len > (3 * sizeof(short))) { 725*ee67461eSJoseph Mingrone ND_TCHECK_LEN(rp, 3 * sizeof(short)); 726*ee67461eSJoseph Mingrone count = GET_LE_U_2(rp); 7274edb46e9SPaul Traina if (count > 1024) 728*ee67461eSJoseph Mingrone return; /* seems to be bogus from here on */ 7294edb46e9SPaul Traina rp += sizeof(short); 7304edb46e9SPaul Traina len -= sizeof(short); 731*ee67461eSJoseph Mingrone id = GET_LE_U_2(rp); 7324edb46e9SPaul Traina rp += sizeof(short); 7334edb46e9SPaul Traina len -= sizeof(short); 734*ee67461eSJoseph Mingrone info = GET_LE_U_2(rp); 7354edb46e9SPaul Traina rp += sizeof(short); 7364edb46e9SPaul Traina len -= sizeof(short); 737*ee67461eSJoseph Mingrone ND_PRINT("{ids %u-%u cost %u hops %u} ", id, id + count, 738*ee67461eSJoseph Mingrone RI_COST(info), RI_HOPS(info)); 7394edb46e9SPaul Traina } 7404edb46e9SPaul Traina } 7414edb46e9SPaul Traina 742*ee67461eSJoseph Mingrone static void 7433c602fabSXin LI print_l2_routes(netdissect_options *ndo, 744*ee67461eSJoseph Mingrone const u_char *rp, u_int len) 7454edb46e9SPaul Traina { 746*ee67461eSJoseph Mingrone u_int count; 747*ee67461eSJoseph Mingrone u_int area; 748*ee67461eSJoseph Mingrone u_int info; 7494edb46e9SPaul Traina 7504edb46e9SPaul Traina /* The last short is a checksum */ 7514edb46e9SPaul Traina while (len > (3 * sizeof(short))) { 752*ee67461eSJoseph Mingrone ND_TCHECK_LEN(rp, 3 * sizeof(short)); 753*ee67461eSJoseph Mingrone count = GET_LE_U_2(rp); 7544edb46e9SPaul Traina if (count > 1024) 755*ee67461eSJoseph Mingrone return; /* seems to be bogus from here on */ 7564edb46e9SPaul Traina rp += sizeof(short); 7574edb46e9SPaul Traina len -= sizeof(short); 758*ee67461eSJoseph Mingrone area = GET_LE_U_2(rp); 7594edb46e9SPaul Traina rp += sizeof(short); 7604edb46e9SPaul Traina len -= sizeof(short); 761*ee67461eSJoseph Mingrone info = GET_LE_U_2(rp); 7624edb46e9SPaul Traina rp += sizeof(short); 7634edb46e9SPaul Traina len -= sizeof(short); 764*ee67461eSJoseph Mingrone ND_PRINT("{areas %u-%u cost %u hops %u} ", area, area + count, 765*ee67461eSJoseph Mingrone RI_COST(info), RI_HOPS(info)); 7664edb46e9SPaul Traina } 7674edb46e9SPaul Traina } 7684edb46e9SPaul Traina 7694edb46e9SPaul Traina static void 7703c602fabSXin LI print_i_info(netdissect_options *ndo, 771*ee67461eSJoseph Mingrone u_int info) 7724edb46e9SPaul Traina { 773*ee67461eSJoseph Mingrone u_int ntype = info & II_TYPEMASK; 7744edb46e9SPaul Traina switch (ntype) { 775*ee67461eSJoseph Mingrone case 0: ND_PRINT("reserved-ntype? "); break; 776*ee67461eSJoseph Mingrone case II_L2ROUT: ND_PRINT("l2rout "); break; 777*ee67461eSJoseph Mingrone case II_L1ROUT: ND_PRINT("l1rout "); break; 778*ee67461eSJoseph Mingrone case II_ENDNODE: ND_PRINT("endnode "); break; 7794edb46e9SPaul Traina } 7804edb46e9SPaul Traina if (info & II_VERIF) 781*ee67461eSJoseph Mingrone ND_PRINT("verif "); 7824edb46e9SPaul Traina if (info & II_NOMCAST) 783*ee67461eSJoseph Mingrone ND_PRINT("nomcast "); 7844edb46e9SPaul Traina if (info & II_BLOCK) 785*ee67461eSJoseph Mingrone ND_PRINT("blo "); 7864edb46e9SPaul Traina } 7874edb46e9SPaul Traina 788*ee67461eSJoseph Mingrone static void 789*ee67461eSJoseph Mingrone print_elist(const u_char *elp _U_, u_int len _U_) 7904edb46e9SPaul Traina { 7914edb46e9SPaul Traina /* Not enough examples available for me to debug this */ 7924edb46e9SPaul Traina } 7934edb46e9SPaul Traina 794f4d0c64aSSam Leffler static int 7953c602fabSXin LI print_nsp(netdissect_options *ndo, 7963c602fabSXin LI const u_char *nspp, u_int nsplen) 7974edb46e9SPaul Traina { 7983340d773SGleb Smirnoff const struct nsphdr *nsphp = (const struct nsphdr *)nspp; 799*ee67461eSJoseph Mingrone u_int dst, src, flags; 8004edb46e9SPaul Traina 801*ee67461eSJoseph Mingrone if (nsplen < sizeof(struct nsphdr)) { 802*ee67461eSJoseph Mingrone ND_PRINT(" (nsplen %u < %zu)", nsplen, sizeof(struct nsphdr)); 803*ee67461eSJoseph Mingrone goto invalid; 804*ee67461eSJoseph Mingrone } 805*ee67461eSJoseph Mingrone flags = GET_U_1(nsphp->nh_flags); 806*ee67461eSJoseph Mingrone dst = GET_LE_U_2(nsphp->nh_dst); 807*ee67461eSJoseph Mingrone src = GET_LE_U_2(nsphp->nh_src); 8084edb46e9SPaul Traina 8094edb46e9SPaul Traina switch (flags & NSP_TYPEMASK) { 8104edb46e9SPaul Traina case MFT_DATA: 8114edb46e9SPaul Traina switch (flags & NSP_SUBMASK) { 8124edb46e9SPaul Traina case MFS_BOM: 8134edb46e9SPaul Traina case MFS_MOM: 8144edb46e9SPaul Traina case MFS_EOM: 8154edb46e9SPaul Traina case MFS_BOM+MFS_EOM: 816*ee67461eSJoseph Mingrone ND_PRINT("data %u>%u ", src, dst); 8174edb46e9SPaul Traina { 8183340d773SGleb Smirnoff const struct seghdr *shp = (const struct seghdr *)nspp; 819*ee67461eSJoseph Mingrone u_int ack; 8204edb46e9SPaul Traina u_int data_off = sizeof(struct minseghdr); 8214edb46e9SPaul Traina 822f4d0c64aSSam Leffler if (nsplen < data_off) 823*ee67461eSJoseph Mingrone goto invalid; 824*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[0]); 8254edb46e9SPaul Traina if (ack & SGQ_ACK) { /* acknum field */ 8264edb46e9SPaul Traina if ((ack & SGQ_NAK) == SGQ_NAK) 827*ee67461eSJoseph Mingrone ND_PRINT("nak %u ", ack & SGQ_MASK); 8284edb46e9SPaul Traina else 829*ee67461eSJoseph Mingrone ND_PRINT("ack %u ", ack & SGQ_MASK); 8304edb46e9SPaul Traina data_off += sizeof(short); 831f4d0c64aSSam Leffler if (nsplen < data_off) 832*ee67461eSJoseph Mingrone goto invalid; 833*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[1]); 8344edb46e9SPaul Traina if (ack & SGQ_OACK) { /* ackoth field */ 8354edb46e9SPaul Traina if ((ack & SGQ_ONAK) == SGQ_ONAK) 836*ee67461eSJoseph Mingrone ND_PRINT("onak %u ", ack & SGQ_MASK); 8374edb46e9SPaul Traina else 838*ee67461eSJoseph Mingrone ND_PRINT("oack %u ", ack & SGQ_MASK); 8394edb46e9SPaul Traina data_off += sizeof(short); 840f4d0c64aSSam Leffler if (nsplen < data_off) 841*ee67461eSJoseph Mingrone goto invalid; 842*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[2]); 8434edb46e9SPaul Traina } 8444edb46e9SPaul Traina } 845*ee67461eSJoseph Mingrone ND_PRINT("seg %u ", ack & SGQ_MASK); 8464edb46e9SPaul Traina } 8474edb46e9SPaul Traina break; 8484edb46e9SPaul Traina case MFS_ILS+MFS_INT: 849*ee67461eSJoseph Mingrone ND_PRINT("intr "); 8504edb46e9SPaul Traina { 8513340d773SGleb Smirnoff const struct seghdr *shp = (const struct seghdr *)nspp; 852*ee67461eSJoseph Mingrone u_int ack; 8534edb46e9SPaul Traina u_int data_off = sizeof(struct minseghdr); 8544edb46e9SPaul Traina 855f4d0c64aSSam Leffler if (nsplen < data_off) 856*ee67461eSJoseph Mingrone goto invalid; 857*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[0]); 8584edb46e9SPaul Traina if (ack & SGQ_ACK) { /* acknum field */ 8594edb46e9SPaul Traina if ((ack & SGQ_NAK) == SGQ_NAK) 860*ee67461eSJoseph Mingrone ND_PRINT("nak %u ", ack & SGQ_MASK); 8614edb46e9SPaul Traina else 862*ee67461eSJoseph Mingrone ND_PRINT("ack %u ", ack & SGQ_MASK); 8634edb46e9SPaul Traina data_off += sizeof(short); 864f4d0c64aSSam Leffler if (nsplen < data_off) 865*ee67461eSJoseph Mingrone goto invalid; 866*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[1]); 8674edb46e9SPaul Traina if (ack & SGQ_OACK) { /* ackdat field */ 8684edb46e9SPaul Traina if ((ack & SGQ_ONAK) == SGQ_ONAK) 869*ee67461eSJoseph Mingrone ND_PRINT("nakdat %u ", ack & SGQ_MASK); 8704edb46e9SPaul Traina else 871*ee67461eSJoseph Mingrone ND_PRINT("ackdat %u ", ack & SGQ_MASK); 8724edb46e9SPaul Traina data_off += sizeof(short); 873f4d0c64aSSam Leffler if (nsplen < data_off) 874*ee67461eSJoseph Mingrone goto invalid; 875*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[2]); 8764edb46e9SPaul Traina } 8774edb46e9SPaul Traina } 878*ee67461eSJoseph Mingrone ND_PRINT("seg %u ", ack & SGQ_MASK); 8794edb46e9SPaul Traina } 8804edb46e9SPaul Traina break; 8814edb46e9SPaul Traina case MFS_ILS: 882*ee67461eSJoseph Mingrone ND_PRINT("link-service %u>%u ", src, dst); 8834edb46e9SPaul Traina { 8843340d773SGleb Smirnoff const struct seghdr *shp = (const struct seghdr *)nspp; 8853340d773SGleb Smirnoff const struct lsmsg *lsmp = 886*ee67461eSJoseph Mingrone (const struct lsmsg *)(nspp + sizeof(struct seghdr)); 887*ee67461eSJoseph Mingrone u_int ack; 888*ee67461eSJoseph Mingrone u_int lsflags, fcval; 8894edb46e9SPaul Traina 890f4d0c64aSSam Leffler if (nsplen < sizeof(struct seghdr) + sizeof(struct lsmsg)) 891*ee67461eSJoseph Mingrone goto invalid; 892*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[0]); 8934edb46e9SPaul Traina if (ack & SGQ_ACK) { /* acknum field */ 8944edb46e9SPaul Traina if ((ack & SGQ_NAK) == SGQ_NAK) 895*ee67461eSJoseph Mingrone ND_PRINT("nak %u ", ack & SGQ_MASK); 8964edb46e9SPaul Traina else 897*ee67461eSJoseph Mingrone ND_PRINT("ack %u ", ack & SGQ_MASK); 898*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[1]); 8994edb46e9SPaul Traina if (ack & SGQ_OACK) { /* ackdat field */ 9004edb46e9SPaul Traina if ((ack & SGQ_ONAK) == SGQ_ONAK) 901*ee67461eSJoseph Mingrone ND_PRINT("nakdat %u ", ack & SGQ_MASK); 9024edb46e9SPaul Traina else 903*ee67461eSJoseph Mingrone ND_PRINT("ackdat %u ", ack & SGQ_MASK); 904*ee67461eSJoseph Mingrone ack = GET_LE_U_2(shp->sh_seq[2]); 9054edb46e9SPaul Traina } 9064edb46e9SPaul Traina } 907*ee67461eSJoseph Mingrone ND_PRINT("seg %u ", ack & SGQ_MASK); 908*ee67461eSJoseph Mingrone lsflags = GET_U_1(lsmp->ls_lsflags); 909*ee67461eSJoseph Mingrone fcval = GET_U_1(lsmp->ls_fcval); 9104edb46e9SPaul Traina switch (lsflags & LSI_MASK) { 9114edb46e9SPaul Traina case LSI_DATA: 912*ee67461eSJoseph Mingrone ND_PRINT("dat seg count %u ", fcval); 9134edb46e9SPaul Traina switch (lsflags & LSM_MASK) { 9144edb46e9SPaul Traina case LSM_NOCHANGE: 9154edb46e9SPaul Traina break; 9164edb46e9SPaul Traina case LSM_DONOTSEND: 917*ee67461eSJoseph Mingrone ND_PRINT("donotsend-data "); 9184edb46e9SPaul Traina break; 9194edb46e9SPaul Traina case LSM_SEND: 920*ee67461eSJoseph Mingrone ND_PRINT("send-data "); 9214edb46e9SPaul Traina break; 9224edb46e9SPaul Traina default: 923*ee67461eSJoseph Mingrone ND_PRINT("reserved-fcmod? %x", lsflags); 9244edb46e9SPaul Traina break; 9254edb46e9SPaul Traina } 9264edb46e9SPaul Traina break; 9274edb46e9SPaul Traina case LSI_INTR: 928*ee67461eSJoseph Mingrone ND_PRINT("intr req count %u ", fcval); 9294edb46e9SPaul Traina break; 9304edb46e9SPaul Traina default: 931*ee67461eSJoseph Mingrone ND_PRINT("reserved-fcval-int? %x", lsflags); 9324edb46e9SPaul Traina break; 9334edb46e9SPaul Traina } 9344edb46e9SPaul Traina } 9354edb46e9SPaul Traina break; 9364edb46e9SPaul Traina default: 937*ee67461eSJoseph Mingrone ND_PRINT("reserved-subtype? %x %u > %u", flags, src, dst); 9384edb46e9SPaul Traina break; 9394edb46e9SPaul Traina } 9404edb46e9SPaul Traina break; 9414edb46e9SPaul Traina case MFT_ACK: 9424edb46e9SPaul Traina switch (flags & NSP_SUBMASK) { 9434edb46e9SPaul Traina case MFS_DACK: 944*ee67461eSJoseph Mingrone ND_PRINT("data-ack %u>%u ", src, dst); 9454edb46e9SPaul Traina { 9463340d773SGleb Smirnoff const struct ackmsg *amp = (const struct ackmsg *)nspp; 947*ee67461eSJoseph Mingrone u_int ack; 9484edb46e9SPaul Traina 949f4d0c64aSSam Leffler if (nsplen < sizeof(struct ackmsg)) 950*ee67461eSJoseph Mingrone goto invalid; 951*ee67461eSJoseph Mingrone ND_TCHECK_SIZE(amp); 952*ee67461eSJoseph Mingrone ack = GET_LE_U_2(amp->ak_acknum[0]); 9534edb46e9SPaul Traina if (ack & SGQ_ACK) { /* acknum field */ 9544edb46e9SPaul Traina if ((ack & SGQ_NAK) == SGQ_NAK) 955*ee67461eSJoseph Mingrone ND_PRINT("nak %u ", ack & SGQ_MASK); 9564edb46e9SPaul Traina else 957*ee67461eSJoseph Mingrone ND_PRINT("ack %u ", ack & SGQ_MASK); 958*ee67461eSJoseph Mingrone ack = GET_LE_U_2(amp->ak_acknum[1]); 9594edb46e9SPaul Traina if (ack & SGQ_OACK) { /* ackoth field */ 9604edb46e9SPaul Traina if ((ack & SGQ_ONAK) == SGQ_ONAK) 961*ee67461eSJoseph Mingrone ND_PRINT("onak %u ", ack & SGQ_MASK); 9624edb46e9SPaul Traina else 963*ee67461eSJoseph Mingrone ND_PRINT("oack %u ", ack & SGQ_MASK); 9644edb46e9SPaul Traina } 9654edb46e9SPaul Traina } 9664edb46e9SPaul Traina } 9674edb46e9SPaul Traina break; 9684edb46e9SPaul Traina case MFS_IACK: 969*ee67461eSJoseph Mingrone ND_PRINT("ils-ack %u>%u ", src, dst); 9704edb46e9SPaul Traina { 9713340d773SGleb Smirnoff const struct ackmsg *amp = (const struct ackmsg *)nspp; 972*ee67461eSJoseph Mingrone u_int ack; 9734edb46e9SPaul Traina 974f4d0c64aSSam Leffler if (nsplen < sizeof(struct ackmsg)) 975*ee67461eSJoseph Mingrone goto invalid; 976*ee67461eSJoseph Mingrone ND_TCHECK_SIZE(amp); 977*ee67461eSJoseph Mingrone ack = GET_LE_U_2(amp->ak_acknum[0]); 9784edb46e9SPaul Traina if (ack & SGQ_ACK) { /* acknum field */ 9794edb46e9SPaul Traina if ((ack & SGQ_NAK) == SGQ_NAK) 980*ee67461eSJoseph Mingrone ND_PRINT("nak %u ", ack & SGQ_MASK); 9814edb46e9SPaul Traina else 982*ee67461eSJoseph Mingrone ND_PRINT("ack %u ", ack & SGQ_MASK); 983*ee67461eSJoseph Mingrone ack = GET_LE_U_2(amp->ak_acknum[1]); 9844edb46e9SPaul Traina if (ack & SGQ_OACK) { /* ackdat field */ 9854edb46e9SPaul Traina if ((ack & SGQ_ONAK) == SGQ_ONAK) 986*ee67461eSJoseph Mingrone ND_PRINT("nakdat %u ", ack & SGQ_MASK); 9874edb46e9SPaul Traina else 988*ee67461eSJoseph Mingrone ND_PRINT("ackdat %u ", ack & SGQ_MASK); 9894edb46e9SPaul Traina } 9904edb46e9SPaul Traina } 9914edb46e9SPaul Traina } 9924edb46e9SPaul Traina break; 9934edb46e9SPaul Traina case MFS_CACK: 994*ee67461eSJoseph Mingrone ND_PRINT("conn-ack %u", dst); 9954edb46e9SPaul Traina break; 9964edb46e9SPaul Traina default: 997*ee67461eSJoseph Mingrone ND_PRINT("reserved-acktype? %x %u > %u", flags, src, dst); 9984edb46e9SPaul Traina break; 9994edb46e9SPaul Traina } 10004edb46e9SPaul Traina break; 10014edb46e9SPaul Traina case MFT_CTL: 10024edb46e9SPaul Traina switch (flags & NSP_SUBMASK) { 10034edb46e9SPaul Traina case MFS_CI: 10044edb46e9SPaul Traina case MFS_RCI: 10054edb46e9SPaul Traina if ((flags & NSP_SUBMASK) == MFS_CI) 1006*ee67461eSJoseph Mingrone ND_PRINT("conn-initiate "); 10074edb46e9SPaul Traina else 1008*ee67461eSJoseph Mingrone ND_PRINT("retrans-conn-initiate "); 1009*ee67461eSJoseph Mingrone ND_PRINT("%u>%u ", src, dst); 10104edb46e9SPaul Traina { 10113340d773SGleb Smirnoff const struct cimsg *cimp = (const struct cimsg *)nspp; 1012*ee67461eSJoseph Mingrone u_int services, info, segsize; 10134edb46e9SPaul Traina 1014f4d0c64aSSam Leffler if (nsplen < sizeof(struct cimsg)) 1015*ee67461eSJoseph Mingrone goto invalid; 1016*ee67461eSJoseph Mingrone services = GET_U_1(cimp->ci_services); 1017*ee67461eSJoseph Mingrone info = GET_U_1(cimp->ci_info); 1018*ee67461eSJoseph Mingrone segsize = GET_LE_U_2(cimp->ci_segsize); 10194edb46e9SPaul Traina 10204edb46e9SPaul Traina switch (services & COS_MASK) { 10214edb46e9SPaul Traina case COS_NONE: 10224edb46e9SPaul Traina break; 10234edb46e9SPaul Traina case COS_SEGMENT: 1024*ee67461eSJoseph Mingrone ND_PRINT("seg "); 10254edb46e9SPaul Traina break; 10264edb46e9SPaul Traina case COS_MESSAGE: 1027*ee67461eSJoseph Mingrone ND_PRINT("msg "); 10284edb46e9SPaul Traina break; 10294edb46e9SPaul Traina } 10304edb46e9SPaul Traina switch (info & COI_MASK) { 10314edb46e9SPaul Traina case COI_32: 1032*ee67461eSJoseph Mingrone ND_PRINT("ver 3.2 "); 10334edb46e9SPaul Traina break; 10344edb46e9SPaul Traina case COI_31: 1035*ee67461eSJoseph Mingrone ND_PRINT("ver 3.1 "); 10364edb46e9SPaul Traina break; 10374edb46e9SPaul Traina case COI_40: 1038*ee67461eSJoseph Mingrone ND_PRINT("ver 4.0 "); 10394edb46e9SPaul Traina break; 10404edb46e9SPaul Traina case COI_41: 1041*ee67461eSJoseph Mingrone ND_PRINT("ver 4.1 "); 10424edb46e9SPaul Traina break; 10434edb46e9SPaul Traina } 1044*ee67461eSJoseph Mingrone ND_PRINT("segsize %u ", segsize); 10454edb46e9SPaul Traina } 10464edb46e9SPaul Traina break; 10474edb46e9SPaul Traina case MFS_CC: 1048*ee67461eSJoseph Mingrone ND_PRINT("conn-confirm %u>%u ", src, dst); 10494edb46e9SPaul Traina { 10503340d773SGleb Smirnoff const struct ccmsg *ccmp = (const struct ccmsg *)nspp; 1051*ee67461eSJoseph Mingrone u_int services, info; 10524edb46e9SPaul Traina u_int segsize, optlen; 10534edb46e9SPaul Traina 1054f4d0c64aSSam Leffler if (nsplen < sizeof(struct ccmsg)) 1055*ee67461eSJoseph Mingrone goto invalid; 1056*ee67461eSJoseph Mingrone services = GET_U_1(ccmp->cc_services); 1057*ee67461eSJoseph Mingrone info = GET_U_1(ccmp->cc_info); 1058*ee67461eSJoseph Mingrone segsize = GET_LE_U_2(ccmp->cc_segsize); 1059*ee67461eSJoseph Mingrone optlen = GET_U_1(ccmp->cc_optlen); 10604edb46e9SPaul Traina 10614edb46e9SPaul Traina switch (services & COS_MASK) { 10624edb46e9SPaul Traina case COS_NONE: 10634edb46e9SPaul Traina break; 10644edb46e9SPaul Traina case COS_SEGMENT: 1065*ee67461eSJoseph Mingrone ND_PRINT("seg "); 10664edb46e9SPaul Traina break; 10674edb46e9SPaul Traina case COS_MESSAGE: 1068*ee67461eSJoseph Mingrone ND_PRINT("msg "); 10694edb46e9SPaul Traina break; 10704edb46e9SPaul Traina } 10714edb46e9SPaul Traina switch (info & COI_MASK) { 10724edb46e9SPaul Traina case COI_32: 1073*ee67461eSJoseph Mingrone ND_PRINT("ver 3.2 "); 10744edb46e9SPaul Traina break; 10754edb46e9SPaul Traina case COI_31: 1076*ee67461eSJoseph Mingrone ND_PRINT("ver 3.1 "); 10774edb46e9SPaul Traina break; 10784edb46e9SPaul Traina case COI_40: 1079*ee67461eSJoseph Mingrone ND_PRINT("ver 4.0 "); 10804edb46e9SPaul Traina break; 10814edb46e9SPaul Traina case COI_41: 1082*ee67461eSJoseph Mingrone ND_PRINT("ver 4.1 "); 10834edb46e9SPaul Traina break; 10844edb46e9SPaul Traina } 1085*ee67461eSJoseph Mingrone ND_PRINT("segsize %u ", segsize); 10864edb46e9SPaul Traina if (optlen) { 1087*ee67461eSJoseph Mingrone ND_PRINT("optlen %u ", optlen); 10884edb46e9SPaul Traina } 10894edb46e9SPaul Traina } 10904edb46e9SPaul Traina break; 10914edb46e9SPaul Traina case MFS_DI: 1092*ee67461eSJoseph Mingrone ND_PRINT("disconn-initiate %u>%u ", src, dst); 10934edb46e9SPaul Traina { 10943340d773SGleb Smirnoff const struct dimsg *dimp = (const struct dimsg *)nspp; 1095*ee67461eSJoseph Mingrone u_int reason; 10964edb46e9SPaul Traina u_int optlen; 10974edb46e9SPaul Traina 1098f4d0c64aSSam Leffler if (nsplen < sizeof(struct dimsg)) 1099*ee67461eSJoseph Mingrone goto invalid; 1100*ee67461eSJoseph Mingrone reason = GET_LE_U_2(dimp->di_reason); 1101*ee67461eSJoseph Mingrone optlen = GET_U_1(dimp->di_optlen); 11024edb46e9SPaul Traina 11033c602fabSXin LI print_reason(ndo, reason); 11044edb46e9SPaul Traina if (optlen) { 1105*ee67461eSJoseph Mingrone ND_PRINT("optlen %u ", optlen); 11064edb46e9SPaul Traina } 11074edb46e9SPaul Traina } 11084edb46e9SPaul Traina break; 11094edb46e9SPaul Traina case MFS_DC: 1110*ee67461eSJoseph Mingrone ND_PRINT("disconn-confirm %u>%u ", src, dst); 11114edb46e9SPaul Traina { 11123340d773SGleb Smirnoff const struct dcmsg *dcmp = (const struct dcmsg *)nspp; 1113*ee67461eSJoseph Mingrone u_int reason; 11144edb46e9SPaul Traina 1115*ee67461eSJoseph Mingrone reason = GET_LE_U_2(dcmp->dc_reason); 11164edb46e9SPaul Traina 11173c602fabSXin LI print_reason(ndo, reason); 11184edb46e9SPaul Traina } 11194edb46e9SPaul Traina break; 11204edb46e9SPaul Traina default: 1121*ee67461eSJoseph Mingrone ND_PRINT("reserved-ctltype? %x %u > %u", flags, src, dst); 11224edb46e9SPaul Traina break; 11234edb46e9SPaul Traina } 11244edb46e9SPaul Traina break; 11254edb46e9SPaul Traina default: 1126*ee67461eSJoseph Mingrone ND_PRINT("reserved-type? %x %u > %u", flags, src, dst); 11274edb46e9SPaul Traina break; 11284edb46e9SPaul Traina } 1129f4d0c64aSSam Leffler return (1); 1130f4d0c64aSSam Leffler 1131*ee67461eSJoseph Mingrone invalid: 1132f4d0c64aSSam Leffler return (0); 11334edb46e9SPaul Traina } 11344edb46e9SPaul Traina 11353c602fabSXin LI static const struct tok reason2str[] = { 11364edb46e9SPaul Traina { UC_OBJREJECT, "object rejected connect" }, 11374edb46e9SPaul Traina { UC_RESOURCES, "insufficient resources" }, 11384edb46e9SPaul Traina { UC_NOSUCHNODE, "unrecognized node name" }, 11394edb46e9SPaul Traina { DI_SHUT, "node is shutting down" }, 11404edb46e9SPaul Traina { UC_NOSUCHOBJ, "unrecognized object" }, 11414edb46e9SPaul Traina { UC_INVOBJFORMAT, "invalid object name format" }, 11424edb46e9SPaul Traina { UC_OBJTOOBUSY, "object too busy" }, 11434edb46e9SPaul Traina { DI_PROTOCOL, "protocol error discovered" }, 11444edb46e9SPaul Traina { DI_TPA, "third party abort" }, 11454edb46e9SPaul Traina { UC_USERABORT, "user abort" }, 11464edb46e9SPaul Traina { UC_INVNODEFORMAT, "invalid node name format" }, 11474edb46e9SPaul Traina { UC_LOCALSHUT, "local node shutting down" }, 11484edb46e9SPaul Traina { DI_LOCALRESRC, "insufficient local resources" }, 11494edb46e9SPaul Traina { DI_REMUSERRESRC, "insufficient remote user resources" }, 11504edb46e9SPaul Traina { UC_ACCESSREJECT, "invalid access control information" }, 11514edb46e9SPaul Traina { DI_BADACCNT, "bad ACCOUNT information" }, 11524edb46e9SPaul Traina { UC_NORESPONSE, "no response from object" }, 11534edb46e9SPaul Traina { UC_UNREACHABLE, "node unreachable" }, 11544edb46e9SPaul Traina { DC_NOLINK, "no link terminate" }, 11554edb46e9SPaul Traina { DC_COMPLETE, "disconnect complete" }, 11564edb46e9SPaul Traina { DI_BADIMAGE, "bad image data in connect" }, 11574edb46e9SPaul Traina { DI_SERVMISMATCH, "cryptographic service mismatch" }, 11584edb46e9SPaul Traina { 0, NULL } 11594edb46e9SPaul Traina }; 11604edb46e9SPaul Traina 11614edb46e9SPaul Traina static void 11623c602fabSXin LI print_reason(netdissect_options *ndo, 1163*ee67461eSJoseph Mingrone u_int reason) 11644edb46e9SPaul Traina { 1165*ee67461eSJoseph Mingrone ND_PRINT("%s ", tok2str(reason2str, "reason-%u", reason)); 11664edb46e9SPaul Traina } 11674edb46e9SPaul Traina 1168a90e161bSBill Fenner const char * 11693340d773SGleb Smirnoff dnnum_string(netdissect_options *ndo, u_short dnaddr) 11704edb46e9SPaul Traina { 11714edb46e9SPaul Traina char *str; 1172685295f4SBill Fenner size_t siz; 1173*ee67461eSJoseph Mingrone u_int area = (u_short)(dnaddr & AREAMASK) >> AREASHIFT; 1174*ee67461eSJoseph Mingrone u_int node = dnaddr & NODEMASK; 11754edb46e9SPaul Traina 1176*ee67461eSJoseph Mingrone /* malloc() return used by the 'dnaddrtable' hash table: do not free() */ 1177685295f4SBill Fenner str = (char *)malloc(siz = sizeof("00.0000")); 11784edb46e9SPaul Traina if (str == NULL) 1179*ee67461eSJoseph Mingrone (*ndo->ndo_error)(ndo, S_ERR_ND_MEM_ALLOC, "%s: malloc", __func__); 1180*ee67461eSJoseph Mingrone snprintf(str, siz, "%u.%u", area, node); 11814edb46e9SPaul Traina return(str); 11824edb46e9SPaul Traina } 1183