2 * Routines for GVRP (GARP VLAN Registration Protocol) dissection
3 * Copyright 2000, Kevin Shi <techishi@ms22.hinet.net>
7 * Wireshark - Network traffic analyzer
8 * By Gerald Combs <gerald@wireshark.org>
9 * Copyright 1998 Gerald Combs
11 * This program is free software; you can redistribute it and/or
12 * modify it under the terms of the GNU General Public License
13 * as published by the Free Software Foundation; either version 2
14 * of the License, or (at your option) any later version.
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
21 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
34 #include <epan/packet.h>
35 #include <epan/llcsaps.h>
37 /* Initialize the protocol and registered fields */
38 static int proto_gvrp = -1;
39 static int hf_gvrp_proto_id = -1;
40 static int hf_gvrp_attribute_type = -1;
41 static int hf_gvrp_attribute_length = -1;
42 static int hf_gvrp_attribute_event = -1;
43 static int hf_gvrp_attribute_value = -1;
44 /*static int hf_gvrp_end_of_mark = -1;*/
46 /* Initialize the subtree pointers */
47 static gint ett_gvrp = -1;
48 /*static gint ett_gvrp_message = -1;
49 static gint ett_gvrp_attribute_list = -1;
50 static gint ett_gvrp_attribute = -1;*/
52 static dissector_handle_t data_handle;
54 /* Constant definitions */
55 #define GARP_DEFAULT_PROTOCOL_ID 0x0001
56 #define GARP_END_OF_MARK 0x00
58 #define GVRP_ATTRIBUTE_TYPE 0x01
60 static const value_string attribute_type_vals[] = {
61 { GVRP_ATTRIBUTE_TYPE, "VID" },
65 /* The length of GVRP LeaveAll attribute should be 2 octets (one for length
66 * and the other for event) */
67 #define GVRP_LENGTH_LEAVEALL (sizeof(guint8)+sizeof(guint8))
69 /* The length of GVRP attribute other than LeaveAll should be 4 octets (one
70 * for length, one for event, and the last two for VID value).
72 #define GVRP_LENGTH_NON_LEAVEALL (sizeof(guint8)+sizeof(guint8)+sizeof(guint16))
74 /* Packet offset definitions */
75 #define GARP_PROTOCOL_ID 0
77 /* Event definitions */
78 #define GVRP_EVENT_LEAVEALL 0
79 #define GVRP_EVENT_JOINEMPTY 1
80 #define GVRP_EVENT_JOININ 2
81 #define GVRP_EVENT_LEAVEEMPTY 3
82 #define GVRP_EVENT_LEAVEIN 4
83 #define GVRP_EVENT_EMPTY 5
85 static const value_string event_vals[] = {
86 { GVRP_EVENT_LEAVEALL, "Leave All" },
87 { GVRP_EVENT_JOINEMPTY, "Join Empty" },
88 { GVRP_EVENT_JOININ, "Join In" },
89 { GVRP_EVENT_LEAVEEMPTY, "Leave Empty" },
90 { GVRP_EVENT_LEAVEIN, "Leave In" },
91 { GVRP_EVENT_EMPTY, "Empty" },
95 /* Code to actually dissect the packets */
97 dissect_gvrp(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
100 proto_tree *gvrp_tree;
103 int msg_index, attr_index, offset = 0, length = tvb_reported_length(tvb);
105 col_set_str(pinfo->cinfo, COL_PROTOCOL, "GVRP");
107 col_set_str(pinfo->cinfo, COL_INFO, "GVRP");
111 ti = proto_tree_add_item(tree, proto_gvrp, tvb, 0, length, FALSE);
113 gvrp_tree = proto_item_add_subtree(ti, ett_gvrp);
115 /* Read in GARP protocol ID */
116 protocol_id = tvb_get_ntohs(tvb, GARP_PROTOCOL_ID);
118 proto_tree_add_uint_format(gvrp_tree, hf_gvrp_proto_id, tvb,
119 GARP_PROTOCOL_ID, sizeof(guint16),
121 "Protocol Identifier: 0x%04x (%s)",
123 protocol_id == GARP_DEFAULT_PROTOCOL_ID ?
124 "GARP VLAN Registration Protocol" :
127 /* Currently only one protocol ID is supported */
128 if (protocol_id != GARP_DEFAULT_PROTOCOL_ID)
130 proto_tree_add_text(gvrp_tree, tvb, GARP_PROTOCOL_ID, sizeof(guint16),
131 " (Warning: this version of Wireshark only knows about protocol id = 1)");
132 call_dissector(data_handle,
133 tvb_new_subset(tvb, GARP_PROTOCOL_ID + sizeof(guint16), -1, -1),
138 offset += sizeof(guint16);
139 length -= sizeof(guint16);
143 /* Begin to parse GARP messages */
146 proto_item *msg_item;
147 int msg_start = offset;
149 /* Read in attribute type. */
150 octet = tvb_get_guint8(tvb, offset);
152 /* Check for end of mark */
153 if (octet == GARP_END_OF_MARK)
155 /* End of GARP PDU */
158 proto_tree_add_text(gvrp_tree, tvb, offset, sizeof(guint8),
164 call_dissector(data_handle,
165 tvb_new_subset_remaining(tvb, offset), pinfo, tree);
170 offset += sizeof(guint8);
171 length -= sizeof(guint8);
173 msg_item = proto_tree_add_text(gvrp_tree, tvb, msg_start, -1,
174 "Message %d", msg_index + 1);
176 proto_tree_add_uint(gvrp_tree, hf_gvrp_attribute_type, tvb,
177 msg_start, sizeof(guint8), octet);
179 /* GVRP only supports one attribute type. */
180 if (octet != GVRP_ATTRIBUTE_TYPE)
182 call_dissector(data_handle, tvb_new_subset_remaining(tvb, offset),
191 int attr_start = offset;
192 proto_item *attr_item;
194 /* Read in attribute length. */
195 octet = tvb_get_guint8(tvb, offset);
197 /* Check for end of mark */
198 if (octet == GARP_END_OF_MARK)
200 /* If at least one message has been already read,
201 * check for another end of mark.
205 proto_tree_add_text(gvrp_tree, tvb, offset,
206 sizeof(guint8), " End of mark");
208 offset += sizeof(guint8);
209 length -= sizeof(guint8);
211 proto_item_set_len(msg_item, offset - msg_start);
216 call_dissector(data_handle,
217 tvb_new_subset_remaining(tvb, offset), pinfo, tree);
225 offset += sizeof(guint8);
226 length -= sizeof(guint8);
228 attr_item = proto_tree_add_text(gvrp_tree, tvb,
229 attr_start, -1, " Attribute %d", attr_index + 1);
231 proto_tree_add_uint(gvrp_tree, hf_gvrp_attribute_length,
232 tvb, attr_start, sizeof(guint8), octet);
234 /* Read in attribute event */
235 event = tvb_get_guint8(tvb, offset);
237 proto_tree_add_uint(gvrp_tree, hf_gvrp_attribute_event,
238 tvb, offset, sizeof(guint8), event);
240 offset += sizeof(guint8);
241 length -= sizeof(guint8);
245 case GVRP_EVENT_LEAVEALL:
246 if (octet != GVRP_LENGTH_LEAVEALL)
248 call_dissector(data_handle,
249 tvb_new_subset_remaining(tvb, offset), pinfo,
255 case GVRP_EVENT_JOINEMPTY:
256 case GVRP_EVENT_JOININ:
257 case GVRP_EVENT_LEAVEEMPTY:
258 case GVRP_EVENT_LEAVEIN:
259 case GVRP_EVENT_EMPTY:
260 if (octet != GVRP_LENGTH_NON_LEAVEALL)
262 call_dissector(data_handle,
263 tvb_new_subset_remaining(tvb, offset),pinfo,
268 /* Show attribute value */
269 proto_tree_add_item(gvrp_tree, hf_gvrp_attribute_value,
270 tvb, offset, sizeof(guint16), FALSE);
272 offset += sizeof(guint16);
273 length -= sizeof(guint16);
277 call_dissector(data_handle,
278 tvb_new_subset_remaining(tvb, offset), pinfo, tree);
283 proto_item_set_len(attr_item, offset - attr_start);
294 /* Register the protocol with Wireshark */
296 proto_register_gvrp(void)
298 static hf_register_info hf[] = {
300 { "Protocol ID", "gvrp.protocol_id",
301 FT_UINT16, BASE_HEX, NULL, 0x0,
304 { &hf_gvrp_attribute_type,
305 { "Type", "gvrp.attribute_type",
306 FT_UINT8, BASE_HEX, VALS(attribute_type_vals), 0x0,
309 { &hf_gvrp_attribute_length,
310 { "Length", "gvrp.attribute_length",
311 FT_UINT8, BASE_DEC, NULL, 0x0,
314 { &hf_gvrp_attribute_event,
315 { "Event", "gvrp.attribute_event",
316 FT_UINT8, BASE_DEC, VALS(event_vals), 0x0,
319 { &hf_gvrp_attribute_value,
320 { "Value", "gvrp.attribute_value",
321 FT_UINT16, BASE_DEC, NULL, 0x0,
326 static gint *ett[] = {
330 /* Register the protocol name and description for GVRP */
331 proto_gvrp = proto_register_protocol("GARP VLAN Registration Protocol",
334 /* Required function calls to register the header fields and subtrees
336 proto_register_field_array(proto_gvrp, hf, array_length(hf));
337 proto_register_subtree_array(ett, array_length(ett));
339 register_dissector("gvrp", dissect_gvrp, proto_gvrp);
343 proto_reg_handoff_gvrp(void){
344 data_handle = find_dissector("data");