]> The Tcpdump Group git mirrors - tcpdump/blob - print-egp.c
Use nd_ types, add EXTRACT_ calls.
[tcpdump] / print-egp.c
1 /*
2 * Copyright (c) 1991, 1992, 1993, 1994, 1995, 1996
3 * The Regents of the University of California. All rights reserved.
4 *
5 * Redistribution and use in source and binary forms are permitted
6 * provided that the above copyright notice and this paragraph are
7 * duplicated in all such forms and that any documentation,
8 * advertising materials, and other materials related to such
9 * distribution and use acknowledge that the software was developed
10 * by the University of California, Lawrence Berkeley Laboratory,
11 * Berkeley, CA. The name of the University may not be used to
12 * endorse or promote products derived from this software without
13 * specific prior written permission.
14 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
15 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
16 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
17 *
18 * Initial contribution from Jeff Honig (jch@MITCHELL.CIT.CORNELL.EDU).
19 */
20
21 /* \summary: Exterior Gateway Protocol (EGP) printer */
22
23 #ifdef HAVE_CONFIG_H
24 #include "config.h"
25 #endif
26
27 #include <netdissect-stdinc.h>
28
29 #include "netdissect.h"
30 #include "addrtoname.h"
31 #include "extract.h"
32
33 struct egp_packet {
34 uint8_t egp_version;
35 #define EGP_VERSION 2
36 uint8_t egp_type;
37 #define EGPT_ACQUIRE 3
38 #define EGPT_REACH 5
39 #define EGPT_POLL 2
40 #define EGPT_UPDATE 1
41 #define EGPT_ERROR 8
42 uint8_t egp_code;
43 #define EGPC_REQUEST 0
44 #define EGPC_CONFIRM 1
45 #define EGPC_REFUSE 2
46 #define EGPC_CEASE 3
47 #define EGPC_CEASEACK 4
48 #define EGPC_HELLO 0
49 #define EGPC_HEARDU 1
50 uint8_t egp_status;
51 #define EGPS_UNSPEC 0
52 #define EGPS_ACTIVE 1
53 #define EGPS_PASSIVE 2
54 #define EGPS_NORES 3
55 #define EGPS_ADMIN 4
56 #define EGPS_GODOWN 5
57 #define EGPS_PARAM 6
58 #define EGPS_PROTO 7
59 #define EGPS_INDET 0
60 #define EGPS_UP 1
61 #define EGPS_DOWN 2
62 #define EGPS_UNSOL 0x80
63 uint16_t egp_checksum;
64 uint16_t egp_as;
65 uint16_t egp_sequence;
66 union {
67 uint16_t egpu_hello;
68 uint8_t egpu_gws[2];
69 uint16_t egpu_reason;
70 #define EGPR_UNSPEC 0
71 #define EGPR_BADHEAD 1
72 #define EGPR_BADDATA 2
73 #define EGPR_NOREACH 3
74 #define EGPR_XSPOLL 4
75 #define EGPR_NORESP 5
76 #define EGPR_UVERSION 6
77 } egp_handg;
78 #define egp_hello egp_handg.egpu_hello
79 #define egp_intgw egp_handg.egpu_gws[0]
80 #define egp_extgw egp_handg.egpu_gws[1]
81 #define egp_reason egp_handg.egpu_reason
82 union {
83 uint16_t egpu_poll;
84 uint32_t egpu_sourcenet;
85 } egp_pands;
86 #define egp_poll egp_pands.egpu_poll
87 #define egp_sourcenet egp_pands.egpu_sourcenet
88 };
89
90 static const char *egp_acquire_codes[] = {
91 "request",
92 "confirm",
93 "refuse",
94 "cease",
95 "cease_ack"
96 };
97
98 static const char *egp_acquire_status[] = {
99 "unspecified",
100 "active_mode",
101 "passive_mode",
102 "insufficient_resources",
103 "administratively_prohibited",
104 "going_down",
105 "parameter_violation",
106 "protocol_violation"
107 };
108
109 static const char *egp_reach_codes[] = {
110 "hello",
111 "i-h-u"
112 };
113
114 static const char *egp_status_updown[] = {
115 "indeterminate",
116 "up",
117 "down"
118 };
119
120 static const char *egp_reasons[] = {
121 "unspecified",
122 "bad_EGP_header_format",
123 "bad_EGP_data_field_format",
124 "reachability_info_unavailable",
125 "excessive_polling_rate",
126 "no_response",
127 "unsupported_version"
128 };
129
130 static void
131 egpnrprint(netdissect_options *ndo,
132 const struct egp_packet *egp, u_int length)
133 {
134 const uint8_t *cp;
135 uint32_t addr;
136 uint32_t net;
137 u_int netlen;
138 u_int gateways, distances, networks;
139 u_int t_gateways;
140 const char *comma;
141
142 addr = egp->egp_sourcenet;
143 if (IN_CLASSA(addr)) {
144 net = addr & IN_CLASSA_NET;
145 netlen = 1;
146 } else if (IN_CLASSB(addr)) {
147 net = addr & IN_CLASSB_NET;
148 netlen = 2;
149 } else if (IN_CLASSC(addr)) {
150 net = addr & IN_CLASSC_NET;
151 netlen = 3;
152 } else {
153 net = 0;
154 netlen = 0;
155 }
156 cp = (const uint8_t *)(egp + 1);
157 length -= sizeof(*egp);
158
159 t_gateways = egp->egp_intgw + egp->egp_extgw;
160 for (gateways = 0; gateways < t_gateways; ++gateways) {
161 /* Pickup host part of gateway address */
162 addr = 0;
163 if (length < 4 - netlen)
164 goto trunc;
165 ND_TCHECK_LEN(cp, 4 - netlen);
166 switch (netlen) {
167
168 case 1:
169 addr = EXTRACT_U_1(cp);
170 cp++;
171 /* fall through */
172 case 2:
173 addr = (addr << 8) | EXTRACT_U_1(cp);
174 cp++;
175 /* fall through */
176 case 3:
177 addr = (addr << 8) | EXTRACT_U_1(cp);
178 cp++;
179 break;
180 }
181 addr |= net;
182 length -= 4 - netlen;
183 if (length < 1)
184 goto trunc;
185 ND_TCHECK_1(cp);
186 distances = EXTRACT_U_1(cp);
187 cp++;
188 length--;
189 ND_PRINT((ndo, " %s %s ",
190 gateways < egp->egp_intgw ? "int" : "ext",
191 ipaddr_string(ndo, &addr)));
192
193 comma = "";
194 ND_PRINT((ndo, "("));
195 while (distances != 0) {
196 if (length < 2)
197 goto trunc;
198 ND_TCHECK_2(cp);
199 ND_PRINT((ndo, "%sd%d:", comma, EXTRACT_U_1(cp)));
200 cp++;
201 comma = ", ";
202 networks = EXTRACT_U_1(cp);
203 cp++;
204 length -= 2;
205 while (networks != 0) {
206 /* Pickup network number */
207 if (length < 1)
208 goto trunc;
209 ND_TCHECK_1(cp);
210 addr = ((uint32_t) EXTRACT_U_1(cp)) << 24;
211 cp++;
212 length--;
213 if (IN_CLASSB(addr)) {
214 if (length < 1)
215 goto trunc;
216 ND_TCHECK_1(cp);
217 addr |= ((uint32_t) EXTRACT_U_1(cp)) << 16;
218 cp++;
219 length--;
220 } else if (!IN_CLASSA(addr)) {
221 if (length < 2)
222 goto trunc;
223 ND_TCHECK_2(cp);
224 addr |= ((uint32_t) EXTRACT_U_1(cp)) << 16;
225 cp++;
226 addr |= ((uint32_t) EXTRACT_U_1(cp)) << 8;
227 cp++;
228 length -= 2;
229 }
230 ND_PRINT((ndo, " %s", ipaddr_string(ndo, &addr)));
231 networks--;
232 }
233 distances--;
234 }
235 ND_PRINT((ndo, ")"));
236 }
237 return;
238 trunc:
239 ND_PRINT((ndo, "[|]"));
240 }
241
242 void
243 egp_print(netdissect_options *ndo,
244 const uint8_t *bp, u_int length)
245 {
246 const struct egp_packet *egp;
247 int status;
248 int code;
249 int type;
250
251 egp = (const struct egp_packet *)bp;
252 if (length < sizeof(*egp) || !ND_TTEST(*egp)) {
253 ND_PRINT((ndo, "[|egp]"));
254 return;
255 }
256
257 if (!ndo->ndo_vflag) {
258 ND_PRINT((ndo, "EGPv%u, AS %u, seq %u, length %u",
259 egp->egp_version,
260 EXTRACT_BE_U_2(&egp->egp_as),
261 EXTRACT_BE_U_2(&egp->egp_sequence),
262 length));
263 return;
264 } else
265 ND_PRINT((ndo, "EGPv%u, length %u",
266 egp->egp_version,
267 length));
268
269 if (egp->egp_version != EGP_VERSION) {
270 ND_PRINT((ndo, "[version %d]", egp->egp_version));
271 return;
272 }
273
274 type = egp->egp_type;
275 code = egp->egp_code;
276 status = egp->egp_status;
277
278 switch (type) {
279 case EGPT_ACQUIRE:
280 ND_PRINT((ndo, " acquire"));
281 switch (code) {
282 case EGPC_REQUEST:
283 case EGPC_CONFIRM:
284 ND_PRINT((ndo, " %s", egp_acquire_codes[code]));
285 switch (status) {
286 case EGPS_UNSPEC:
287 case EGPS_ACTIVE:
288 case EGPS_PASSIVE:
289 ND_PRINT((ndo, " %s", egp_acquire_status[status]));
290 break;
291
292 default:
293 ND_PRINT((ndo, " [status %d]", status));
294 break;
295 }
296 ND_PRINT((ndo, " hello:%d poll:%d",
297 EXTRACT_BE_U_2(&egp->egp_hello),
298 EXTRACT_BE_U_2(&egp->egp_poll)));
299 break;
300
301 case EGPC_REFUSE:
302 case EGPC_CEASE:
303 case EGPC_CEASEACK:
304 ND_PRINT((ndo, " %s", egp_acquire_codes[code]));
305 switch (status ) {
306 case EGPS_UNSPEC:
307 case EGPS_NORES:
308 case EGPS_ADMIN:
309 case EGPS_GODOWN:
310 case EGPS_PARAM:
311 case EGPS_PROTO:
312 ND_PRINT((ndo, " %s", egp_acquire_status[status]));
313 break;
314
315 default:
316 ND_PRINT((ndo, "[status %d]", status));
317 break;
318 }
319 break;
320
321 default:
322 ND_PRINT((ndo, "[code %d]", code));
323 break;
324 }
325 break;
326
327 case EGPT_REACH:
328 switch (code) {
329
330 case EGPC_HELLO:
331 case EGPC_HEARDU:
332 ND_PRINT((ndo, " %s", egp_reach_codes[code]));
333 if (status <= EGPS_DOWN)
334 ND_PRINT((ndo, " state:%s", egp_status_updown[status]));
335 else
336 ND_PRINT((ndo, " [status %d]", status));
337 break;
338
339 default:
340 ND_PRINT((ndo, "[reach code %d]", code));
341 break;
342 }
343 break;
344
345 case EGPT_POLL:
346 ND_PRINT((ndo, " poll"));
347 if (egp->egp_status <= EGPS_DOWN)
348 ND_PRINT((ndo, " state:%s", egp_status_updown[status]));
349 else
350 ND_PRINT((ndo, " [status %d]", status));
351 ND_PRINT((ndo, " net:%s", ipaddr_string(ndo, &egp->egp_sourcenet)));
352 break;
353
354 case EGPT_UPDATE:
355 ND_PRINT((ndo, " update"));
356 if (status & EGPS_UNSOL) {
357 status &= ~EGPS_UNSOL;
358 ND_PRINT((ndo, " unsolicited"));
359 }
360 if (status <= EGPS_DOWN)
361 ND_PRINT((ndo, " state:%s", egp_status_updown[status]));
362 else
363 ND_PRINT((ndo, " [status %d]", status));
364 ND_PRINT((ndo, " %s int %d ext %d",
365 ipaddr_string(ndo, &egp->egp_sourcenet),
366 egp->egp_intgw,
367 egp->egp_extgw));
368 if (ndo->ndo_vflag)
369 egpnrprint(ndo, egp, length);
370 break;
371
372 case EGPT_ERROR:
373 ND_PRINT((ndo, " error"));
374 if (status <= EGPS_DOWN)
375 ND_PRINT((ndo, " state:%s", egp_status_updown[status]));
376 else
377 ND_PRINT((ndo, " [status %d]", status));
378
379 if (EXTRACT_BE_U_2(&egp->egp_reason) <= EGPR_UVERSION)
380 ND_PRINT((ndo, " %s", egp_reasons[EXTRACT_BE_U_2(&egp->egp_reason)]));
381 else
382 ND_PRINT((ndo, " [reason %d]", EXTRACT_BE_U_2(&egp->egp_reason)));
383 break;
384
385 default:
386 ND_PRINT((ndo, "[type %d]", type));
387 break;
388 }
389 }