blob: e095ef71262eaf36888c9f2b8f47cd487d6ab3bd [file] [log] [blame]
hannescb08bf92007-03-27 10:44:09 +00001/*
2 * Copyright (c) 1998-2007 The TCPDUMP project
Florian Forster289e0ae2009-05-17 00:05:30 +02003 * Copyright (c) 2009 Florian Forster
hannescb08bf92007-03-27 10:44:09 +00004 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that: (1) source code
7 * distributions retain the above copyright notice and this paragraph
8 * in its entirety, and (2) distributions including binary code include
9 * the above copyright notice and this paragraph in its entirety in
10 * the documentation or other materials provided with the distribution.
11 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND
12 * WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, WITHOUT
13 * LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
14 * FOR A PARTICULAR PURPOSE.
15 *
hannescb08bf92007-03-27 10:44:09 +000016 * Original code by Hannes Gredler <hannes@juniper.net>
Florian Forster289e0ae2009-05-17 00:05:30 +020017 * IPv6 additions by Florian Forster <octo at verplant.org>
hannescb08bf92007-03-27 10:44:09 +000018 */
19
Francois-Xavier Le Bailb38f3242016-08-15 15:27:28 +020020/* \summary: Optimized Link State Routing Protocol (OLSR) printer */
21
22/* specification: RFC 3626 */
23
hannescb08bf92007-03-27 10:44:09 +000024#ifdef HAVE_CONFIG_H
25#include "config.h"
26#endif
27
Francois-Xavier Le Bail99c91c32015-09-09 22:42:38 +020028#include <netdissect-stdinc.h>
hannescb08bf92007-03-27 10:44:09 +000029
Francois-Xavier Le Bailc1c3c772015-09-05 23:35:58 +020030#include "netdissect.h"
hannescb08bf92007-03-27 10:44:09 +000031#include "addrtoname.h"
Guy Harrisb0dc4a52009-05-21 10:14:22 -070032#include "extract.h"
hannescb08bf92007-03-27 10:44:09 +000033
34/*
35 * RFC 3626 common header
36 *
37 * 0 1 2 3
38 * 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
39 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
40 * | Packet Length | Packet Sequence Number |
41 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
42 * | Message Type | Vtime | Message Size |
43 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
44 * | Originator Address |
45 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
46 * | Time To Live | Hop Count | Message Sequence Number |
47 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
48 * | |
49 * : MESSAGE :
50 * | |
51 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
52 * | Message Type | Vtime | Message Size |
53 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
54 * | Originator Address |
55 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
56 * | Time To Live | Hop Count | Message Sequence Number |
57 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
58 * | |
59 * : MESSAGE :
60 * | |
61 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
62 * : :
63 */
64
65struct olsr_common {
Guy Harrised85e202014-04-23 00:20:40 -070066 uint8_t packet_len[2];
67 uint8_t packet_seq[2];
hannescb08bf92007-03-27 10:44:09 +000068};
69
70#define OLSR_HELLO_MSG 1 /* rfc3626 */
71#define OLSR_TC_MSG 2 /* rfc3626 */
72#define OLSR_MID_MSG 3 /* rfc3626 */
73#define OLSR_HNA_MSG 4 /* rfc3626 */
74#define OLSR_POWERINFO_MSG 128
75#define OLSR_NAMESERVICE_MSG 130
76#define OLSR_HELLO_LQ_MSG 201 /* LQ extensions olsr.org */
77#define OLSR_TC_LQ_MSG 202 /* LQ extensions olsr.org */
78
Denis Ovsienko7de35592013-09-24 20:46:24 +040079static const struct tok olsr_msg_values[] = {
hannescb08bf92007-03-27 10:44:09 +000080 { OLSR_HELLO_MSG, "Hello" },
81 { OLSR_TC_MSG, "TC" },
82 { OLSR_MID_MSG, "MID" },
83 { OLSR_HNA_MSG, "HNA" },
84 { OLSR_POWERINFO_MSG, "Powerinfo" },
85 { OLSR_NAMESERVICE_MSG, "Nameservice" },
86 { OLSR_HELLO_LQ_MSG, "Hello-LQ" },
87 { OLSR_TC_LQ_MSG, "TC-LQ" },
88 { 0, NULL}
89};
90
Florian Forster289e0ae2009-05-17 00:05:30 +020091struct olsr_msg4 {
Guy Harrised85e202014-04-23 00:20:40 -070092 uint8_t msg_type;
93 uint8_t vtime;
94 uint8_t msg_len[2];
95 uint8_t originator[4];
96 uint8_t ttl;
97 uint8_t hopcount;
98 uint8_t msg_seq[2];
hannescb08bf92007-03-27 10:44:09 +000099};
100
Florian Forster289e0ae2009-05-17 00:05:30 +0200101struct olsr_msg6 {
Guy Harrised85e202014-04-23 00:20:40 -0700102 uint8_t msg_type;
103 uint8_t vtime;
104 uint8_t msg_len[2];
105 uint8_t originator[16];
106 uint8_t ttl;
107 uint8_t hopcount;
108 uint8_t msg_seq[2];
Florian Forster289e0ae2009-05-17 00:05:30 +0200109};
110
hannes1716c032007-03-28 07:34:55 +0000111struct olsr_hello {
Guy Harrised85e202014-04-23 00:20:40 -0700112 uint8_t res[2];
113 uint8_t htime;
114 uint8_t will;
hannescb08bf92007-03-27 10:44:09 +0000115};
116
hannes1716c032007-03-28 07:34:55 +0000117struct olsr_hello_link {
Guy Harrised85e202014-04-23 00:20:40 -0700118 uint8_t link_code;
119 uint8_t res;
120 uint8_t len[2];
hannescb08bf92007-03-27 10:44:09 +0000121};
122
hannes1716c032007-03-28 07:34:55 +0000123struct olsr_tc {
Guy Harrised85e202014-04-23 00:20:40 -0700124 uint8_t ans_seq[2];
125 uint8_t res[2];
hannes1716c032007-03-28 07:34:55 +0000126};
127
Florian Forster289e0ae2009-05-17 00:05:30 +0200128struct olsr_hna4 {
Guy Harrised85e202014-04-23 00:20:40 -0700129 uint8_t network[4];
130 uint8_t mask[4];
hannes1716c032007-03-28 07:34:55 +0000131};
132
Florian Forster289e0ae2009-05-17 00:05:30 +0200133struct olsr_hna6 {
Guy Harrised85e202014-04-23 00:20:40 -0700134 uint8_t network[16];
135 uint8_t mask[16];
Florian Forster289e0ae2009-05-17 00:05:30 +0200136};
137
hannes1716c032007-03-28 07:34:55 +0000138
Ferry Huberts4b4106c2015-04-24 14:43:58 +0200139/** gateway HNA flags */
140enum gateway_hna_flags {
141 GW_HNA_FLAG_LINKSPEED = 1 << 0,
142 GW_HNA_FLAG_IPV4 = 1 << 1,
143 GW_HNA_FLAG_IPV4_NAT = 1 << 2,
144 GW_HNA_FLAG_IPV6 = 1 << 3,
145 GW_HNA_FLAG_IPV6PREFIX = 1 << 4
146};
147
148/** gateway HNA field byte offsets in the netmask field of the HNA */
149enum gateway_hna_fields {
150 GW_HNA_PAD = 0,
151 GW_HNA_FLAGS = 1,
152 GW_HNA_UPLINK = 2,
153 GW_HNA_DOWNLINK = 3,
154 GW_HNA_V6PREFIXLEN = 4,
155 GW_HNA_V6PREFIX = 5
156};
157
158
hannescb08bf92007-03-27 10:44:09 +0000159#define OLSR_EXTRACT_LINK_TYPE(link_code) (link_code & 0x3)
160#define OLSR_EXTRACT_NEIGHBOR_TYPE(link_code) (link_code >> 2)
161
Denis Ovsienko7de35592013-09-24 20:46:24 +0400162static const struct tok olsr_link_type_values[] = {
hannescb08bf92007-03-27 10:44:09 +0000163 { 0, "Unspecified" },
164 { 1, "Asymmetric" },
165 { 2, "Symmetric" },
166 { 3, "Lost" },
167 { 0, NULL}
168};
169
Denis Ovsienko7de35592013-09-24 20:46:24 +0400170static const struct tok olsr_neighbor_type_values[] = {
hannescb08bf92007-03-27 10:44:09 +0000171 { 0, "Not-Neighbor" },
172 { 1, "Symmetric" },
173 { 2, "Symmetric-MPR" },
174 { 0, NULL}
175};
176
Florian Forster289e0ae2009-05-17 00:05:30 +0200177struct olsr_lq_neighbor4 {
Guy Harrised85e202014-04-23 00:20:40 -0700178 uint8_t neighbor[4];
179 uint8_t link_quality;
180 uint8_t neighbor_link_quality;
181 uint8_t res[2];
hannescb08bf92007-03-27 10:44:09 +0000182};
183
Florian Forster289e0ae2009-05-17 00:05:30 +0200184struct olsr_lq_neighbor6 {
Guy Harrised85e202014-04-23 00:20:40 -0700185 uint8_t neighbor[16];
186 uint8_t link_quality;
187 uint8_t neighbor_link_quality;
188 uint8_t res[2];
Florian Forster289e0ae2009-05-17 00:05:30 +0200189};
190
Ferry Huberts4b4106c2015-04-24 14:43:58 +0200191#define MAX_SMARTGW_SPEED 320000000
192
193/**
194 * Convert an encoded 1 byte transport value (5 bits mantissa, 3 bits exponent)
195 * to an uplink/downlink speed value
196 *
197 * @param value the encoded 1 byte transport value
198 * @return the uplink/downlink speed value (in kbit/s)
199 */
200static uint32_t deserialize_gw_speed(uint8_t value) {
201 uint32_t speed;
202 uint32_t exp;
203
204 if (!value) {
205 return 0;
206 }
207
208 if (value == UINT8_MAX) {
209 /* maximum value: also return maximum value */
210 return MAX_SMARTGW_SPEED;
211 }
212
213 speed = (value >> 3) + 1;
214 exp = value & 7;
215
216 while (exp-- > 0) {
217 speed *= 10;
218 }
219 return speed;
220}
221
hannescb08bf92007-03-27 10:44:09 +0000222/*
223 * macro to convert the 8-bit mantissa/exponent to a double float
hannes1716c032007-03-28 07:34:55 +0000224 * taken from olsr.org.
hannescb08bf92007-03-27 10:44:09 +0000225 */
226#define VTIME_SCALE_FACTOR 0.0625
227#define ME_TO_DOUBLE(me) \
228 (double)(VTIME_SCALE_FACTOR*(1+(double)(me>>4)/16)*(double)(1<<(me&0x0F)))
229
hannes1716c032007-03-28 07:34:55 +0000230/*
231 * print a neighbor list with LQ extensions.
232 */
Guy Harris4038f832014-11-11 16:49:39 -0800233static int
Denis Ovsienko62a51842014-03-27 14:58:23 +0400234olsr_print_lq_neighbor4(netdissect_options *ndo,
235 const u_char *msg_data, u_int hello_len)
hannes1716c032007-03-28 07:34:55 +0000236{
Guy Harris69cb46a2015-04-26 17:24:42 -0700237 const struct olsr_lq_neighbor4 *lq_neighbor;
hannes1716c032007-03-28 07:34:55 +0000238
Florian Forster289e0ae2009-05-17 00:05:30 +0200239 while (hello_len >= sizeof(struct olsr_lq_neighbor4)) {
hannes1716c032007-03-28 07:34:55 +0000240
Guy Harris69cb46a2015-04-26 17:24:42 -0700241 lq_neighbor = (const struct olsr_lq_neighbor4 *)msg_data;
Guy Harris4038f832014-11-11 16:49:39 -0800242 if (!ND_TTEST(*lq_neighbor))
243 return (-1);
hannes1716c032007-03-28 07:34:55 +0000244
Francois-Xavier Le Bail568866a2015-08-18 18:15:45 +0200245 ND_PRINT((ndo, "\n\t neighbor %s, link-quality %.2f%%"
246 ", neighbor-link-quality %.2f%%",
Guy Harris1cde6432014-04-04 00:43:46 -0700247 ipaddr_string(ndo, lq_neighbor->neighbor),
hannes1716c032007-03-28 07:34:55 +0000248 ((double)lq_neighbor->link_quality/2.55),
Denis Ovsienko62a51842014-03-27 14:58:23 +0400249 ((double)lq_neighbor->neighbor_link_quality/2.55)));
hannes1716c032007-03-28 07:34:55 +0000250
Florian Forster289e0ae2009-05-17 00:05:30 +0200251 msg_data += sizeof(struct olsr_lq_neighbor4);
252 hello_len -= sizeof(struct olsr_lq_neighbor4);
hannes1716c032007-03-28 07:34:55 +0000253 }
Guy Harris4038f832014-11-11 16:49:39 -0800254 return (0);
hannes1716c032007-03-28 07:34:55 +0000255}
256
Guy Harris4038f832014-11-11 16:49:39 -0800257static int
Denis Ovsienko62a51842014-03-27 14:58:23 +0400258olsr_print_lq_neighbor6(netdissect_options *ndo,
259 const u_char *msg_data, u_int hello_len)
Florian Forster289e0ae2009-05-17 00:05:30 +0200260{
Guy Harris69cb46a2015-04-26 17:24:42 -0700261 const struct olsr_lq_neighbor6 *lq_neighbor;
Florian Forster289e0ae2009-05-17 00:05:30 +0200262
263 while (hello_len >= sizeof(struct olsr_lq_neighbor6)) {
264
Guy Harris69cb46a2015-04-26 17:24:42 -0700265 lq_neighbor = (const struct olsr_lq_neighbor6 *)msg_data;
Guy Harris4038f832014-11-11 16:49:39 -0800266 if (!ND_TTEST(*lq_neighbor))
267 return (-1);
Florian Forster289e0ae2009-05-17 00:05:30 +0200268
Francois-Xavier Le Bail568866a2015-08-18 18:15:45 +0200269 ND_PRINT((ndo, "\n\t neighbor %s, link-quality %.2f%%"
270 ", neighbor-link-quality %.2f%%",
Guy Harris1cde6432014-04-04 00:43:46 -0700271 ip6addr_string(ndo, lq_neighbor->neighbor),
Florian Forster289e0ae2009-05-17 00:05:30 +0200272 ((double)lq_neighbor->link_quality/2.55),
Denis Ovsienko62a51842014-03-27 14:58:23 +0400273 ((double)lq_neighbor->neighbor_link_quality/2.55)));
Florian Forster289e0ae2009-05-17 00:05:30 +0200274
275 msg_data += sizeof(struct olsr_lq_neighbor6);
276 hello_len -= sizeof(struct olsr_lq_neighbor6);
277 }
Guy Harris4038f832014-11-11 16:49:39 -0800278 return (0);
Florian Forster289e0ae2009-05-17 00:05:30 +0200279}
Florian Forster289e0ae2009-05-17 00:05:30 +0200280
hannes1716c032007-03-28 07:34:55 +0000281/*
282 * print a neighbor list.
283 */
Guy Harris4038f832014-11-11 16:49:39 -0800284static int
Denis Ovsienko62a51842014-03-27 14:58:23 +0400285olsr_print_neighbor(netdissect_options *ndo,
286 const u_char *msg_data, u_int hello_len)
hannes1716c032007-03-28 07:34:55 +0000287{
288 int neighbor;
289
Denis Ovsienko62a51842014-03-27 14:58:23 +0400290 ND_PRINT((ndo, "\n\t neighbor\n\t\t"));
hannes1716c032007-03-28 07:34:55 +0000291 neighbor = 1;
292
293 while (hello_len >= sizeof(struct in_addr)) {
294
Guy Harris4038f832014-11-11 16:49:39 -0800295 if (!ND_TTEST2(*msg_data, sizeof(struct in_addr)))
296 return (-1);
hannes1716c032007-03-28 07:34:55 +0000297 /* print 4 neighbors per line */
298
Guy Harris1cde6432014-04-04 00:43:46 -0700299 ND_PRINT((ndo, "%s%s", ipaddr_string(ndo, msg_data),
Denis Ovsienko62a51842014-03-27 14:58:23 +0400300 neighbor % 4 == 0 ? "\n\t\t" : " "));
hannes1716c032007-03-28 07:34:55 +0000301
302 msg_data += sizeof(struct in_addr);
303 hello_len -= sizeof(struct in_addr);
304 }
Guy Harris4038f832014-11-11 16:49:39 -0800305 return (0);
hannes1716c032007-03-28 07:34:55 +0000306}
307
308
hannescb08bf92007-03-27 10:44:09 +0000309void
Denis Ovsienko62a51842014-03-27 14:58:23 +0400310olsr_print(netdissect_options *ndo,
311 const u_char *pptr, u_int length, int is_ipv6)
hannescb08bf92007-03-27 10:44:09 +0000312{
313 union {
314 const struct olsr_common *common;
Florian Forster289e0ae2009-05-17 00:05:30 +0200315 const struct olsr_msg4 *msg4;
316 const struct olsr_msg6 *msg6;
hannes1716c032007-03-28 07:34:55 +0000317 const struct olsr_hello *hello;
318 const struct olsr_hello_link *hello_link;
hannes1716c032007-03-28 07:34:55 +0000319 const struct olsr_tc *tc;
Florian Forster289e0ae2009-05-17 00:05:30 +0200320 const struct olsr_hna4 *hna;
hannescb08bf92007-03-27 10:44:09 +0000321 } ptr;
322
Florian Forster289e0ae2009-05-17 00:05:30 +0200323 u_int msg_type, msg_len, msg_tlen, hello_len;
Guy Harrised85e202014-04-23 00:20:40 -0700324 uint16_t name_entry_type, name_entry_len;
Guy Harris429c8b42009-06-28 12:16:43 -0700325 u_int name_entry_padding;
Guy Harrised85e202014-04-23 00:20:40 -0700326 uint8_t link_type, neighbor_type;
hannescb08bf92007-03-27 10:44:09 +0000327 const u_char *tptr, *msg_data;
328
Guy Harrisb0dc4a52009-05-21 10:14:22 -0700329 tptr = pptr;
hannescb08bf92007-03-27 10:44:09 +0000330
331 if (length < sizeof(struct olsr_common)) {
332 goto trunc;
333 }
334
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400335 ND_TCHECK2(*tptr, sizeof(struct olsr_common));
hannescb08bf92007-03-27 10:44:09 +0000336
Guy Harris69cb46a2015-04-26 17:24:42 -0700337 ptr.common = (const struct olsr_common *)tptr;
Denis Ovsienkoec376152014-03-15 23:30:58 +0400338 length = min(length, EXTRACT_16BITS(ptr.common->packet_len));
hannescb08bf92007-03-27 10:44:09 +0000339
Denis Ovsienko62a51842014-03-27 14:58:23 +0400340 ND_PRINT((ndo, "OLSRv%i, seq 0x%04x, length %u",
Florian Forster289e0ae2009-05-17 00:05:30 +0200341 (is_ipv6 == 0) ? 4 : 6,
342 EXTRACT_16BITS(ptr.common->packet_seq),
Denis Ovsienko62a51842014-03-27 14:58:23 +0400343 length));
hannescb08bf92007-03-27 10:44:09 +0000344
345 tptr += sizeof(struct olsr_common);
346
347 /*
348 * In non-verbose mode, just print version.
349 */
Denis Ovsienko62a51842014-03-27 14:58:23 +0400350 if (ndo->ndo_vflag < 1) {
Florian Forster289e0ae2009-05-17 00:05:30 +0200351 return;
hannescb08bf92007-03-27 10:44:09 +0000352 }
353
354 while (tptr < (pptr+length)) {
Florian Forster289e0ae2009-05-17 00:05:30 +0200355 union
356 {
Guy Harris69cb46a2015-04-26 17:24:42 -0700357 const struct olsr_msg4 *v4;
358 const struct olsr_msg6 *v6;
Florian Forster289e0ae2009-05-17 00:05:30 +0200359 } msgptr;
360 int msg_len_valid = 0;
hannescb08bf92007-03-27 10:44:09 +0000361
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400362 ND_TCHECK2(*tptr, sizeof(struct olsr_msg4));
hannescb08bf92007-03-27 10:44:09 +0000363
Florian Forster289e0ae2009-05-17 00:05:30 +0200364 if (is_ipv6)
365 {
Guy Harris69cb46a2015-04-26 17:24:42 -0700366 msgptr.v6 = (const struct olsr_msg6 *) tptr;
Florian Forster289e0ae2009-05-17 00:05:30 +0200367 msg_type = msgptr.v6->msg_type;
368 msg_len = EXTRACT_16BITS(msgptr.v6->msg_len);
369 if ((msg_len >= sizeof (struct olsr_msg6))
370 && (msg_len <= length))
371 msg_len_valid = 1;
hannescb08bf92007-03-27 10:44:09 +0000372
Florian Forster289e0ae2009-05-17 00:05:30 +0200373 /* infinite loop check */
374 if (msg_type == 0 || msg_len == 0) {
375 return;
376 }
hannescb08bf92007-03-27 10:44:09 +0000377
Denis Ovsienko62a51842014-03-27 14:58:23 +0400378 ND_PRINT((ndo, "\n\t%s Message (%#04x), originator %s, ttl %u, hop %u"
Francois-Xavier Le Bail568866a2015-08-18 18:15:45 +0200379 "\n\t vtime %.3fs, msg-seq 0x%04x, length %u%s",
Florian Forster289e0ae2009-05-17 00:05:30 +0200380 tok2str(olsr_msg_values, "Unknown", msg_type),
Guy Harris1cde6432014-04-04 00:43:46 -0700381 msg_type, ip6addr_string(ndo, msgptr.v6->originator),
Florian Forster289e0ae2009-05-17 00:05:30 +0200382 msgptr.v6->ttl,
383 msgptr.v6->hopcount,
384 ME_TO_DOUBLE(msgptr.v6->vtime),
385 EXTRACT_16BITS(msgptr.v6->msg_seq),
Denis Ovsienko62a51842014-03-27 14:58:23 +0400386 msg_len, (msg_len_valid == 0) ? " (invalid)" : ""));
Guy Harris4038f832014-11-11 16:49:39 -0800387 if (!msg_len_valid) {
388 return;
389 }
Florian Forster289e0ae2009-05-17 00:05:30 +0200390
391 msg_tlen = msg_len - sizeof(struct olsr_msg6);
392 msg_data = tptr + sizeof(struct olsr_msg6);
hannescb08bf92007-03-27 10:44:09 +0000393 }
Florian Forster289e0ae2009-05-17 00:05:30 +0200394 else /* (!is_ipv6) */
Florian Forster289e0ae2009-05-17 00:05:30 +0200395 {
Guy Harris69cb46a2015-04-26 17:24:42 -0700396 msgptr.v4 = (const struct olsr_msg4 *) tptr;
Florian Forster289e0ae2009-05-17 00:05:30 +0200397 msg_type = msgptr.v4->msg_type;
398 msg_len = EXTRACT_16BITS(msgptr.v4->msg_len);
399 if ((msg_len >= sizeof (struct olsr_msg4))
400 && (msg_len <= length))
401 msg_len_valid = 1;
hannescb08bf92007-03-27 10:44:09 +0000402
Florian Forster289e0ae2009-05-17 00:05:30 +0200403 /* infinite loop check */
404 if (msg_type == 0 || msg_len == 0) {
405 return;
406 }
hannescb08bf92007-03-27 10:44:09 +0000407
Denis Ovsienko62a51842014-03-27 14:58:23 +0400408 ND_PRINT((ndo, "\n\t%s Message (%#04x), originator %s, ttl %u, hop %u"
Francois-Xavier Le Bail568866a2015-08-18 18:15:45 +0200409 "\n\t vtime %.3fs, msg-seq 0x%04x, length %u%s",
Florian Forster289e0ae2009-05-17 00:05:30 +0200410 tok2str(olsr_msg_values, "Unknown", msg_type),
Guy Harris1cde6432014-04-04 00:43:46 -0700411 msg_type, ipaddr_string(ndo, msgptr.v4->originator),
Florian Forster289e0ae2009-05-17 00:05:30 +0200412 msgptr.v4->ttl,
413 msgptr.v4->hopcount,
414 ME_TO_DOUBLE(msgptr.v4->vtime),
415 EXTRACT_16BITS(msgptr.v4->msg_seq),
Denis Ovsienko62a51842014-03-27 14:58:23 +0400416 msg_len, (msg_len_valid == 0) ? " (invalid)" : ""));
Guy Harris4038f832014-11-11 16:49:39 -0800417 if (!msg_len_valid) {
418 return;
419 }
Florian Forster289e0ae2009-05-17 00:05:30 +0200420
421 msg_tlen = msg_len - sizeof(struct olsr_msg4);
422 msg_data = tptr + sizeof(struct olsr_msg4);
423 }
hannescb08bf92007-03-27 10:44:09 +0000424
425 switch (msg_type) {
hannes1716c032007-03-28 07:34:55 +0000426 case OLSR_HELLO_MSG:
hannescb08bf92007-03-27 10:44:09 +0000427 case OLSR_HELLO_LQ_MSG:
Guy Harris4038f832014-11-11 16:49:39 -0800428 if (msg_tlen < sizeof(struct olsr_hello))
429 goto trunc;
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400430 ND_TCHECK2(*msg_data, sizeof(struct olsr_hello));
hannescb08bf92007-03-27 10:44:09 +0000431
Guy Harris69cb46a2015-04-26 17:24:42 -0700432 ptr.hello = (const struct olsr_hello *)msg_data;
Francois-Xavier Le Bail568866a2015-08-18 18:15:45 +0200433 ND_PRINT((ndo, "\n\t hello-time %.3fs, MPR willingness %u",
Denis Ovsienko62a51842014-03-27 14:58:23 +0400434 ME_TO_DOUBLE(ptr.hello->htime), ptr.hello->will));
hannes1716c032007-03-28 07:34:55 +0000435 msg_data += sizeof(struct olsr_hello);
436 msg_tlen -= sizeof(struct olsr_hello);
437
438 while (msg_tlen >= sizeof(struct olsr_hello_link)) {
Florian Forsterefb465c2009-05-16 13:01:52 +0200439 int hello_len_valid = 0;
hannescb08bf92007-03-27 10:44:09 +0000440
441 /*
442 * link-type.
443 */
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400444 ND_TCHECK2(*msg_data, sizeof(struct olsr_hello_link));
hannescb08bf92007-03-27 10:44:09 +0000445
Guy Harris69cb46a2015-04-26 17:24:42 -0700446 ptr.hello_link = (const struct olsr_hello_link *)msg_data;
hannes1716c032007-03-28 07:34:55 +0000447
448 hello_len = EXTRACT_16BITS(ptr.hello_link->len);
449 link_type = OLSR_EXTRACT_LINK_TYPE(ptr.hello_link->link_code);
450 neighbor_type = OLSR_EXTRACT_NEIGHBOR_TYPE(ptr.hello_link->link_code);
hannescb08bf92007-03-27 10:44:09 +0000451
Florian Forsterefb465c2009-05-16 13:01:52 +0200452 if ((hello_len <= msg_tlen)
453 && (hello_len >= sizeof(struct olsr_hello_link)))
454 hello_len_valid = 1;
455
Denis Ovsienko62a51842014-03-27 14:58:23 +0400456 ND_PRINT((ndo, "\n\t link-type %s, neighbor-type %s, len %u%s",
hannescb08bf92007-03-27 10:44:09 +0000457 tok2str(olsr_link_type_values, "Unknown", link_type),
458 tok2str(olsr_neighbor_type_values, "Unknown", neighbor_type),
Florian Forsterefb465c2009-05-16 13:01:52 +0200459 hello_len,
Denis Ovsienko62a51842014-03-27 14:58:23 +0400460 (hello_len_valid == 0) ? " (invalid)" : ""));
Florian Forsterefb465c2009-05-16 13:01:52 +0200461
462 if (hello_len_valid == 0)
463 break;
hannescb08bf92007-03-27 10:44:09 +0000464
hannes1716c032007-03-28 07:34:55 +0000465 msg_data += sizeof(struct olsr_hello_link);
466 msg_tlen -= sizeof(struct olsr_hello_link);
467 hello_len -= sizeof(struct olsr_hello_link);
hannescb08bf92007-03-27 10:44:09 +0000468
Guy Harris4038f832014-11-11 16:49:39 -0800469 ND_TCHECK2(*msg_data, hello_len);
hannes1716c032007-03-28 07:34:55 +0000470 if (msg_type == OLSR_HELLO_MSG) {
Guy Harris4038f832014-11-11 16:49:39 -0800471 if (olsr_print_neighbor(ndo, msg_data, hello_len) == -1)
472 goto trunc;
hannes1716c032007-03-28 07:34:55 +0000473 } else {
Guy Harris4038f832014-11-11 16:49:39 -0800474 if (is_ipv6) {
475 if (olsr_print_lq_neighbor6(ndo, msg_data, hello_len) == -1)
476 goto trunc;
Guy Harris11f73ad2015-09-17 14:56:44 -0700477 } else {
Guy Harris4038f832014-11-11 16:49:39 -0800478 if (olsr_print_lq_neighbor4(ndo, msg_data, hello_len) == -1)
479 goto trunc;
480 }
hannescb08bf92007-03-27 10:44:09 +0000481 }
hannes1716c032007-03-28 07:34:55 +0000482
483 msg_data += hello_len;
484 msg_tlen -= hello_len;
485 }
486 break;
487
488 case OLSR_TC_MSG:
489 case OLSR_TC_LQ_MSG:
Guy Harris4038f832014-11-11 16:49:39 -0800490 if (msg_tlen < sizeof(struct olsr_tc))
491 goto trunc;
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400492 ND_TCHECK2(*msg_data, sizeof(struct olsr_tc));
hannes1716c032007-03-28 07:34:55 +0000493
Guy Harris69cb46a2015-04-26 17:24:42 -0700494 ptr.tc = (const struct olsr_tc *)msg_data;
Denis Ovsienko62a51842014-03-27 14:58:23 +0400495 ND_PRINT((ndo, "\n\t advertised neighbor seq 0x%04x",
496 EXTRACT_16BITS(ptr.tc->ans_seq)));
hannes1716c032007-03-28 07:34:55 +0000497 msg_data += sizeof(struct olsr_tc);
498 msg_tlen -= sizeof(struct olsr_tc);
499
500 if (msg_type == OLSR_TC_MSG) {
Guy Harris4038f832014-11-11 16:49:39 -0800501 if (olsr_print_neighbor(ndo, msg_data, msg_tlen) == -1)
502 goto trunc;
hannes1716c032007-03-28 07:34:55 +0000503 } else {
Guy Harris4038f832014-11-11 16:49:39 -0800504 if (is_ipv6) {
505 if (olsr_print_lq_neighbor6(ndo, msg_data, msg_tlen) == -1)
506 goto trunc;
Guy Harris11f73ad2015-09-17 14:56:44 -0700507 } else {
Guy Harris4038f832014-11-11 16:49:39 -0800508 if (olsr_print_lq_neighbor4(ndo, msg_data, msg_tlen) == -1)
509 goto trunc;
510 }
hannes1716c032007-03-28 07:34:55 +0000511 }
512 break;
513
514 case OLSR_MID_MSG:
Florian Forster289e0ae2009-05-17 00:05:30 +0200515 {
516 size_t addr_size = sizeof(struct in_addr);
517
Florian Forster289e0ae2009-05-17 00:05:30 +0200518 if (is_ipv6)
519 addr_size = sizeof(struct in6_addr);
Florian Forster289e0ae2009-05-17 00:05:30 +0200520
Florian Forster289e0ae2009-05-17 00:05:30 +0200521 while (msg_tlen >= addr_size) {
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400522 ND_TCHECK2(*msg_data, addr_size);
Gisle Vanema8ace282014-04-03 09:33:25 +0400523 ND_PRINT((ndo, "\n\t interface address %s",
Guy Harris1cde6432014-04-04 00:43:46 -0700524 is_ipv6 ? ip6addr_string(ndo, msg_data) :
525 ipaddr_string(ndo, msg_data)));
Gisle Vanema8ace282014-04-03 09:33:25 +0400526
Florian Forster289e0ae2009-05-17 00:05:30 +0200527 msg_data += addr_size;
528 msg_tlen -= addr_size;
hannes1716c032007-03-28 07:34:55 +0000529 }
530 break;
Florian Forster289e0ae2009-05-17 00:05:30 +0200531 }
hannes1716c032007-03-28 07:34:55 +0000532
533 case OLSR_HNA_MSG:
Florian Forster289e0ae2009-05-17 00:05:30 +0200534 if (is_ipv6)
535 {
536 int i = 0;
Ferry Hubertsd8b204e2015-04-24 14:43:14 +0200537
538 ND_PRINT((ndo, "\n\t Advertised networks (total %u)",
539 (unsigned int) (msg_tlen / sizeof(struct olsr_hna6))));
540
Florian Forster289e0ae2009-05-17 00:05:30 +0200541 while (msg_tlen >= sizeof(struct olsr_hna6)) {
Guy Harris69cb46a2015-04-26 17:24:42 -0700542 const struct olsr_hna6 *hna6;
hannes1716c032007-03-28 07:34:55 +0000543
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400544 ND_TCHECK2(*msg_data, sizeof(struct olsr_hna6));
hannes1716c032007-03-28 07:34:55 +0000545
Guy Harris69cb46a2015-04-26 17:24:42 -0700546 hna6 = (const struct olsr_hna6 *)msg_data;
hannes1716c032007-03-28 07:34:55 +0000547
Denis Ovsienko62a51842014-03-27 14:58:23 +0400548 ND_PRINT((ndo, "\n\t #%i: %s/%u",
Guy Harris1cde6432014-04-04 00:43:46 -0700549 i, ip6addr_string(ndo, hna6->network),
Denis Ovsienko62a51842014-03-27 14:58:23 +0400550 mask62plen (hna6->mask)));
hannes1716c032007-03-28 07:34:55 +0000551
Florian Forster289e0ae2009-05-17 00:05:30 +0200552 msg_data += sizeof(struct olsr_hna6);
553 msg_tlen -= sizeof(struct olsr_hna6);
554 }
555 }
556 else
Florian Forster289e0ae2009-05-17 00:05:30 +0200557 {
558 int col = 0;
Ferry Hubertsd8b204e2015-04-24 14:43:14 +0200559
560 ND_PRINT((ndo, "\n\t Advertised networks (total %u)",
561 (unsigned int) (msg_tlen / sizeof(struct olsr_hna4))));
562
Florian Forster289e0ae2009-05-17 00:05:30 +0200563 while (msg_tlen >= sizeof(struct olsr_hna4)) {
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400564 ND_TCHECK2(*msg_data, sizeof(struct olsr_hna4));
Florian Forster289e0ae2009-05-17 00:05:30 +0200565
Guy Harris69cb46a2015-04-26 17:24:42 -0700566 ptr.hna = (const struct olsr_hna4 *)msg_data;
Florian Forster289e0ae2009-05-17 00:05:30 +0200567
568 /* print 4 prefixes per line */
Francois-Xavier Le Bail9ba41b42015-05-04 15:25:54 +0200569 if (!ptr.hna->network[0] && !ptr.hna->network[1] &&
570 !ptr.hna->network[2] && !ptr.hna->network[3] &&
571 !ptr.hna->mask[GW_HNA_PAD] &&
572 ptr.hna->mask[GW_HNA_FLAGS]) {
Ferry Huberts4b4106c2015-04-24 14:43:58 +0200573 /* smart gateway */
574 ND_PRINT((ndo, "%sSmart-Gateway:%s%s%s%s%s %u/%u",
Francois-Xavier Le Bail9ba41b42015-05-04 15:25:54 +0200575 col == 0 ? "\n\t " : ", ", /* indent */
576 /* sgw */
577 /* LINKSPEED */
578 (ptr.hna->mask[GW_HNA_FLAGS] &
579 GW_HNA_FLAG_LINKSPEED) ? " LINKSPEED" : "",
580 /* IPV4 */
581 (ptr.hna->mask[GW_HNA_FLAGS] &
582 GW_HNA_FLAG_IPV4) ? " IPV4" : "",
583 /* IPV4-NAT */
584 (ptr.hna->mask[GW_HNA_FLAGS] &
585 GW_HNA_FLAG_IPV4_NAT) ? " IPV4-NAT" : "",
586 /* IPV6 */
587 (ptr.hna->mask[GW_HNA_FLAGS] &
588 GW_HNA_FLAG_IPV6) ? " IPV6" : "",
589 /* IPv6PREFIX */
590 (ptr.hna->mask[GW_HNA_FLAGS] &
591 GW_HNA_FLAG_IPV6PREFIX) ? " IPv6-PREFIX" : "",
592 /* uplink */
593 (ptr.hna->mask[GW_HNA_FLAGS] &
594 GW_HNA_FLAG_LINKSPEED) ?
595 deserialize_gw_speed(ptr.hna->mask[GW_HNA_UPLINK]) : 0,
596 /* downlink */
597 (ptr.hna->mask[GW_HNA_FLAGS] &
598 GW_HNA_FLAG_LINKSPEED) ?
599 deserialize_gw_speed(ptr.hna->mask[GW_HNA_DOWNLINK]) : 0
Ferry Huberts4b4106c2015-04-24 14:43:58 +0200600 ));
601 } else {
602 /* normal route */
603 ND_PRINT((ndo, "%s%s/%u",
604 col == 0 ? "\n\t " : ", ",
605 ipaddr_string(ndo, ptr.hna->network),
606 mask2plen(EXTRACT_32BITS(ptr.hna->mask))));
607 }
Florian Forster289e0ae2009-05-17 00:05:30 +0200608
609 msg_data += sizeof(struct olsr_hna4);
610 msg_tlen -= sizeof(struct olsr_hna4);
611
612 col = (col + 1) % 4;
613 }
hannescb08bf92007-03-27 10:44:09 +0000614 }
615 break;
616
hannesc695e7f2008-08-16 11:23:09 +0000617 case OLSR_NAMESERVICE_MSG:
Florian Forster289e0ae2009-05-17 00:05:30 +0200618 {
Guy Harris57bc7f62009-05-21 10:48:19 -0700619 u_int name_entries = EXTRACT_16BITS(msg_data+2);
620 u_int addr_size = 4;
Florian Forster289e0ae2009-05-17 00:05:30 +0200621 int name_entries_valid = 0;
Guy Harris57bc7f62009-05-21 10:48:19 -0700622 u_int i;
Florian Forster289e0ae2009-05-17 00:05:30 +0200623
624 if (is_ipv6)
625 addr_size = 16;
626
627 if ((name_entries > 0)
628 && ((name_entries * (4 + addr_size)) <= msg_tlen))
629 name_entries_valid = 1;
630
Guy Harris429c8b42009-06-28 12:16:43 -0700631 if (msg_tlen < 4)
632 goto trunc;
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400633 ND_TCHECK2(*msg_data, 4);
Guy Harris429c8b42009-06-28 12:16:43 -0700634
Denis Ovsienko62a51842014-03-27 14:58:23 +0400635 ND_PRINT((ndo, "\n\t Version %u, Entries %u%s",
Florian Forster289e0ae2009-05-17 00:05:30 +0200636 EXTRACT_16BITS(msg_data),
Denis Ovsienko62a51842014-03-27 14:58:23 +0400637 name_entries, (name_entries_valid == 0) ? " (invalid)" : ""));
Florian Forster289e0ae2009-05-17 00:05:30 +0200638
639 if (name_entries_valid == 0)
640 break;
641
hannesc695e7f2008-08-16 11:23:09 +0000642 msg_data += 4;
643 msg_tlen -= 4;
644
Florian Forster289e0ae2009-05-17 00:05:30 +0200645 for (i = 0; i < name_entries; i++) {
646 int name_entry_len_valid = 0;
647
648 if (msg_tlen < 4)
649 break;
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400650 ND_TCHECK2(*msg_data, 4);
hannesc695e7f2008-08-16 11:23:09 +0000651
652 name_entry_type = EXTRACT_16BITS(msg_data);
653 name_entry_len = EXTRACT_16BITS(msg_data+2);
Florian Forster289e0ae2009-05-17 00:05:30 +0200654
hannesc695e7f2008-08-16 11:23:09 +0000655 msg_data += 4;
656 msg_tlen -= 4;
657
Florian Forster289e0ae2009-05-17 00:05:30 +0200658 if ((name_entry_len > 0) && ((addr_size + name_entry_len) <= msg_tlen))
659 name_entry_len_valid = 1;
hannesc695e7f2008-08-16 11:23:09 +0000660
Denis Ovsienko62a51842014-03-27 14:58:23 +0400661 ND_PRINT((ndo, "\n\t #%u: type %#06x, length %u%s",
Florian Forster289e0ae2009-05-17 00:05:30 +0200662 (unsigned int) i, name_entry_type,
Denis Ovsienko62a51842014-03-27 14:58:23 +0400663 name_entry_len, (name_entry_len_valid == 0) ? " (invalid)" : ""));
hannesc695e7f2008-08-16 11:23:09 +0000664
Florian Forster289e0ae2009-05-17 00:05:30 +0200665 if (name_entry_len_valid == 0)
666 break;
667
Florian Forster289e0ae2009-05-17 00:05:30 +0200668 /* 32-bit alignment */
Guy Harris429c8b42009-06-28 12:16:43 -0700669 name_entry_padding = 0;
hannesc695e7f2008-08-16 11:23:09 +0000670 if (name_entry_len%4 != 0)
Guy Harris429c8b42009-06-28 12:16:43 -0700671 name_entry_padding = 4-(name_entry_len%4);
hannesc695e7f2008-08-16 11:23:09 +0000672
Guy Harris429c8b42009-06-28 12:16:43 -0700673 if (msg_tlen < addr_size + name_entry_len + name_entry_padding)
674 goto trunc;
675
Denis Ovsienkof0f2e6d2014-06-12 14:59:03 +0400676 ND_TCHECK2(*msg_data, addr_size + name_entry_len + name_entry_padding);
Guy Harris429c8b42009-06-28 12:16:43 -0700677
Guy Harris429c8b42009-06-28 12:16:43 -0700678 if (is_ipv6)
Denis Ovsienko62a51842014-03-27 14:58:23 +0400679 ND_PRINT((ndo, ", address %s, name \"",
Guy Harris1cde6432014-04-04 00:43:46 -0700680 ip6addr_string(ndo, msg_data)));
Guy Harris429c8b42009-06-28 12:16:43 -0700681 else
Denis Ovsienko62a51842014-03-27 14:58:23 +0400682 ND_PRINT((ndo, ", address %s, name \"",
Guy Harris1cde6432014-04-04 00:43:46 -0700683 ipaddr_string(ndo, msg_data)));
Francois-Xavier Le Bail07dfc4e2014-12-08 22:18:16 +0100684 (void)fn_printn(ndo, msg_data + addr_size, name_entry_len, NULL);
Denis Ovsienko62a51842014-03-27 14:58:23 +0400685 ND_PRINT((ndo, "\""));
Guy Harris429c8b42009-06-28 12:16:43 -0700686
687 msg_data += addr_size + name_entry_len + name_entry_padding;
688 msg_tlen -= addr_size + name_entry_len + name_entry_padding;
Florian Forster289e0ae2009-05-17 00:05:30 +0200689 } /* for (i = 0; i < name_entries; i++) */
hannesc695e7f2008-08-16 11:23:09 +0000690 break;
Florian Forster289e0ae2009-05-17 00:05:30 +0200691 } /* case OLSR_NAMESERVICE_MSG */
hannesc695e7f2008-08-16 11:23:09 +0000692
hannescb08bf92007-03-27 10:44:09 +0000693 /*
694 * FIXME those are the defined messages that lack a decoder
695 * you are welcome to contribute code ;-)
696 */
hannescb08bf92007-03-27 10:44:09 +0000697 case OLSR_POWERINFO_MSG:
hannescb08bf92007-03-27 10:44:09 +0000698 default:
Denis Ovsienko62a51842014-03-27 14:58:23 +0400699 print_unknown_data(ndo, msg_data, "\n\t ", msg_tlen);
hannescb08bf92007-03-27 10:44:09 +0000700 break;
Florian Forster289e0ae2009-05-17 00:05:30 +0200701 } /* switch (msg_type) */
hannescb08bf92007-03-27 10:44:09 +0000702 tptr += msg_len;
Florian Forster289e0ae2009-05-17 00:05:30 +0200703 } /* while (tptr < (pptr+length)) */
hannescb08bf92007-03-27 10:44:09 +0000704
705 return;
706
707 trunc:
Denis Ovsienko62a51842014-03-27 14:58:23 +0400708 ND_PRINT((ndo, "[|olsr]"));
hannescb08bf92007-03-27 10:44:09 +0000709}
710
711/*
712 * Local Variables:
713 * c-style: whitesmith
714 * c-basic-offset: 4
715 * End:
716 */