2 * Routines for Geneve - Generic Network Virtualization Encapsulation
3 * http://tools.ietf.org/html/draft-gross-geneve-00
5 * Copyright (c) 2014 VMware, Inc. All Rights Reserved.
6 * Author: Jesse Gross <jesse@nicira.com>
8 * Wireshark - Network traffic analyzer
9 * By Gerald Combs <gerald@wireshark.org>
10 * Copyright 1998 Gerald Combs
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation; either version 2
15 * of the License, or (at your option) any later version.
17 * This program is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, write to the Free Software
24 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
30 #include <epan/etypes.h>
31 #include <epan/expert.h>
32 #include <epan/packet.h>
34 #define UDP_PORT_GENEVE 6081
37 #define HDR_OPTS_LEN_MASK 0x3F
39 #define FLAG_OAM (1 << 7)
41 #define OPT_TYPE_CRITICAL (1 << 7)
42 #define OPT_FLAGS_SHIFT 5
43 #define OPT_LEN_MASK 0x1F
45 static const range_string class_id_names[] = {
46 { 0, 0xFF, "Standard" },
47 { 0xFFFF, 0xFFFF, "Experimental" },
51 void proto_register_geneve(void);
52 void proto_reg_handoff_geneve(void);
54 static int proto_geneve = -1;
56 static int hf_geneve_version = -1;
57 static int hf_geneve_opt_len = -1;
58 static int hf_geneve_flags = -1;
59 static int hf_geneve_flag_oam = -1;
60 static int hf_geneve_flag_critical = -1;
61 static int hf_geneve_flag_reserved = -1;
62 static int hf_geneve_proto_type = -1;
63 static int hf_geneve_vni = -1;
64 static int hf_geneve_reserved = -1;
65 static int hf_geneve_options = -1;
66 static int hf_geneve_option_class = -1;
67 static int hf_geneve_option_type = -1;
68 static int hf_geneve_option_type_critical = -1;
69 static int hf_geneve_option_flags = -1;
70 static int hf_geneve_option_flags_reserved = -1;
71 static int hf_geneve_option_length = -1;
72 static int hf_geneve_opt_unknown = -1;
73 static int hf_geneve_opt_unknown_data = -1;
75 static int ett_geneve = -1;
76 static int ett_geneve_flags = -1;
77 static int ett_geneve_opt_flags = -1;
78 static int ett_geneve_options = -1;
79 static int ett_geneve_unknown_opt = -1;
81 static expert_field ei_geneve_opt_len_invalid = EI_INIT;
83 static dissector_table_t ethertype_dissector_table;
84 static dissector_handle_t data_handle;
87 print_flags(guint8 flags, proto_item *flag_item)
89 static const char flag_names[][5] = {"OAM", "CRIT"};
96 proto_item_append_text(flag_item, " (");
98 for (i = 0; i < array_length(flag_names); i++) {
99 guint8 bit = 1 << (7 - i);
102 proto_item_append_text(flag_item, "%s", flag_names[i]);
106 proto_item_append_text(flag_item, ", ");
112 proto_item_append_text(flag_item, "RSVD");
115 proto_item_append_text(flag_item, ")");
119 format_unknown_option_name(guint16 opt_class, guint8 opt_type)
123 name = wmem_strdup_printf(wmem_packet_scope(),
124 "Unknown, Class: %s (0x%04x) Type: 0x%02x",
125 rval_to_str_const(opt_class, class_id_names, "Unknown"),
126 opt_class, opt_type);
132 dissect_unknown_option(tvbuff_t *tvb, proto_tree *opts_tree, int offset,
133 guint16 opt_class, guint8 opt_type, int len)
135 proto_item *opt_item, *type_item, *hidden_item, *flag_item;
136 proto_tree *opt_tree, *flag_tree;
137 const char *critical;
140 critical = opt_type & OPT_TYPE_CRITICAL ? "Critical" : "Non-critical";
142 opt_item = proto_tree_add_item(opts_tree, hf_geneve_opt_unknown,
143 tvb, offset, len, ENC_NA);
144 proto_item_set_text(opt_item, "%s (%s)",
145 format_unknown_option_name(opt_class, opt_type),
148 opt_tree = proto_item_add_subtree(opt_item, ett_geneve_unknown_opt);
150 proto_tree_add_item(opt_tree, hf_geneve_option_class, tvb,
151 offset, 2, ENC_BIG_ENDIAN);
154 type_item = proto_tree_add_item(opt_tree, hf_geneve_option_type, tvb,
155 offset, 1, ENC_BIG_ENDIAN);
156 proto_item_append_text(type_item, " (%s)", critical);
157 hidden_item = proto_tree_add_item(opt_tree, hf_geneve_option_type_critical,
158 tvb, offset, 1, ENC_BIG_ENDIAN);
159 PROTO_ITEM_SET_HIDDEN(hidden_item);
162 flags = tvb_get_guint8(tvb, offset) >> OPT_FLAGS_SHIFT;
163 flag_item = proto_tree_add_uint(opt_tree, hf_geneve_option_flags, tvb,
165 flag_tree = proto_item_add_subtree(flag_item, ett_geneve_opt_flags);
166 proto_tree_add_item(flag_tree, hf_geneve_option_flags_reserved, tvb,
167 offset, 1, ENC_BIG_ENDIAN);
169 proto_item_append_text(flag_item, " (RSVD)");
171 PROTO_ITEM_SET_HIDDEN(flag_item);
174 proto_tree_add_uint_format_value(opt_tree, hf_geneve_option_length, tvb,
175 offset, 1, len, "%u bytes", len);
178 proto_tree_add_item(opt_tree, hf_geneve_opt_unknown_data, tvb, offset,
183 dissect_geneve_options(tvbuff_t *tvb, packet_info *pinfo,
184 proto_tree *geneve_tree, int offset, int len)
186 proto_item *opts_item;
187 proto_tree *opts_tree;
192 opts_item = proto_tree_add_item(geneve_tree, hf_geneve_options, tvb,
193 offset, len, ENC_NA);
194 proto_item_set_text(opts_item, "Options: (%u bytes)", len);
195 opts_tree = proto_item_add_subtree(opts_item, ett_geneve_options);
198 opt_class = tvb_get_ntohs(tvb, offset);
199 opt_type = tvb_get_guint8(tvb, offset + 2);
200 opt_len = 4 + ((tvb_get_guint8(tvb, offset + 3) & OPT_LEN_MASK) * 4);
203 proto_tree_add_expert_format(opts_tree, pinfo,
204 &ei_geneve_opt_len_invalid, tvb,
206 "%s (length of %u is past end of options)",
207 format_unknown_option_name(opt_class,
213 dissect_unknown_option(tvb, opts_tree, offset,
214 opt_class, opt_type, opt_len);
222 dissect_geneve(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
224 proto_item *ti, *flag_item, *rsvd_item;
225 proto_tree *geneve_tree, *flag_tree;
233 col_set_str(pinfo->cinfo, COL_PROTOCOL, "Geneve");
234 col_clear(pinfo->cinfo, COL_INFO);
236 proto_type = tvb_get_ntohs(tvb, 2);
237 col_add_fstr(pinfo->cinfo, COL_INFO, "Encapsulated %s",
238 val_to_str(proto_type, etype_vals, "0x%04x (unknown)"));
240 flags = tvb_get_guint8(tvb, 1);
241 ti = proto_tree_add_protocol_format(tree, proto_geneve, tvb, 0, -1,
242 "Generic Network Virtualization Encapsuation, VNI: 0x%06x"
244 tvb_get_ntoh24(tvb, 4),
245 flags & FLAG_OAM ? ", OAM" : "");
247 geneve_tree = proto_item_add_subtree(ti, ett_geneve);
249 /* Version and option length. */
250 ver_opt = tvb_get_guint8(tvb, offset);
251 proto_tree_add_uint(geneve_tree, hf_geneve_version, tvb,
252 offset, 1, ver_opt >> VER_SHIFT);
253 opts_len = (ver_opt & HDR_OPTS_LEN_MASK) * 4;
254 proto_tree_add_uint_format_value(geneve_tree, hf_geneve_opt_len, tvb,
255 offset, 1, opts_len, "%u bytes", opts_len);
260 flag_item = proto_tree_add_item(geneve_tree, hf_geneve_flags, tvb,
261 offset, 1, ENC_BIG_ENDIAN);
262 print_flags(flags, flag_item);
264 flag_tree = proto_item_add_subtree(flag_item, ett_geneve_flags);
266 proto_tree_add_item(flag_tree, hf_geneve_flag_oam, tvb, offset,
268 proto_tree_add_item(flag_tree, hf_geneve_flag_critical, tvb, offset,
270 proto_tree_add_item(flag_tree, hf_geneve_flag_reserved, tvb, offset,
276 proto_tree_add_item(geneve_tree, hf_geneve_proto_type, tvb,
277 offset, 2, ENC_BIG_ENDIAN);
281 proto_tree_add_item(geneve_tree, hf_geneve_vni, tvb, offset, 3, ENC_BIG_ENDIAN);
285 rsvd_item = proto_tree_add_item(geneve_tree, hf_geneve_reserved, tvb, offset,
287 if (!tvb_get_guint8(tvb, offset)) {
288 PROTO_ITEM_SET_HIDDEN(rsvd_item);
293 if (tree && opts_len) {
294 dissect_geneve_options(tvb, pinfo, geneve_tree, offset, opts_len);
298 proto_item_set_len(ti, offset);
300 next_tvb = tvb_new_subset_remaining(tvb, offset);
301 if (!dissector_try_uint(ethertype_dissector_table, proto_type, next_tvb, pinfo, tree))
302 call_dissector(data_handle, next_tvb, pinfo, tree);
305 /* Register Geneve with Wireshark */
307 proto_register_geneve(void)
309 static hf_register_info hf[] = {
310 { &hf_geneve_version,
311 { "Version", "geneve.version",
312 FT_UINT8, BASE_DEC, NULL, 0x00,
315 { &hf_geneve_opt_len,
316 { "Options Length", "geneve.options_length",
317 FT_UINT8, BASE_DEC, NULL, 0x0,
321 { "Flags", "geneve.flags",
322 FT_UINT8, BASE_HEX, NULL, 0x00,
325 { &hf_geneve_flag_oam,
326 { "Operations, Administration and Management Frame", "geneve.flags.oam",
327 FT_BOOLEAN, 8, NULL, 0x80,
330 { &hf_geneve_flag_critical,
331 { "Critical Options Present", "geneve.flags.critical",
332 FT_BOOLEAN, 8, NULL, 0x40,
335 { &hf_geneve_flag_reserved,
336 { "Reserved", "geneve.flags.reserved",
337 FT_BOOLEAN, 8, NULL, 0x3F,
340 { &hf_geneve_proto_type,
341 { "Protocol Type", "geneve.proto_type",
342 FT_UINT16, BASE_HEX, VALS(etype_vals), 0x0,
346 { "Virtual Network Identifier (VNI)", "geneve.vni",
347 FT_UINT24, BASE_HEX, NULL, 0x0,
350 { &hf_geneve_reserved,
351 { "Reserved", "geneve.reserved",
352 FT_UINT8, BASE_HEX, NULL, 0x00,
355 { &hf_geneve_options,
356 { "Geneve Options", "geneve.options",
357 FT_BYTES, BASE_NONE, NULL, 0x00,
360 { &hf_geneve_option_class,
361 { "Class", "geneve.option.class",
362 FT_UINT16, BASE_HEX | BASE_RANGE_STRING, RVALS(class_id_names), 0x00,
365 { &hf_geneve_option_type,
366 { "Type", "geneve.option.type",
367 FT_UINT8, BASE_HEX, NULL, 0x00,
370 { &hf_geneve_option_type_critical,
371 { "Critical Option", "geneve.option.type.critical",
372 FT_BOOLEAN, 8, NULL, 0x80,
375 { &hf_geneve_option_flags,
376 { "Flags", "geneve.option.flags",
377 FT_UINT8, BASE_HEX, NULL, 0x00,
380 { &hf_geneve_option_flags_reserved,
381 { "Reserved", "geneve.option.flags.reserved",
382 FT_BOOLEAN, 8, NULL, 0xE0,
385 { &hf_geneve_option_length,
386 { "Length", "geneve.option.length",
387 FT_UINT8, BASE_DEC, NULL, 0x00,
390 { &hf_geneve_opt_unknown,
391 { "Unknown Option", "geneve.option.unknown",
392 FT_BYTES, BASE_NONE, NULL, 0x00,
395 { &hf_geneve_opt_unknown_data,
396 { "Option Data", "geneve.option.unknown.data",
397 FT_BYTES, BASE_NONE, NULL, 0x00,
402 static gint *ett[] = {
406 &ett_geneve_opt_flags,
407 &ett_geneve_unknown_opt,
410 static ei_register_info ei[] = {
411 { &ei_geneve_opt_len_invalid, { "geneve.option.length.invalid",
412 PI_SEQUENCE, PI_NOTE, "Invalid length for option", EXPFILL }},
415 expert_module_t *expert_geneve;
417 /* Register the protocol name and description */
418 proto_geneve = proto_register_protocol("Generic Network Virtualization Encapsulation",
421 proto_register_field_array(proto_geneve, hf, array_length(hf));
422 proto_register_subtree_array(ett, array_length(ett));
424 expert_geneve = expert_register_protocol(proto_geneve);
425 expert_register_field_array(expert_geneve, ei, array_length(ei));
429 proto_reg_handoff_geneve(void)
431 dissector_handle_t geneve_handle;
433 geneve_handle = create_dissector_handle(dissect_geneve, proto_geneve);
434 dissector_add_uint("udp.port", UDP_PORT_GENEVE, geneve_handle);
435 dissector_add_handle("udp.port", geneve_handle); /* For 'Decode As' */
437 ethertype_dissector_table = find_dissector_table("ethertype");
438 data_handle = find_dissector("data");
442 * Editor modelines - http://www.wireshark.org/tools/modelines.html
447 * indent-tabs-mode: nil
450 * vi: set shiftwidth=4 tabstop=8 expandtab:
451 * :indentSize=4:tabSize=8:noTabs=true: