blob: 0b7ff6e514728fd92eaf2d5332106f7bfef428d3 [file] [log] [blame]
Elliott Hughes892a68b2015-10-19 14:43:53 -07001/*
2 * Copyright (c) 2014 VMware, Inc. All Rights Reserved.
3 *
4 * Jesse Gross <jesse@nicira.com>
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that: (1) source code
8 * distributions retain the above copyright notice and this paragraph
9 * in its entirety, and (2) distributions including binary code include
10 * the above copyright notice and this paragraph in its entirety in
11 * the documentation or other materials provided with the distribution.
12 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND
13 * WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, WITHOUT
14 * LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
15 * FOR A PARTICULAR PURPOSE.
16 */
17
Elliott Hughese2e3bd12017-05-15 10:59:29 -070018/* \summary: Generic Network Virtualization Encapsulation (Geneve) printer */
19
Elliott Hughes892a68b2015-10-19 14:43:53 -070020#ifdef HAVE_CONFIG_H
Elliott Hughes820eced2021-08-20 18:00:50 -070021#include <config.h>
Elliott Hughes892a68b2015-10-19 14:43:53 -070022#endif
23
Elliott Hughes820eced2021-08-20 18:00:50 -070024#include "netdissect-stdinc.h"
Elliott Hughes892a68b2015-10-19 14:43:53 -070025
Elliott Hughese2e3bd12017-05-15 10:59:29 -070026#include "netdissect.h"
Elliott Hughes892a68b2015-10-19 14:43:53 -070027#include "extract.h"
28#include "ethertype.h"
29
30/*
Elliott Hughese2e3bd12017-05-15 10:59:29 -070031 * Geneve header, draft-ietf-nvo3-geneve
Elliott Hughes892a68b2015-10-19 14:43:53 -070032 *
33 * 0 1 2 3
34 * 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
35 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
36 * |Ver| Opt Len |O|C| Rsvd. | Protocol Type |
37 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
38 * | Virtual Network Identifier (VNI) | Reserved |
39 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
40 * | Variable Length Options |
41 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
42 *
43 * Options:
44 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
45 * | Option Class | Type |R|R|R| Length |
46 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
47 * | Variable Option Data |
48 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
49 */
50
51#define VER_SHIFT 6
52#define HDR_OPTS_LEN_MASK 0x3F
53
54#define FLAG_OAM (1 << 7)
55#define FLAG_CRITICAL (1 << 6)
56#define FLAG_R1 (1 << 5)
57#define FLAG_R2 (1 << 4)
58#define FLAG_R3 (1 << 3)
59#define FLAG_R4 (1 << 2)
60#define FLAG_R5 (1 << 1)
61#define FLAG_R6 (1 << 0)
62
63#define OPT_TYPE_CRITICAL (1 << 7)
64#define OPT_LEN_MASK 0x1F
65
66static const struct tok geneve_flag_values[] = {
67 { FLAG_OAM, "O" },
68 { FLAG_CRITICAL, "C" },
69 { FLAG_R1, "R1" },
70 { FLAG_R2, "R2" },
71 { FLAG_R3, "R3" },
72 { FLAG_R4, "R4" },
73 { FLAG_R5, "R5" },
74 { FLAG_R6, "R6" },
75 { 0, NULL }
76};
77
78static const char *
79format_opt_class(uint16_t opt_class)
80{
Elliott Hughese2e3bd12017-05-15 10:59:29 -070081 switch (opt_class) {
82 case 0x0100:
83 return "Linux";
84 case 0x0101:
85 return "Open vSwitch";
86 case 0x0102:
87 return "Open Virtual Networking (OVN)";
88 case 0x0103:
89 return "In-band Network Telemetry (INT)";
90 case 0x0104:
91 return "VMware";
92 default:
93 if (opt_class <= 0x00ff)
94 return "Standard";
95 else if (opt_class >= 0xfff0)
96 return "Experimental";
97 }
98
99 return "Unknown";
Elliott Hughes892a68b2015-10-19 14:43:53 -0700100}
101
102static void
103geneve_opts_print(netdissect_options *ndo, const u_char *bp, u_int len)
104{
105 const char *sep = "";
106
107 while (len > 0) {
108 uint16_t opt_class;
109 uint8_t opt_type;
110 uint8_t opt_len;
111
Elliott Hughes820eced2021-08-20 18:00:50 -0700112 ND_PRINT("%s", sep);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700113 sep = ", ";
114
Elliott Hughes820eced2021-08-20 18:00:50 -0700115 opt_class = GET_BE_U_2(bp);
116 opt_type = GET_U_1(bp + 2);
117 opt_len = 4 + ((GET_U_1(bp + 3) & OPT_LEN_MASK) * 4);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700118
Elliott Hughes820eced2021-08-20 18:00:50 -0700119 ND_PRINT("class %s (0x%x) type 0x%x%s len %u",
Elliott Hughes892a68b2015-10-19 14:43:53 -0700120 format_opt_class(opt_class), opt_class, opt_type,
Elliott Hughes820eced2021-08-20 18:00:50 -0700121 opt_type & OPT_TYPE_CRITICAL ? "(C)" : "", opt_len);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700122
123 if (opt_len > len) {
Elliott Hughes820eced2021-08-20 18:00:50 -0700124 ND_PRINT(" [bad length]");
Elliott Hughes892a68b2015-10-19 14:43:53 -0700125 return;
126 }
127
128 if (ndo->ndo_vflag > 1 && opt_len > 4) {
Elliott Hughese2e3bd12017-05-15 10:59:29 -0700129 const uint32_t *data = (const uint32_t *)(bp + 4);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700130 int i;
131
Elliott Hughes820eced2021-08-20 18:00:50 -0700132 ND_PRINT(" data");
Elliott Hughes892a68b2015-10-19 14:43:53 -0700133
134 for (i = 4; i < opt_len; i += 4) {
Elliott Hughes820eced2021-08-20 18:00:50 -0700135 ND_PRINT(" %08x", GET_BE_U_4(data));
Elliott Hughese2e3bd12017-05-15 10:59:29 -0700136 data++;
Elliott Hughes892a68b2015-10-19 14:43:53 -0700137 }
138 }
139
140 bp += opt_len;
141 len -= opt_len;
142 }
143}
144
145void
146geneve_print(netdissect_options *ndo, const u_char *bp, u_int len)
147{
148 uint8_t ver_opt;
Elliott Hughese2e3bd12017-05-15 10:59:29 -0700149 u_int version;
Elliott Hughes892a68b2015-10-19 14:43:53 -0700150 uint8_t flags;
151 uint16_t prot;
152 uint32_t vni;
153 uint8_t reserved;
154 u_int opts_len;
155
Elliott Hughes820eced2021-08-20 18:00:50 -0700156 ndo->ndo_protocol = "geneve";
157 ND_PRINT("Geneve");
Elliott Hughes892a68b2015-10-19 14:43:53 -0700158
Elliott Hughes820eced2021-08-20 18:00:50 -0700159 if (len < 8) {
160 ND_PRINT(" [length %u < 8]", len);
161 nd_print_invalid(ndo);
162 return;
163 }
Elliott Hughes892a68b2015-10-19 14:43:53 -0700164
Elliott Hughes820eced2021-08-20 18:00:50 -0700165 ND_TCHECK_8(bp);
166
167 ver_opt = GET_U_1(bp);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700168 bp += 1;
169 len -= 1;
170
171 version = ver_opt >> VER_SHIFT;
172 if (version != 0) {
Elliott Hughes820eced2021-08-20 18:00:50 -0700173 ND_PRINT(" ERROR: unknown-version %u", version);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700174 return;
175 }
176
Elliott Hughes820eced2021-08-20 18:00:50 -0700177 flags = GET_U_1(bp);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700178 bp += 1;
179 len -= 1;
180
Elliott Hughes820eced2021-08-20 18:00:50 -0700181 prot = GET_BE_U_2(bp);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700182 bp += 2;
183 len -= 2;
184
Elliott Hughes820eced2021-08-20 18:00:50 -0700185 vni = GET_BE_U_3(bp);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700186 bp += 3;
187 len -= 3;
188
Elliott Hughes820eced2021-08-20 18:00:50 -0700189 reserved = GET_U_1(bp);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700190 bp += 1;
191 len -= 1;
192
Elliott Hughes820eced2021-08-20 18:00:50 -0700193 ND_PRINT(", Flags [%s]",
194 bittok2str_nosep(geneve_flag_values, "none", flags));
195 ND_PRINT(", vni 0x%x", vni);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700196
197 if (reserved)
Elliott Hughes820eced2021-08-20 18:00:50 -0700198 ND_PRINT(", rsvd 0x%x", reserved);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700199
200 if (ndo->ndo_eflag)
Elliott Hughes820eced2021-08-20 18:00:50 -0700201 ND_PRINT(", proto %s (0x%04x)",
202 tok2str(ethertype_values, "unknown", prot), prot);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700203
204 opts_len = (ver_opt & HDR_OPTS_LEN_MASK) * 4;
205
206 if (len < opts_len) {
Elliott Hughes820eced2021-08-20 18:00:50 -0700207 ND_PRINT(" truncated-geneve - %u bytes missing",
208 opts_len - len);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700209 return;
210 }
211
Elliott Hughes820eced2021-08-20 18:00:50 -0700212 ND_TCHECK_LEN(bp, opts_len);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700213
214 if (opts_len > 0) {
Elliott Hughes820eced2021-08-20 18:00:50 -0700215 ND_PRINT(", options [");
Elliott Hughes892a68b2015-10-19 14:43:53 -0700216
217 if (ndo->ndo_vflag)
218 geneve_opts_print(ndo, bp, opts_len);
219 else
Elliott Hughes820eced2021-08-20 18:00:50 -0700220 ND_PRINT("%u bytes", opts_len);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700221
Elliott Hughes820eced2021-08-20 18:00:50 -0700222 ND_PRINT("]");
Elliott Hughes892a68b2015-10-19 14:43:53 -0700223 }
224
225 bp += opts_len;
226 len -= opts_len;
227
228 if (ndo->ndo_vflag < 1)
Elliott Hughes820eced2021-08-20 18:00:50 -0700229 ND_PRINT(": ");
Elliott Hughes892a68b2015-10-19 14:43:53 -0700230 else
Elliott Hughes820eced2021-08-20 18:00:50 -0700231 ND_PRINT("\n\t");
Elliott Hughes892a68b2015-10-19 14:43:53 -0700232
Elliott Hughes820eced2021-08-20 18:00:50 -0700233 if (ethertype_print(ndo, prot, bp, len, ND_BYTES_AVAILABLE_AFTER(bp), NULL, NULL) == 0) {
Elliott Hughes892a68b2015-10-19 14:43:53 -0700234 if (prot == ETHERTYPE_TEB)
Elliott Hughes820eced2021-08-20 18:00:50 -0700235 ether_print(ndo, bp, len, ND_BYTES_AVAILABLE_AFTER(bp), NULL, NULL);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700236 else
Elliott Hughes820eced2021-08-20 18:00:50 -0700237 ND_PRINT("geneve-proto-0x%x", prot);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700238 }
239
240 return;
241
242trunc:
Elliott Hughes820eced2021-08-20 18:00:50 -0700243 nd_print_trunc(ndo);
Elliott Hughes892a68b2015-10-19 14:43:53 -0700244}