xref: /freebsd/contrib/tcpdump/print-wb.c (revision 0a7e5f1f02aad2ff5fff1c60f44c6975fd07e1d9)
14edb46e9SPaul Traina /*
24edb46e9SPaul Traina  * Copyright (c) 1993, 1994, 1995, 1996
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: White Board printer */
233340d773SGleb Smirnoff 
24*ee67461eSJoseph Mingrone #include <config.h>
254edb46e9SPaul Traina 
26*ee67461eSJoseph Mingrone #include "netdissect-stdinc.h"
274edb46e9SPaul Traina 
28*ee67461eSJoseph Mingrone #define ND_LONGJMP_FROM_TCHECK
293340d773SGleb Smirnoff #include "netdissect.h"
304edb46e9SPaul Traina #include "addrtoname.h"
315b0fe478SBruce M Simpson #include "extract.h"
324edb46e9SPaul Traina 
333c602fabSXin LI 
34*ee67461eSJoseph Mingrone #if 0
354edb46e9SPaul Traina /*
364edb46e9SPaul Traina  * Largest packet size.  Everything should fit within this space.
374edb46e9SPaul Traina  * For instance, multiline objects are sent piecewise.
384edb46e9SPaul Traina  */
394edb46e9SPaul Traina #define MAXFRAMESIZE 1024
40*ee67461eSJoseph Mingrone #endif
414edb46e9SPaul Traina 
424edb46e9SPaul Traina /*
434edb46e9SPaul Traina  * Multiple drawing ops can be sent in one packet.  Each one starts on a
444edb46e9SPaul Traina  * an even multiple of DOP_ALIGN bytes, which must be a power of two.
454edb46e9SPaul Traina  */
464edb46e9SPaul Traina #define DOP_ALIGN 4
47*ee67461eSJoseph Mingrone #define DOP_ROUNDUP(x)	roundup2(x, DOP_ALIGN)
484edb46e9SPaul Traina #define DOP_NEXT(d)\
493340d773SGleb Smirnoff 	((const struct dophdr *)((const u_char *)(d) + \
50*ee67461eSJoseph Mingrone 				DOP_ROUNDUP(GET_BE_U_2((d)->dh_len) + sizeof(*(d)))))
514edb46e9SPaul Traina 
524edb46e9SPaul Traina /*
534edb46e9SPaul Traina  * Format of the whiteboard packet header.
544edb46e9SPaul Traina  * The transport level header.
554edb46e9SPaul Traina  */
564edb46e9SPaul Traina struct pkt_hdr {
57*ee67461eSJoseph Mingrone 	nd_uint32_t ph_src;	/* site id of source */
58*ee67461eSJoseph Mingrone 	nd_uint32_t ph_ts;	/* time stamp (for skew computation) */
59*ee67461eSJoseph Mingrone 	nd_uint16_t ph_version;	/* version number */
60*ee67461eSJoseph Mingrone 	nd_uint8_t ph_type;	/* message type */
61*ee67461eSJoseph Mingrone 	nd_uint8_t ph_flags;	/* message flags */
624edb46e9SPaul Traina };
634edb46e9SPaul Traina 
644edb46e9SPaul Traina /* Packet types */
654edb46e9SPaul Traina #define PT_DRAWOP	0	/* drawing operation */
664edb46e9SPaul Traina #define PT_ID		1	/* announcement packet */
674edb46e9SPaul Traina #define PT_RREQ		2	/* repair request */
684edb46e9SPaul Traina #define PT_RREP		3	/* repair reply */
694edb46e9SPaul Traina #define PT_KILL		4	/* terminate participation */
704edb46e9SPaul Traina #define PT_PREQ         5       /* page vector request */
714edb46e9SPaul Traina #define PT_PREP         7       /* page vector reply */
724edb46e9SPaul Traina 
73*ee67461eSJoseph Mingrone #if 0
74a90e161bSBill Fenner #ifdef PF_USER
75a90e161bSBill Fenner #undef PF_USER			/* {Digital,Tru64} UNIX define this, alas */
76a90e161bSBill Fenner #endif
77a90e161bSBill Fenner 
784edb46e9SPaul Traina /* flags */
794edb46e9SPaul Traina #define PF_USER		0x01	/* hint that packet has interactive data */
804edb46e9SPaul Traina #define PF_VIS		0x02	/* only visible ops wanted */
81*ee67461eSJoseph Mingrone #endif
824edb46e9SPaul Traina 
834edb46e9SPaul Traina struct PageID {
84*ee67461eSJoseph Mingrone 	nd_uint32_t p_sid;		/* session id of initiator */
85*ee67461eSJoseph Mingrone 	nd_uint32_t p_uid;		/* page number */
864edb46e9SPaul Traina };
874edb46e9SPaul Traina 
884edb46e9SPaul Traina struct dophdr {
89*ee67461eSJoseph Mingrone 	nd_uint32_t	dh_ts;		/* sender's timestamp */
90*ee67461eSJoseph Mingrone 	nd_uint16_t	dh_len;		/* body length */
91*ee67461eSJoseph Mingrone 	nd_uint8_t	dh_flags;
92*ee67461eSJoseph Mingrone 	nd_uint8_t	dh_type;	/* body type */
934edb46e9SPaul Traina 	/* body follows */
944edb46e9SPaul Traina };
954edb46e9SPaul Traina /*
964edb46e9SPaul Traina  * Drawing op sub-types.
974edb46e9SPaul Traina  */
984edb46e9SPaul Traina #define DT_RECT         2
994edb46e9SPaul Traina #define DT_LINE         3
1004edb46e9SPaul Traina #define DT_ML           4
1014edb46e9SPaul Traina #define DT_DEL          5
1024edb46e9SPaul Traina #define DT_XFORM        6
1034edb46e9SPaul Traina #define DT_ELL          7
1044edb46e9SPaul Traina #define DT_CHAR         8
1054edb46e9SPaul Traina #define DT_STR          9
1064edb46e9SPaul Traina #define DT_NOP          10
1074edb46e9SPaul Traina #define DT_PSCODE       11
1084edb46e9SPaul Traina #define DT_PSCOMP       12
1094edb46e9SPaul Traina #define DT_REF          13
1104edb46e9SPaul Traina #define DT_SKIP         14
1114edb46e9SPaul Traina #define DT_HOLE         15
112*ee67461eSJoseph Mingrone static const struct tok dop_str[] = {
113*ee67461eSJoseph Mingrone 	{ DT_RECT,   "RECT"   },
114*ee67461eSJoseph Mingrone 	{ DT_LINE,   "LINE"   },
115*ee67461eSJoseph Mingrone 	{ DT_ML,     "ML"     },
116*ee67461eSJoseph Mingrone 	{ DT_DEL,    "DEL"    },
117*ee67461eSJoseph Mingrone 	{ DT_XFORM,  "XFORM"  },
118*ee67461eSJoseph Mingrone 	{ DT_ELL,    "ELL"    },
119*ee67461eSJoseph Mingrone 	{ DT_CHAR,   "CHAR"   },
120*ee67461eSJoseph Mingrone 	{ DT_STR,    "STR"    },
121*ee67461eSJoseph Mingrone 	{ DT_NOP,    "NOP"    },
122*ee67461eSJoseph Mingrone 	{ DT_PSCODE, "PSCODE" },
123*ee67461eSJoseph Mingrone 	{ DT_PSCOMP, "PSCOMP" },
124*ee67461eSJoseph Mingrone 	{ DT_REF,    "REF"    },
125*ee67461eSJoseph Mingrone 	{ DT_SKIP,   "SKIP"   },
126*ee67461eSJoseph Mingrone 	{ DT_HOLE,   "HOLE"   },
127*ee67461eSJoseph Mingrone 	{ 0, NULL }
128*ee67461eSJoseph Mingrone };
1294edb46e9SPaul Traina 
1304edb46e9SPaul Traina /*
1314edb46e9SPaul Traina  * A drawing operation.
1324edb46e9SPaul Traina  */
1334edb46e9SPaul Traina struct pkt_dop {
1344edb46e9SPaul Traina 	struct PageID pd_page;	/* page that operations apply to */
135*ee67461eSJoseph Mingrone 	nd_uint32_t	pd_sseq;	/* start sequence number */
136*ee67461eSJoseph Mingrone 	nd_uint32_t	pd_eseq;	/* end sequence number */
1374edb46e9SPaul Traina 	/* drawing ops follow */
1384edb46e9SPaul Traina };
1394edb46e9SPaul Traina 
1404edb46e9SPaul Traina /*
1414edb46e9SPaul Traina  * A repair request.
1424edb46e9SPaul Traina  */
1434edb46e9SPaul Traina struct pkt_rreq {
144*ee67461eSJoseph Mingrone         nd_uint32_t pr_id;        /* source id of drawops to be repaired */
1454edb46e9SPaul Traina         struct PageID pr_page;    /* page of drawops */
146*ee67461eSJoseph Mingrone         nd_uint32_t pr_sseq;      /* start seqno */
147*ee67461eSJoseph Mingrone         nd_uint32_t pr_eseq;      /* end seqno */
1484edb46e9SPaul Traina };
1494edb46e9SPaul Traina 
1504edb46e9SPaul Traina /*
1514edb46e9SPaul Traina  * A repair reply.
1524edb46e9SPaul Traina  */
1534edb46e9SPaul Traina struct pkt_rrep {
154*ee67461eSJoseph Mingrone 	nd_uint32_t pr_id;	/* original site id of ops  */
1554edb46e9SPaul Traina 	struct pkt_dop pr_dop;
1564edb46e9SPaul Traina 	/* drawing ops follow */
1574edb46e9SPaul Traina };
1584edb46e9SPaul Traina 
1594edb46e9SPaul Traina struct id_off {
160*ee67461eSJoseph Mingrone         nd_uint32_t id;
161*ee67461eSJoseph Mingrone         nd_uint32_t off;
1624edb46e9SPaul Traina };
1634edb46e9SPaul Traina 
1644edb46e9SPaul Traina struct pgstate {
165*ee67461eSJoseph Mingrone 	nd_uint32_t slot;
1664edb46e9SPaul Traina 	struct PageID page;
167*ee67461eSJoseph Mingrone 	nd_uint16_t nid;
168*ee67461eSJoseph Mingrone 	nd_uint16_t rsvd;
1694edb46e9SPaul Traina         /* seqptr's */
1704edb46e9SPaul Traina };
1714edb46e9SPaul Traina 
1724edb46e9SPaul Traina /*
1734edb46e9SPaul Traina  * An announcement packet.
1744edb46e9SPaul Traina  */
1754edb46e9SPaul Traina struct pkt_id {
176*ee67461eSJoseph Mingrone 	nd_uint32_t pi_mslot;
1774edb46e9SPaul Traina         struct PageID    pi_mpage;        /* current page */
1784edb46e9SPaul Traina 	struct pgstate pi_ps;
1794edb46e9SPaul Traina         /* seqptr's */
1804edb46e9SPaul Traina         /* null-terminated site name */
1814edb46e9SPaul Traina };
1824edb46e9SPaul Traina 
1834edb46e9SPaul Traina struct pkt_preq {
1844edb46e9SPaul Traina         struct PageID  pp_page;
185*ee67461eSJoseph Mingrone         nd_uint32_t  pp_low;
186*ee67461eSJoseph Mingrone         nd_uint32_t  pp_high;
1874edb46e9SPaul Traina };
1884edb46e9SPaul Traina 
1894edb46e9SPaul Traina struct pkt_prep {
190*ee67461eSJoseph Mingrone         nd_uint32_t  pp_n;           /* size of pageid array */
1914edb46e9SPaul Traina         /* pgstate's follow */
1924edb46e9SPaul Traina };
1934edb46e9SPaul Traina 
1944edb46e9SPaul Traina static int
wb_id(netdissect_options * ndo,const struct pkt_id * id,u_int len)1953c602fabSXin LI wb_id(netdissect_options *ndo,
1963c602fabSXin LI       const struct pkt_id *id, u_int len)
1974edb46e9SPaul Traina {
198*ee67461eSJoseph Mingrone 	u_int i;
199*ee67461eSJoseph Mingrone 	const u_char *sitename;
2004edb46e9SPaul Traina 	const struct id_off *io;
2014edb46e9SPaul Traina 	char c;
202*ee67461eSJoseph Mingrone 	u_int nid;
2034edb46e9SPaul Traina 
204*ee67461eSJoseph Mingrone 	ND_PRINT(" wb-id:");
205*ee67461eSJoseph Mingrone 	if (len < sizeof(*id))
2064edb46e9SPaul Traina 		return (-1);
2074644f044SBill Fenner 	len -= sizeof(*id);
2084edb46e9SPaul Traina 
209*ee67461eSJoseph Mingrone 	ND_PRINT(" %u/%s:%u (max %u/%s:%u) ",
210*ee67461eSJoseph Mingrone 	       GET_BE_U_4(id->pi_ps.slot),
211*ee67461eSJoseph Mingrone 	       GET_IPADDR_STRING(id->pi_ps.page.p_sid),
212*ee67461eSJoseph Mingrone 	       GET_BE_U_4(id->pi_ps.page.p_uid),
213*ee67461eSJoseph Mingrone 	       GET_BE_U_4(id->pi_mslot),
214*ee67461eSJoseph Mingrone 	       GET_IPADDR_STRING(id->pi_mpage.p_sid),
215*ee67461eSJoseph Mingrone 	       GET_BE_U_4(id->pi_mpage.p_uid));
216*ee67461eSJoseph Mingrone 	/* now the rest of the fixed-size part of struct pkt_id */
217*ee67461eSJoseph Mingrone 	ND_TCHECK_SIZE(id);
2184edb46e9SPaul Traina 
219*ee67461eSJoseph Mingrone 	nid = GET_BE_U_2(id->pi_ps.nid);
220*ee67461eSJoseph Mingrone 	if (len < sizeof(*io) * nid)
221*ee67461eSJoseph Mingrone 		return (-1);
2224edb46e9SPaul Traina 	len -= sizeof(*io) * nid;
2233340d773SGleb Smirnoff 	io = (const struct id_off *)(id + 1);
224*ee67461eSJoseph Mingrone 	sitename = (const u_char *)(io + nid);
2254edb46e9SPaul Traina 
2264edb46e9SPaul Traina 	c = '<';
227*ee67461eSJoseph Mingrone 	for (i = 0; i < nid; ++io, ++i) {
228*ee67461eSJoseph Mingrone 		ND_PRINT("%c%s:%u",
229*ee67461eSJoseph Mingrone 		    c, GET_IPADDR_STRING(io->id), GET_BE_U_4(io->off));
2304edb46e9SPaul Traina 		c = ',';
2314edb46e9SPaul Traina 	}
232*ee67461eSJoseph Mingrone 	ND_PRINT("> \"");
233*ee67461eSJoseph Mingrone 	nd_printjnp(ndo, sitename, len);
234*ee67461eSJoseph Mingrone 	ND_PRINT("\"");
2354edb46e9SPaul Traina 	return (0);
2364edb46e9SPaul Traina }
2374edb46e9SPaul Traina 
2384edb46e9SPaul Traina static int
wb_rreq(netdissect_options * ndo,const struct pkt_rreq * rreq,u_int len)2393c602fabSXin LI wb_rreq(netdissect_options *ndo,
2403c602fabSXin LI         const struct pkt_rreq *rreq, u_int len)
2414edb46e9SPaul Traina {
242*ee67461eSJoseph Mingrone 	ND_PRINT(" wb-rreq:");
243*ee67461eSJoseph Mingrone 	if (len < sizeof(*rreq))
2444edb46e9SPaul Traina 		return (-1);
2454edb46e9SPaul Traina 
246*ee67461eSJoseph Mingrone 	ND_PRINT(" please repair %s %s:%u<%u:%u>",
247*ee67461eSJoseph Mingrone 	       GET_IPADDR_STRING(rreq->pr_id),
248*ee67461eSJoseph Mingrone 	       GET_IPADDR_STRING(rreq->pr_page.p_sid),
249*ee67461eSJoseph Mingrone 	       GET_BE_U_4(rreq->pr_page.p_uid),
250*ee67461eSJoseph Mingrone 	       GET_BE_U_4(rreq->pr_sseq),
251*ee67461eSJoseph Mingrone 	       GET_BE_U_4(rreq->pr_eseq));
2524edb46e9SPaul Traina 	return (0);
2534edb46e9SPaul Traina }
2544edb46e9SPaul Traina 
2554edb46e9SPaul Traina static int
wb_preq(netdissect_options * ndo,const struct pkt_preq * preq,u_int len)2563c602fabSXin LI wb_preq(netdissect_options *ndo,
2573c602fabSXin LI         const struct pkt_preq *preq, u_int len)
2584edb46e9SPaul Traina {
259*ee67461eSJoseph Mingrone 	ND_PRINT(" wb-preq:");
260*ee67461eSJoseph Mingrone 	if (len < sizeof(*preq))
2614edb46e9SPaul Traina 		return (-1);
2624edb46e9SPaul Traina 
263*ee67461eSJoseph Mingrone 	ND_PRINT(" need %u/%s:%u",
264*ee67461eSJoseph Mingrone 	       GET_BE_U_4(preq->pp_low),
265*ee67461eSJoseph Mingrone 	       GET_IPADDR_STRING(preq->pp_page.p_sid),
266*ee67461eSJoseph Mingrone 	       GET_BE_U_4(preq->pp_page.p_uid));
267*ee67461eSJoseph Mingrone 	/* now the rest of the fixed-size part of struct pkt_req */
268*ee67461eSJoseph Mingrone 	ND_TCHECK_SIZE(preq);
2694edb46e9SPaul Traina 	return (0);
2704edb46e9SPaul Traina }
2714edb46e9SPaul Traina 
2724edb46e9SPaul Traina static int
wb_prep(netdissect_options * ndo,const struct pkt_prep * prep,u_int len)2733c602fabSXin LI wb_prep(netdissect_options *ndo,
2743c602fabSXin LI         const struct pkt_prep *prep, u_int len)
2754edb46e9SPaul Traina {
276*ee67461eSJoseph Mingrone 	u_int n;
2774edb46e9SPaul Traina 	const struct pgstate *ps;
2784edb46e9SPaul Traina 
279*ee67461eSJoseph Mingrone 	ND_PRINT(" wb-prep:");
280*ee67461eSJoseph Mingrone 	if (len < sizeof(*prep))
2814edb46e9SPaul Traina 		return (-1);
282*ee67461eSJoseph Mingrone 	n = GET_BE_U_4(prep->pp_n);
2834edb46e9SPaul Traina 	ps = (const struct pgstate *)(prep + 1);
284*ee67461eSJoseph Mingrone 	while (n != 0) {
2854edb46e9SPaul Traina 		const struct id_off *io, *ie;
2864edb46e9SPaul Traina 		char c = '<';
2874edb46e9SPaul Traina 
288*ee67461eSJoseph Mingrone 		ND_PRINT(" %u/%s:%u",
289*ee67461eSJoseph Mingrone 		    GET_BE_U_4(ps->slot),
290*ee67461eSJoseph Mingrone 		    GET_IPADDR_STRING(ps->page.p_sid),
291*ee67461eSJoseph Mingrone 		    GET_BE_U_4(ps->page.p_uid));
292*ee67461eSJoseph Mingrone 		/* now the rest of the fixed-size part of struct pgstate */
293*ee67461eSJoseph Mingrone 		ND_TCHECK_SIZE(ps);
2943340d773SGleb Smirnoff 		io = (const struct id_off *)(ps + 1);
295*ee67461eSJoseph Mingrone 		for (ie = io + GET_U_1(ps->nid); io < ie; ++io) {
296*ee67461eSJoseph Mingrone 			ND_PRINT("%c%s:%u", c, GET_IPADDR_STRING(io->id),
297*ee67461eSJoseph Mingrone 			    GET_BE_U_4(io->off));
2984edb46e9SPaul Traina 			c = ',';
2994edb46e9SPaul Traina 		}
300*ee67461eSJoseph Mingrone 		ND_PRINT(">");
3013340d773SGleb Smirnoff 		ps = (const struct pgstate *)io;
302*ee67461eSJoseph Mingrone 		n--;
3034edb46e9SPaul Traina 	}
304*ee67461eSJoseph Mingrone 	return 0;
3054edb46e9SPaul Traina }
3064edb46e9SPaul Traina 
307*ee67461eSJoseph Mingrone static void
wb_dops(netdissect_options * ndo,const struct pkt_dop * dop,uint32_t ss,uint32_t es)3088bdc5a62SPatrick Kelsey wb_dops(netdissect_options *ndo, const struct pkt_dop *dop,
3098bdc5a62SPatrick Kelsey         uint32_t ss, uint32_t es)
3104edb46e9SPaul Traina {
3118bdc5a62SPatrick Kelsey 	const struct dophdr *dh = (const struct dophdr *)((const u_char *)dop + sizeof(*dop));
3128bdc5a62SPatrick Kelsey 
313*ee67461eSJoseph Mingrone 	ND_PRINT(" <");
3144edb46e9SPaul Traina 	for ( ; ss <= es; ++ss) {
315*ee67461eSJoseph Mingrone 		u_int t;
3168bdc5a62SPatrick Kelsey 
317*ee67461eSJoseph Mingrone 		t = GET_U_1(dh->dh_type);
3184edb46e9SPaul Traina 
319*ee67461eSJoseph Mingrone 		ND_PRINT(" %s", tok2str(dop_str, "dop-%u!", t));
3204edb46e9SPaul Traina 		if (t == DT_SKIP || t == DT_HOLE) {
321*ee67461eSJoseph Mingrone 			uint32_t ts = GET_BE_U_4(dh->dh_ts);
322*ee67461eSJoseph Mingrone 			ND_PRINT("%u", ts - ss + 1);
3234edb46e9SPaul Traina 			if (ss > ts || ts > es) {
324*ee67461eSJoseph Mingrone 				ND_PRINT("[|]");
3254edb46e9SPaul Traina 				if (ts < ss)
326*ee67461eSJoseph Mingrone 					return;
3274edb46e9SPaul Traina 			}
3284edb46e9SPaul Traina 			ss = ts;
3294edb46e9SPaul Traina 		}
3304edb46e9SPaul Traina 		dh = DOP_NEXT(dh);
3314edb46e9SPaul Traina 	}
332*ee67461eSJoseph Mingrone 	ND_PRINT(" >");
3334edb46e9SPaul Traina }
3344edb46e9SPaul Traina 
3354edb46e9SPaul Traina static int
wb_rrep(netdissect_options * ndo,const struct pkt_rrep * rrep,u_int len)3363c602fabSXin LI wb_rrep(netdissect_options *ndo,
3373c602fabSXin LI         const struct pkt_rrep *rrep, u_int len)
3384edb46e9SPaul Traina {
3394edb46e9SPaul Traina 	const struct pkt_dop *dop = &rrep->pr_dop;
3404edb46e9SPaul Traina 
341*ee67461eSJoseph Mingrone 	ND_PRINT(" wb-rrep:");
342*ee67461eSJoseph Mingrone 	if (len < sizeof(*rrep))
3434edb46e9SPaul Traina 		return (-1);
3444644f044SBill Fenner 	len -= sizeof(*rrep);
3454edb46e9SPaul Traina 
346*ee67461eSJoseph Mingrone 	ND_PRINT(" for %s %s:%u<%u:%u>",
347*ee67461eSJoseph Mingrone 	    GET_IPADDR_STRING(rrep->pr_id),
348*ee67461eSJoseph Mingrone 	    GET_IPADDR_STRING(dop->pd_page.p_sid),
349*ee67461eSJoseph Mingrone 	    GET_BE_U_4(dop->pd_page.p_uid),
350*ee67461eSJoseph Mingrone 	    GET_BE_U_4(dop->pd_sseq),
351*ee67461eSJoseph Mingrone 	    GET_BE_U_4(dop->pd_eseq));
3524edb46e9SPaul Traina 
3533c602fabSXin LI 	if (ndo->ndo_vflag)
354*ee67461eSJoseph Mingrone 		wb_dops(ndo, dop,
355*ee67461eSJoseph Mingrone 		        GET_BE_U_4(dop->pd_sseq),
356*ee67461eSJoseph Mingrone 		        GET_BE_U_4(dop->pd_eseq));
3574edb46e9SPaul Traina 	return (0);
3584edb46e9SPaul Traina }
3594edb46e9SPaul Traina 
3604edb46e9SPaul Traina static int
wb_drawop(netdissect_options * ndo,const struct pkt_dop * dop,u_int len)3613c602fabSXin LI wb_drawop(netdissect_options *ndo,
3623c602fabSXin LI           const struct pkt_dop *dop, u_int len)
3634edb46e9SPaul Traina {
364*ee67461eSJoseph Mingrone 	ND_PRINT(" wb-dop:");
365*ee67461eSJoseph Mingrone 	if (len < sizeof(*dop))
3664edb46e9SPaul Traina 		return (-1);
3674644f044SBill Fenner 	len -= sizeof(*dop);
3684edb46e9SPaul Traina 
369*ee67461eSJoseph Mingrone 	ND_PRINT(" %s:%u<%u:%u>",
370*ee67461eSJoseph Mingrone 	    GET_IPADDR_STRING(dop->pd_page.p_sid),
371*ee67461eSJoseph Mingrone 	    GET_BE_U_4(dop->pd_page.p_uid),
372*ee67461eSJoseph Mingrone 	    GET_BE_U_4(dop->pd_sseq),
373*ee67461eSJoseph Mingrone 	    GET_BE_U_4(dop->pd_eseq));
3744edb46e9SPaul Traina 
3753c602fabSXin LI 	if (ndo->ndo_vflag)
376*ee67461eSJoseph Mingrone 		wb_dops(ndo, dop,
377*ee67461eSJoseph Mingrone 		        GET_BE_U_4(dop->pd_sseq),
378*ee67461eSJoseph Mingrone 		        GET_BE_U_4(dop->pd_eseq));
3794edb46e9SPaul Traina 	return (0);
3804edb46e9SPaul Traina }
3814edb46e9SPaul Traina 
3824edb46e9SPaul Traina /*
3834edb46e9SPaul Traina  * Print whiteboard multicast packets.
3844edb46e9SPaul Traina  */
3854edb46e9SPaul Traina void
wb_print(netdissect_options * ndo,const u_char * hdr,u_int len)3863c602fabSXin LI wb_print(netdissect_options *ndo,
387*ee67461eSJoseph Mingrone          const u_char *hdr, u_int len)
3884edb46e9SPaul Traina {
389*ee67461eSJoseph Mingrone 	const struct pkt_hdr *ph;
390*ee67461eSJoseph Mingrone 	uint8_t type;
391*ee67461eSJoseph Mingrone 	int print_result;
3924edb46e9SPaul Traina 
393*ee67461eSJoseph Mingrone 	ndo->ndo_protocol = "wb";
3944edb46e9SPaul Traina 	ph = (const struct pkt_hdr *)hdr;
395*ee67461eSJoseph Mingrone 	if (len < sizeof(*ph))
396*ee67461eSJoseph Mingrone 		goto invalid;
397*ee67461eSJoseph Mingrone 	ND_TCHECK_SIZE(ph);
3984edb46e9SPaul Traina 	len -= sizeof(*ph);
3994644f044SBill Fenner 
400*ee67461eSJoseph Mingrone 	if (GET_U_1(ph->ph_flags))
401*ee67461eSJoseph Mingrone 		ND_PRINT("*");
402*ee67461eSJoseph Mingrone 	type = GET_U_1(ph->ph_type);
403*ee67461eSJoseph Mingrone 	switch (type) {
4044edb46e9SPaul Traina 
4054edb46e9SPaul Traina 	case PT_KILL:
406*ee67461eSJoseph Mingrone 		ND_PRINT(" wb-kill");
4074edb46e9SPaul Traina 		return;
4084edb46e9SPaul Traina 
4094edb46e9SPaul Traina 	case PT_ID:
410*ee67461eSJoseph Mingrone 		print_result = wb_id(ndo, (const struct pkt_id *)(ph + 1), len);
4114edb46e9SPaul Traina 		break;
4124edb46e9SPaul Traina 
4134edb46e9SPaul Traina 	case PT_RREQ:
414*ee67461eSJoseph Mingrone 		print_result = wb_rreq(ndo, (const struct pkt_rreq *)(ph + 1), len);
4154edb46e9SPaul Traina 		break;
4164edb46e9SPaul Traina 
4174edb46e9SPaul Traina 	case PT_RREP:
418*ee67461eSJoseph Mingrone 		print_result = wb_rrep(ndo, (const struct pkt_rrep *)(ph + 1), len);
4194edb46e9SPaul Traina 		break;
4204edb46e9SPaul Traina 
4214edb46e9SPaul Traina 	case PT_DRAWOP:
422*ee67461eSJoseph Mingrone 		print_result = wb_drawop(ndo, (const struct pkt_dop *)(ph + 1), len);
4234edb46e9SPaul Traina 		break;
4244edb46e9SPaul Traina 
4254edb46e9SPaul Traina 	case PT_PREQ:
426*ee67461eSJoseph Mingrone 		print_result = wb_preq(ndo, (const struct pkt_preq *)(ph + 1), len);
4274edb46e9SPaul Traina 		break;
4284edb46e9SPaul Traina 
4294edb46e9SPaul Traina 	case PT_PREP:
430*ee67461eSJoseph Mingrone 		print_result = wb_prep(ndo, (const struct pkt_prep *)(ph + 1), len);
4314edb46e9SPaul Traina 		break;
4324edb46e9SPaul Traina 
4334edb46e9SPaul Traina 	default:
434*ee67461eSJoseph Mingrone 		ND_PRINT(" wb-%u!", type);
435*ee67461eSJoseph Mingrone 		print_result = -1;
4364edb46e9SPaul Traina 	}
437*ee67461eSJoseph Mingrone 	if (print_result < 0)
438*ee67461eSJoseph Mingrone 		goto invalid;
439*ee67461eSJoseph Mingrone 	return;
440*ee67461eSJoseph Mingrone 
441*ee67461eSJoseph Mingrone invalid:
442*ee67461eSJoseph Mingrone 	nd_print_invalid(ndo);
4434edb46e9SPaul Traina }
444