print-rip.c revision 1.4.4.1 1 1.1 christos /*
2 1.1 christos * Copyright (c) 1989, 1990, 1991, 1993, 1994, 1996
3 1.1 christos * The Regents of the University of California. All rights reserved.
4 1.1 christos *
5 1.1 christos * Redistribution and use in source and binary forms, with or without
6 1.1 christos * modification, are permitted provided that: (1) source code distributions
7 1.1 christos * retain the above copyright notice and this paragraph in its entirety, (2)
8 1.1 christos * distributions including binary code include the above copyright notice and
9 1.1 christos * this paragraph in its entirety in the documentation or other materials
10 1.1 christos * provided with the distribution, and (3) all advertising materials mentioning
11 1.1 christos * features or use of this software display the following acknowledgement:
12 1.1 christos * ``This product includes software developed by the University of California,
13 1.1 christos * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
14 1.1 christos * the University nor the names of its contributors may be used to endorse
15 1.1 christos * or promote products derived from this software without specific prior
16 1.1 christos * written permission.
17 1.1 christos * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
18 1.1 christos * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
19 1.1 christos * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
20 1.1 christos */
21 1.1 christos
22 1.2 christos #include <sys/cdefs.h>
23 1.1 christos #ifndef lint
24 1.4.4.1 snj __RCSID("$NetBSD: print-rip.c,v 1.4.4.1 2017/02/19 05:01:16 snj Exp $");
25 1.1 christos #endif
26 1.1 christos
27 1.4.4.1 snj /* \summary: Routing Information Protocol (RIP) printer */
28 1.4.4.1 snj
29 1.1 christos #ifdef HAVE_CONFIG_H
30 1.1 christos #include "config.h"
31 1.1 christos #endif
32 1.1 christos
33 1.4.4.1 snj #include <netdissect-stdinc.h>
34 1.1 christos
35 1.1 christos #include <stdio.h>
36 1.1 christos
37 1.4.4.1 snj #include "netdissect.h"
38 1.1 christos #include "addrtoname.h"
39 1.4.4.1 snj #include "extract.h"
40 1.1 christos
41 1.1 christos #include "af.h"
42 1.1 christos
43 1.4.4.1 snj static const char tstr[] = "[|rip]";
44 1.4.4.1 snj
45 1.1 christos struct rip {
46 1.4.4.1 snj uint8_t rip_cmd; /* request/response */
47 1.4.4.1 snj uint8_t rip_vers; /* protocol version # */
48 1.4.4.1 snj uint8_t unused[2]; /* unused */
49 1.1 christos };
50 1.1 christos
51 1.1 christos #define RIPCMD_REQUEST 1 /* want info */
52 1.1 christos #define RIPCMD_RESPONSE 2 /* responding to request */
53 1.1 christos #define RIPCMD_TRACEON 3 /* turn tracing on */
54 1.1 christos #define RIPCMD_TRACEOFF 4 /* turn it off */
55 1.1 christos #define RIPCMD_POLL 5 /* want info from everybody */
56 1.1 christos #define RIPCMD_POLLENTRY 6 /* poll for entry */
57 1.1 christos
58 1.1 christos static const struct tok rip_cmd_values[] = {
59 1.1 christos { RIPCMD_REQUEST, "Request" },
60 1.1 christos { RIPCMD_RESPONSE, "Response" },
61 1.1 christos { RIPCMD_TRACEON, "Trace on" },
62 1.1 christos { RIPCMD_TRACEOFF, "Trace off" },
63 1.1 christos { RIPCMD_POLL, "Poll" },
64 1.1 christos { RIPCMD_POLLENTRY, "Poll Entry" },
65 1.1 christos { 0, NULL}
66 1.1 christos };
67 1.1 christos
68 1.1 christos #define RIP_AUTHLEN 16
69 1.1 christos #define RIP_ROUTELEN 20
70 1.1 christos
71 1.1 christos /*
72 1.1 christos * rfc 1723
73 1.4.4.1 snj *
74 1.1 christos * 0 1 2 3 3
75 1.1 christos * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
76 1.1 christos * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
77 1.1 christos * | Command (1) | Version (1) | unused |
78 1.1 christos * +---------------+---------------+-------------------------------+
79 1.1 christos * | Address Family Identifier (2) | Route Tag (2) |
80 1.1 christos * +-------------------------------+-------------------------------+
81 1.1 christos * | IP Address (4) |
82 1.1 christos * +---------------------------------------------------------------+
83 1.1 christos * | Subnet Mask (4) |
84 1.1 christos * +---------------------------------------------------------------+
85 1.1 christos * | Next Hop (4) |
86 1.1 christos * +---------------------------------------------------------------+
87 1.1 christos * | Metric (4) |
88 1.1 christos * +---------------------------------------------------------------+
89 1.1 christos *
90 1.1 christos */
91 1.1 christos
92 1.1 christos struct rip_netinfo {
93 1.4.4.1 snj uint16_t rip_family;
94 1.4.4.1 snj uint16_t rip_tag;
95 1.4.4.1 snj uint32_t rip_dest;
96 1.4.4.1 snj uint32_t rip_dest_mask;
97 1.4.4.1 snj uint32_t rip_router;
98 1.4.4.1 snj uint32_t rip_metric; /* cost of route */
99 1.1 christos };
100 1.1 christos
101 1.1 christos static void
102 1.4.4.1 snj rip_entry_print_v1(netdissect_options *ndo,
103 1.4.4.1 snj register const struct rip_netinfo *ni)
104 1.1 christos {
105 1.1 christos register u_short family;
106 1.1 christos
107 1.1 christos /* RFC 1058 */
108 1.1 christos family = EXTRACT_16BITS(&ni->rip_family);
109 1.4 christos if (family != BSD_AFNUM_INET && family != 0) {
110 1.4.4.1 snj ND_PRINT((ndo, "\n\t AFI %s, ", tok2str(bsd_af_values, "Unknown (%u)", family)));
111 1.4.4.1 snj print_unknown_data(ndo, (const uint8_t *)&ni->rip_family, "\n\t ", RIP_ROUTELEN);
112 1.1 christos return;
113 1.1 christos }
114 1.1 christos if (EXTRACT_16BITS(&ni->rip_tag) ||
115 1.1 christos EXTRACT_32BITS(&ni->rip_dest_mask) ||
116 1.1 christos EXTRACT_32BITS(&ni->rip_router)) {
117 1.1 christos /* MBZ fields not zero */
118 1.4.4.1 snj print_unknown_data(ndo, (const uint8_t *)&ni->rip_family, "\n\t ", RIP_ROUTELEN);
119 1.1 christos return;
120 1.4 christos }
121 1.4 christos if (family == 0) {
122 1.4.4.1 snj ND_PRINT((ndo, "\n\t AFI 0, %s, metric: %u",
123 1.4.4.1 snj ipaddr_string(ndo, &ni->rip_dest),
124 1.4.4.1 snj EXTRACT_32BITS(&ni->rip_metric)));
125 1.4 christos return;
126 1.1 christos } /* BSD_AFNUM_INET */
127 1.4.4.1 snj ND_PRINT((ndo, "\n\t %s, metric: %u",
128 1.4.4.1 snj ipaddr_string(ndo, &ni->rip_dest),
129 1.4.4.1 snj EXTRACT_32BITS(&ni->rip_metric)));
130 1.1 christos }
131 1.1 christos
132 1.4 christos static unsigned
133 1.4.4.1 snj rip_entry_print_v2(netdissect_options *ndo,
134 1.4.4.1 snj register const struct rip_netinfo *ni, const unsigned remaining)
135 1.1 christos {
136 1.1 christos register u_short family;
137 1.1 christos
138 1.1 christos family = EXTRACT_16BITS(&ni->rip_family);
139 1.4 christos if (family == 0xFFFF) { /* variable-sized authentication structures */
140 1.4.4.1 snj uint16_t auth_type = EXTRACT_16BITS(&ni->rip_tag);
141 1.4 christos if (auth_type == 2) {
142 1.4.4.1 snj register const u_char *p = (const u_char *)&ni->rip_dest;
143 1.4 christos u_int i = 0;
144 1.4.4.1 snj ND_PRINT((ndo, "\n\t Simple Text Authentication data: "));
145 1.4 christos for (; i < RIP_AUTHLEN; p++, i++)
146 1.4.4.1 snj ND_PRINT((ndo, "%c", ND_ISPRINT(*p) ? *p : '.'));
147 1.4 christos } else if (auth_type == 3) {
148 1.4.4.1 snj ND_PRINT((ndo, "\n\t Auth header:"));
149 1.4.4.1 snj ND_PRINT((ndo, " Packet Len %u,", EXTRACT_16BITS((const uint8_t *)ni + 4)));
150 1.4.4.1 snj ND_PRINT((ndo, " Key-ID %u,", *((const uint8_t *)ni + 6)));
151 1.4.4.1 snj ND_PRINT((ndo, " Auth Data Len %u,", *((const uint8_t *)ni + 7)));
152 1.4.4.1 snj ND_PRINT((ndo, " SeqNo %u,", EXTRACT_32BITS(&ni->rip_dest_mask)));
153 1.4.4.1 snj ND_PRINT((ndo, " MBZ %u,", EXTRACT_32BITS(&ni->rip_router)));
154 1.4.4.1 snj ND_PRINT((ndo, " MBZ %u", EXTRACT_32BITS(&ni->rip_metric)));
155 1.4 christos } else if (auth_type == 1) {
156 1.4.4.1 snj ND_PRINT((ndo, "\n\t Auth trailer:"));
157 1.4.4.1 snj print_unknown_data(ndo, (const uint8_t *)&ni->rip_dest, "\n\t ", remaining);
158 1.4 christos return remaining; /* AT spans till the packet end */
159 1.4.4.1 snj } else {
160 1.4.4.1 snj ND_PRINT((ndo, "\n\t Unknown (%u) Authentication data:",
161 1.4.4.1 snj EXTRACT_16BITS(&ni->rip_tag)));
162 1.4.4.1 snj print_unknown_data(ndo, (const uint8_t *)&ni->rip_dest, "\n\t ", remaining);
163 1.1 christos }
164 1.4 christos } else if (family != BSD_AFNUM_INET && family != 0) {
165 1.4.4.1 snj ND_PRINT((ndo, "\n\t AFI %s", tok2str(bsd_af_values, "Unknown (%u)", family)));
166 1.4.4.1 snj print_unknown_data(ndo, (const uint8_t *)&ni->rip_tag, "\n\t ", RIP_ROUTELEN-2);
167 1.4 christos } else { /* BSD_AFNUM_INET or AFI 0 */
168 1.4.4.1 snj ND_PRINT((ndo, "\n\t AFI %s, %15s/%-2d, tag 0x%04x, metric: %u, next-hop: ",
169 1.4 christos tok2str(bsd_af_values, "%u", family),
170 1.4.4.1 snj ipaddr_string(ndo, &ni->rip_dest),
171 1.1 christos mask2plen(EXTRACT_32BITS(&ni->rip_dest_mask)),
172 1.1 christos EXTRACT_16BITS(&ni->rip_tag),
173 1.4.4.1 snj EXTRACT_32BITS(&ni->rip_metric)));
174 1.1 christos if (EXTRACT_32BITS(&ni->rip_router))
175 1.4.4.1 snj ND_PRINT((ndo, "%s", ipaddr_string(ndo, &ni->rip_router)));
176 1.4.4.1 snj else
177 1.4.4.1 snj ND_PRINT((ndo, "self"));
178 1.1 christos }
179 1.4 christos return sizeof (*ni);
180 1.1 christos }
181 1.1 christos
182 1.1 christos void
183 1.4.4.1 snj rip_print(netdissect_options *ndo,
184 1.4.4.1 snj const u_char *dat, u_int length)
185 1.1 christos {
186 1.1 christos register const struct rip *rp;
187 1.1 christos register const struct rip_netinfo *ni;
188 1.1 christos register u_int i, j;
189 1.1 christos
190 1.4.4.1 snj if (ndo->ndo_snapend < dat) {
191 1.4.4.1 snj ND_PRINT((ndo, " %s", tstr));
192 1.1 christos return;
193 1.1 christos }
194 1.4.4.1 snj i = ndo->ndo_snapend - dat;
195 1.1 christos if (i > length)
196 1.1 christos i = length;
197 1.1 christos if (i < sizeof(*rp)) {
198 1.4.4.1 snj ND_PRINT((ndo, " %s", tstr));
199 1.1 christos return;
200 1.1 christos }
201 1.1 christos i -= sizeof(*rp);
202 1.1 christos
203 1.4.4.1 snj rp = (const struct rip *)dat;
204 1.1 christos
205 1.4.4.1 snj ND_PRINT((ndo, "%sRIPv%u",
206 1.4.4.1 snj (ndo->ndo_vflag >= 1) ? "\n\t" : "",
207 1.4.4.1 snj rp->rip_vers));
208 1.1 christos
209 1.1 christos switch (rp->rip_vers) {
210 1.1 christos case 0:
211 1.1 christos /*
212 1.1 christos * RFC 1058.
213 1.1 christos *
214 1.1 christos * XXX - RFC 1058 says
215 1.1 christos *
216 1.1 christos * 0 Datagrams whose version number is zero are to be ignored.
217 1.1 christos * These are from a previous version of the protocol, whose
218 1.1 christos * packet format was machine-specific.
219 1.1 christos *
220 1.1 christos * so perhaps we should just dump the packet, in hex.
221 1.1 christos */
222 1.4.4.1 snj print_unknown_data(ndo, (const uint8_t *)&rp->rip_cmd, "\n\t", length);
223 1.1 christos break;
224 1.1 christos default:
225 1.1 christos /* dump version and lets see if we know the commands name*/
226 1.4.4.1 snj ND_PRINT((ndo, ", %s, length: %u",
227 1.1 christos tok2str(rip_cmd_values,
228 1.1 christos "unknown command (%u)",
229 1.1 christos rp->rip_cmd),
230 1.4.4.1 snj length));
231 1.1 christos
232 1.4.4.1 snj if (ndo->ndo_vflag < 1)
233 1.1 christos return;
234 1.1 christos
235 1.1 christos switch (rp->rip_cmd) {
236 1.4 christos case RIPCMD_REQUEST:
237 1.1 christos case RIPCMD_RESPONSE:
238 1.1 christos j = length / sizeof(*ni);
239 1.4.4.1 snj ND_PRINT((ndo, ", routes: %u%s", j, rp->rip_vers == 2 ? " or less" : ""));
240 1.4.4.1 snj ni = (const struct rip_netinfo *)(rp + 1);
241 1.1 christos for (; i >= sizeof(*ni); ++ni) {
242 1.1 christos if (rp->rip_vers == 1)
243 1.4 christos {
244 1.4.4.1 snj rip_entry_print_v1(ndo, ni);
245 1.4 christos i -= sizeof(*ni);
246 1.4 christos }
247 1.1 christos else if (rp->rip_vers == 2)
248 1.4.4.1 snj i -= rip_entry_print_v2(ndo, ni, i);
249 1.1 christos else
250 1.1 christos break;
251 1.1 christos }
252 1.4 christos if (i)
253 1.4.4.1 snj ND_PRINT((ndo, "%s", tstr));
254 1.1 christos break;
255 1.1 christos
256 1.1 christos case RIPCMD_TRACEOFF:
257 1.1 christos case RIPCMD_POLL:
258 1.1 christos case RIPCMD_POLLENTRY:
259 1.1 christos break;
260 1.1 christos
261 1.1 christos case RIPCMD_TRACEON:
262 1.1 christos /* fall through */
263 1.1 christos default:
264 1.4.4.1 snj if (ndo->ndo_vflag <= 1) {
265 1.4.4.1 snj if(!print_unknown_data(ndo, (const uint8_t *)rp, "\n\t", length))
266 1.1 christos return;
267 1.1 christos }
268 1.1 christos break;
269 1.1 christos }
270 1.1 christos /* do we want to see an additionally hexdump ? */
271 1.4.4.1 snj if (ndo->ndo_vflag> 1) {
272 1.4.4.1 snj if(!print_unknown_data(ndo, (const uint8_t *)rp, "\n\t", length))
273 1.1 christos return;
274 1.1 christos }
275 1.1 christos }
276 1.1 christos }
277 1.1 christos
278 1.1 christos
279