xref: /libbtbb/wireshark/plugins/btbb/packet-btbb.c (revision 4cc286f39977de0d2c7d8305e14882bbf313e363)
1*4cc286f3SDominic Spill /* packet-btbb.c
2*4cc286f3SDominic Spill  * Routines for Bluetooth baseband dissection
3*4cc286f3SDominic Spill  * Copyright 2009, Michael Ossmann <[email protected]>
4*4cc286f3SDominic Spill  *
5*4cc286f3SDominic Spill  * $Id$
6*4cc286f3SDominic Spill  *
7*4cc286f3SDominic Spill  * Wireshark - Network traffic analyzer
8*4cc286f3SDominic Spill  * By Gerald Combs <[email protected]>
9*4cc286f3SDominic Spill  * Copyright 1998 Gerald Combs
10*4cc286f3SDominic Spill  *
11*4cc286f3SDominic Spill  * This program is free software; you can redistribute it and/or
12*4cc286f3SDominic Spill  * modify it under the terms of the GNU General Public License
13*4cc286f3SDominic Spill  * as published by the Free Software Foundation; either version 2
14*4cc286f3SDominic Spill  * of the License, or (at your option) any later version.
15*4cc286f3SDominic Spill  *
16*4cc286f3SDominic Spill  * This program is distributed in the hope that it will be useful,
17*4cc286f3SDominic Spill  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18*4cc286f3SDominic Spill  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19*4cc286f3SDominic Spill  * GNU General Public License for more details.
20*4cc286f3SDominic Spill  *
21*4cc286f3SDominic Spill  * You should have received a copy of the GNU General Public License
22*4cc286f3SDominic Spill  * along with this program; if not, write to the Free Software
23*4cc286f3SDominic Spill  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
24*4cc286f3SDominic Spill  */
25*4cc286f3SDominic Spill 
26*4cc286f3SDominic Spill #ifdef HAVE_CONFIG_H
27*4cc286f3SDominic Spill #include "config.h"
28*4cc286f3SDominic Spill #else
29*4cc286f3SDominic Spill #include <wireshark/config.h>
30*4cc286f3SDominic Spill #endif
31*4cc286f3SDominic Spill 
32*4cc286f3SDominic Spill #include <epan/packet.h>
33*4cc286f3SDominic Spill #include <epan/prefs.h>
34*4cc286f3SDominic Spill 
35*4cc286f3SDominic Spill /* function prototypes */
36*4cc286f3SDominic Spill void proto_reg_handoff_btbb(void);
37*4cc286f3SDominic Spill 
38*4cc286f3SDominic Spill /* initialize the protocol and registered fields */
39*4cc286f3SDominic Spill static int proto_btbb = -1;
40*4cc286f3SDominic Spill static int hf_btbb_meta = -1;
41*4cc286f3SDominic Spill static int hf_btbb_dir = -1;
42*4cc286f3SDominic Spill static int hf_btbb_clk = -1;
43*4cc286f3SDominic Spill static int hf_btbb_channel = -1;
44*4cc286f3SDominic Spill static int hf_btbb_addrbits = -1;
45*4cc286f3SDominic Spill static int hf_btbb_clkbits = -1;
46*4cc286f3SDominic Spill static int hf_btbb_pkthdr = -1;
47*4cc286f3SDominic Spill static int hf_btbb_ltaddr = -1;
48*4cc286f3SDominic Spill static int hf_btbb_type = -1;
49*4cc286f3SDominic Spill static int hf_btbb_flags = -1;
50*4cc286f3SDominic Spill static int hf_btbb_flow = -1;
51*4cc286f3SDominic Spill static int hf_btbb_arqn = -1;
52*4cc286f3SDominic Spill static int hf_btbb_seqn = -1;
53*4cc286f3SDominic Spill static int hf_btbb_hec = -1;
54*4cc286f3SDominic Spill static int hf_btbb_payload = -1;
55*4cc286f3SDominic Spill static int hf_btbb_pldhdr = -1;
56*4cc286f3SDominic Spill static int hf_btbb_llid = -1;
57*4cc286f3SDominic Spill static int hf_btbb_pldflow = -1;
58*4cc286f3SDominic Spill static int hf_btbb_length = -1;
59*4cc286f3SDominic Spill static int hf_btbb_pldbody = -1;
60*4cc286f3SDominic Spill static int hf_btbb_crc = -1;
61*4cc286f3SDominic Spill static int hf_btbb_fhs_parity = -1;
62*4cc286f3SDominic Spill static int hf_btbb_fhs_lap = -1;
63*4cc286f3SDominic Spill static int hf_btbb_fhs_eir = -1;
64*4cc286f3SDominic Spill static int hf_btbb_fhs_sr = -1;
65*4cc286f3SDominic Spill static int hf_btbb_fhs_uap = -1;
66*4cc286f3SDominic Spill static int hf_btbb_fhs_nap = -1;
67*4cc286f3SDominic Spill static int hf_btbb_fhs_class = -1;
68*4cc286f3SDominic Spill static int hf_btbb_fhs_ltaddr = -1;
69*4cc286f3SDominic Spill static int hf_btbb_fhs_clk = -1;
70*4cc286f3SDominic Spill static int hf_btbb_fhs_psmode = -1;
71*4cc286f3SDominic Spill 
72*4cc286f3SDominic Spill /* field values */
73*4cc286f3SDominic Spill static const true_false_string direction = {
74*4cc286f3SDominic Spill 	"Slave to Master",
75*4cc286f3SDominic Spill 	"Master to Slave"
76*4cc286f3SDominic Spill };
77*4cc286f3SDominic Spill 
78*4cc286f3SDominic Spill static const true_false_string clock_bits = {
79*4cc286f3SDominic Spill 	"27",
80*4cc286f3SDominic Spill 	"6"
81*4cc286f3SDominic Spill };
82*4cc286f3SDominic Spill 
83*4cc286f3SDominic Spill static const true_false_string address_bits = {
84*4cc286f3SDominic Spill 	"48 (NAP known)",
85*4cc286f3SDominic Spill 	"32 (NAP unknown)"
86*4cc286f3SDominic Spill };
87*4cc286f3SDominic Spill 
88*4cc286f3SDominic Spill static const value_string packet_types[] = {
89*4cc286f3SDominic Spill 	/* generic names for unknown logical transport */
90*4cc286f3SDominic Spill 	{ 0x0, "NULL" },
91*4cc286f3SDominic Spill 	{ 0x1, "POLL" },
92*4cc286f3SDominic Spill 	{ 0x2, "FHS" },
93*4cc286f3SDominic Spill 	{ 0x3, "DM1" },
94*4cc286f3SDominic Spill 	{ 0x4, "DH1/2-DH1" },
95*4cc286f3SDominic Spill 	{ 0x5, "HV1" },
96*4cc286f3SDominic Spill 	{ 0x6, "HV2/2-EV3" },
97*4cc286f3SDominic Spill 	{ 0x7, "HV3/EV3/3-EV3" },
98*4cc286f3SDominic Spill 	{ 0x8, "DV/3-DH1" },
99*4cc286f3SDominic Spill 	{ 0x9, "AUX1" },
100*4cc286f3SDominic Spill 	{ 0xa, "DM3/2-DH3" },
101*4cc286f3SDominic Spill 	{ 0xb, "DH3/3-DH3" },
102*4cc286f3SDominic Spill 	{ 0xc, "EV4/2-EV5" },
103*4cc286f3SDominic Spill 	{ 0xd, "EV5/3-EV5" },
104*4cc286f3SDominic Spill 	{ 0xe, "DM5/2-DH5" },
105*4cc286f3SDominic Spill 	{ 0xf, "DH5/3-DH5" },
106*4cc286f3SDominic Spill 	{ 0, NULL }
107*4cc286f3SDominic Spill };
108*4cc286f3SDominic Spill 
109*4cc286f3SDominic Spill static const value_string sr_modes[] = {
110*4cc286f3SDominic Spill 	{ 0x0, "R0" },
111*4cc286f3SDominic Spill 	{ 0x1, "R1" },
112*4cc286f3SDominic Spill 	{ 0x2, "R2" },
113*4cc286f3SDominic Spill 	{ 0x3, "Reserved" },
114*4cc286f3SDominic Spill 	{ 0, NULL }
115*4cc286f3SDominic Spill };
116*4cc286f3SDominic Spill 
117*4cc286f3SDominic Spill static const range_string ps_modes[] = {
118*4cc286f3SDominic Spill 	{ 0x0, 0x0, "Mandatory scan mode" },
119*4cc286f3SDominic Spill 	{ 0x1, 0x7, "Reserved" },
120*4cc286f3SDominic Spill 	{ 0, 0, NULL }
121*4cc286f3SDominic Spill };
122*4cc286f3SDominic Spill 
123*4cc286f3SDominic Spill static const value_string llid_codes[] = {
124*4cc286f3SDominic Spill 	{ 0x0, "undefined" },
125*4cc286f3SDominic Spill 	{ 0x1, "Continuation fragment of an L2CAP message (ACL-U)" },
126*4cc286f3SDominic Spill 	{ 0x2, "Start of an L2CAP message or no fragmentation (ACL-U)" },
127*4cc286f3SDominic Spill 	{ 0x3, "LMP message (ACL-C)" },
128*4cc286f3SDominic Spill 	{ 0, NULL }
129*4cc286f3SDominic Spill };
130*4cc286f3SDominic Spill 
131*4cc286f3SDominic Spill /* initialize the subtree pointers */
132*4cc286f3SDominic Spill static gint ett_btbb = -1;
133*4cc286f3SDominic Spill static gint ett_btbb_meta = -1;
134*4cc286f3SDominic Spill static gint ett_btbb_pkthdr = -1;
135*4cc286f3SDominic Spill static gint ett_btbb_flags = -1;
136*4cc286f3SDominic Spill static gint ett_btbb_payload = -1;
137*4cc286f3SDominic Spill static gint ett_btbb_pldhdr = -1;
138*4cc286f3SDominic Spill 
139*4cc286f3SDominic Spill /* subdissectors */
140*4cc286f3SDominic Spill static dissector_handle_t btbrlmp_handle = NULL;
141*4cc286f3SDominic Spill static dissector_handle_t btl2cap_handle = NULL;
142*4cc286f3SDominic Spill 
143*4cc286f3SDominic Spill /* packet header flags */
144*4cc286f3SDominic Spill static const int *flag_fields[] = {
145*4cc286f3SDominic Spill 	&hf_btbb_flow,
146*4cc286f3SDominic Spill 	&hf_btbb_arqn,
147*4cc286f3SDominic Spill 	&hf_btbb_seqn,
148*4cc286f3SDominic Spill 	NULL
149*4cc286f3SDominic Spill };
150*4cc286f3SDominic Spill 
151*4cc286f3SDominic Spill /* one byte payload header */
152*4cc286f3SDominic Spill int
dissect_payload_header1(proto_tree * tree,tvbuff_t * tvb,int offset)153*4cc286f3SDominic Spill dissect_payload_header1(proto_tree *tree, tvbuff_t *tvb, int offset)
154*4cc286f3SDominic Spill {
155*4cc286f3SDominic Spill 	proto_item *hdr_item;
156*4cc286f3SDominic Spill 	proto_tree *hdr_tree;
157*4cc286f3SDominic Spill 
158*4cc286f3SDominic Spill 	DISSECTOR_ASSERT(tvb_reported_length_remaining(tvb, offset) >= 1);
159*4cc286f3SDominic Spill 
160*4cc286f3SDominic Spill 	hdr_item = proto_tree_add_item(tree, hf_btbb_pldhdr, tvb, offset, 1, ENC_NA);
161*4cc286f3SDominic Spill 	hdr_tree = proto_item_add_subtree(hdr_item, ett_btbb_pldhdr);
162*4cc286f3SDominic Spill 
163*4cc286f3SDominic Spill 	proto_tree_add_item(hdr_tree, hf_btbb_llid, tvb, offset, 1, ENC_NA);
164*4cc286f3SDominic Spill 	proto_tree_add_item(hdr_tree, hf_btbb_pldflow, tvb, offset, 1, ENC_NA);
165*4cc286f3SDominic Spill 	proto_tree_add_item(hdr_tree, hf_btbb_length, tvb, offset, 1, ENC_NA);
166*4cc286f3SDominic Spill 
167*4cc286f3SDominic Spill 	/* payload length */
168*4cc286f3SDominic Spill 	return tvb_get_guint8(tvb, offset) >> 3;
169*4cc286f3SDominic Spill }
170*4cc286f3SDominic Spill 
171*4cc286f3SDominic Spill void
dissect_fhs(proto_tree * tree,tvbuff_t * tvb,int offset)172*4cc286f3SDominic Spill dissect_fhs(proto_tree *tree, tvbuff_t *tvb, int offset)
173*4cc286f3SDominic Spill {
174*4cc286f3SDominic Spill 	proto_item *fhs_item, *psmode_item;
175*4cc286f3SDominic Spill 	proto_tree *fhs_tree;
176*4cc286f3SDominic Spill     const gchar *description;
177*4cc286f3SDominic Spill 	guint8 psmode;
178*4cc286f3SDominic Spill 
179*4cc286f3SDominic Spill 	DISSECTOR_ASSERT(tvb_reported_length_remaining(tvb, offset) == 20);
180*4cc286f3SDominic Spill 
181*4cc286f3SDominic Spill 	fhs_item = proto_tree_add_item(tree, hf_btbb_payload, tvb, offset, -1, ENC_NA);
182*4cc286f3SDominic Spill 	fhs_tree = proto_item_add_subtree(fhs_item, ett_btbb_payload);
183*4cc286f3SDominic Spill 
184*4cc286f3SDominic Spill 	/* Use proto_tree_add_bits_item() to get around 32bit limit on bitmasks */
185*4cc286f3SDominic Spill 	proto_tree_add_bits_item(fhs_tree, hf_btbb_fhs_parity, tvb, offset*8, 34, ENC_LITTLE_ENDIAN);
186*4cc286f3SDominic Spill 	/* proto_tree_add_item(fhs_tree, hf_btbb_fhs_parity, tvb, offset, 5, ENC_LITTLE_ENDIAN); */
187*4cc286f3SDominic Spill 	offset += 4;
188*4cc286f3SDominic Spill 
189*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_fhs_lap, tvb, offset, 4, ENC_LITTLE_ENDIAN);
190*4cc286f3SDominic Spill 	offset += 3;
191*4cc286f3SDominic Spill 
192*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_fhs_eir, tvb, offset, 1, ENC_NA);
193*4cc286f3SDominic Spill 	/* skipping 1 undefined bit */
194*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_fhs_sr, tvb, offset, 1, ENC_NA);
195*4cc286f3SDominic Spill 	/* skipping 2 reserved bits */
196*4cc286f3SDominic Spill 	offset += 1;
197*4cc286f3SDominic Spill 
198*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_fhs_uap, tvb, offset, 1, ENC_NA);
199*4cc286f3SDominic Spill 	offset += 1;
200*4cc286f3SDominic Spill 
201*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_fhs_nap, tvb, offset, 2, ENC_LITTLE_ENDIAN);
202*4cc286f3SDominic Spill 	offset += 2;
203*4cc286f3SDominic Spill 
204*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_fhs_class, tvb, offset, 3, ENC_LITTLE_ENDIAN);
205*4cc286f3SDominic Spill 	offset += 3;
206*4cc286f3SDominic Spill 
207*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_fhs_ltaddr, tvb, offset, 1, ENC_NA);
208*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_fhs_clk, tvb, offset, 4, ENC_LITTLE_ENDIAN);
209*4cc286f3SDominic Spill 	offset += 3;
210*4cc286f3SDominic Spill 
211*4cc286f3SDominic Spill 	psmode = tvb_get_guint8(tvb, offset);
212*4cc286f3SDominic Spill 	description = try_rval_to_str(psmode, ps_modes);
213*4cc286f3SDominic Spill 	psmode_item = proto_tree_add_item(fhs_tree, hf_btbb_fhs_psmode, tvb, offset, 1, ENC_NA);
214*4cc286f3SDominic Spill 	if (description)
215*4cc286f3SDominic Spill         proto_item_append_text(psmode_item, " (%s)", description);
216*4cc286f3SDominic Spill 	offset += 1;
217*4cc286f3SDominic Spill 
218*4cc286f3SDominic Spill 	proto_tree_add_item(fhs_tree, hf_btbb_crc, tvb, offset, 2, ENC_LITTLE_ENDIAN);
219*4cc286f3SDominic Spill 	offset += 2;
220*4cc286f3SDominic Spill }
221*4cc286f3SDominic Spill 
222*4cc286f3SDominic Spill void
dissect_dm1(proto_tree * tree,tvbuff_t * tvb,packet_info * pinfo,int offset)223*4cc286f3SDominic Spill dissect_dm1(proto_tree *tree, tvbuff_t *tvb, packet_info *pinfo, int offset)
224*4cc286f3SDominic Spill {
225*4cc286f3SDominic Spill 	int len;	/* payload length indicated by payload header */
226*4cc286f3SDominic Spill 	int llid;	/* logical link id */
227*4cc286f3SDominic Spill 	int l2len;	/* length indicated by l2cap header */
228*4cc286f3SDominic Spill 	proto_item *dm1_item;
229*4cc286f3SDominic Spill 	proto_tree *dm1_tree;
230*4cc286f3SDominic Spill 	tvbuff_t *pld_tvb;
231*4cc286f3SDominic Spill 
232*4cc286f3SDominic Spill 	/*
233*4cc286f3SDominic Spill 	 * FIXME
234*4cc286f3SDominic Spill 	 * I'm probably doing a terrible, terrible thing here, but it gets my
235*4cc286f3SDominic Spill 	 * initial test cases working.
236*4cc286f3SDominic Spill 	 */
237*4cc286f3SDominic Spill 	guint16 fake_acl_data;
238*4cc286f3SDominic Spill 
239*4cc286f3SDominic Spill 	DISSECTOR_ASSERT(tvb_reported_length_remaining(tvb, offset) >= 3);
240*4cc286f3SDominic Spill 
241*4cc286f3SDominic Spill 	dm1_item = proto_tree_add_item(tree, hf_btbb_payload, tvb, offset, -1, ENC_NA);
242*4cc286f3SDominic Spill 	dm1_tree = proto_item_add_subtree(dm1_item, ett_btbb_payload);
243*4cc286f3SDominic Spill 
244*4cc286f3SDominic Spill 	len = dissect_payload_header1(dm1_tree, tvb, offset);
245*4cc286f3SDominic Spill 	llid = tvb_get_guint8(tvb, offset) & 0x3;
246*4cc286f3SDominic Spill 	offset += 1;
247*4cc286f3SDominic Spill 
248*4cc286f3SDominic Spill 	DISSECTOR_ASSERT(tvb_reported_length_remaining(tvb, offset) == len + 2);
249*4cc286f3SDominic Spill 
250*4cc286f3SDominic Spill 	if (llid == 3 && btbrlmp_handle) {
251*4cc286f3SDominic Spill 		/* LMP */
252*4cc286f3SDominic Spill 		pld_tvb = tvb_new_subset(tvb, offset, len, len);
253*4cc286f3SDominic Spill 		call_dissector(btbrlmp_handle, pld_tvb, pinfo, dm1_tree);
254*4cc286f3SDominic Spill 	} else if (llid == 2 && btl2cap_handle) {
255*4cc286f3SDominic Spill 		/* unfragmented L2CAP or start of fragment */
256*4cc286f3SDominic Spill 		l2len = tvb_get_letohs(tvb, offset);
257*4cc286f3SDominic Spill 		if (l2len + 4 == len) {
258*4cc286f3SDominic Spill 			/* unfragmented */
259*4cc286f3SDominic Spill 			pld_tvb = tvb_new_subset(tvb, offset, len, len);
260*4cc286f3SDominic Spill 			call_dissector_with_data(btl2cap_handle, pld_tvb, pinfo, dm1_tree, &fake_acl_data);
261*4cc286f3SDominic Spill 		} else {
262*4cc286f3SDominic Spill 			/* start of fragment */
263*4cc286f3SDominic Spill 			proto_tree_add_item(dm1_tree, hf_btbb_pldbody, tvb, offset, len, ENC_NA);
264*4cc286f3SDominic Spill 		}
265*4cc286f3SDominic Spill 	} else {
266*4cc286f3SDominic Spill 		proto_tree_add_item(dm1_tree, hf_btbb_pldbody, tvb, offset, len, ENC_NA);
267*4cc286f3SDominic Spill 	}
268*4cc286f3SDominic Spill 	offset += len;
269*4cc286f3SDominic Spill 
270*4cc286f3SDominic Spill 	proto_tree_add_item(dm1_tree, hf_btbb_crc, tvb, offset, 2, ENC_LITTLE_ENDIAN);
271*4cc286f3SDominic Spill 	offset += 2;
272*4cc286f3SDominic Spill }
273*4cc286f3SDominic Spill 
274*4cc286f3SDominic Spill /* dissect a packet */
275*4cc286f3SDominic Spill static int
dissect_btbb(tvbuff_t * tvb,packet_info * pinfo,proto_tree * tree,void * data _U_)276*4cc286f3SDominic Spill dissect_btbb(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_)
277*4cc286f3SDominic Spill {
278*4cc286f3SDominic Spill 	proto_item *btbb_item, *meta_item, *pkthdr_item;
279*4cc286f3SDominic Spill 	proto_tree *btbb_tree, *meta_tree, *pkthdr_tree;
280*4cc286f3SDominic Spill 	int offset;
281*4cc286f3SDominic Spill 	/* Avoid error: 'type' may be used uninitialized in this function */
282*4cc286f3SDominic Spill 	guint8 type = 0xff;
283*4cc286f3SDominic Spill 	const gchar *info;
284*4cc286f3SDominic Spill 
285*4cc286f3SDominic Spill 	/* sanity check: length */
286*4cc286f3SDominic Spill 	if (tvb_reported_length(tvb) > 0 && tvb_reported_length(tvb) < 9)
287*4cc286f3SDominic Spill 		/* bad length: look for a different dissector */
288*4cc286f3SDominic Spill 		return 0;
289*4cc286f3SDominic Spill 
290*4cc286f3SDominic Spill 	/* maybe should verify HEC */
291*4cc286f3SDominic Spill 
292*4cc286f3SDominic Spill 	/* make entries in protocol column and info column on summary display */
293*4cc286f3SDominic Spill 	col_set_str(pinfo->cinfo, COL_PROTOCOL, "Bluetooth");
294*4cc286f3SDominic Spill 
295*4cc286f3SDominic Spill 	if (tvb_reported_length(tvb) == 0) {
296*4cc286f3SDominic Spill 		info = "ID";
297*4cc286f3SDominic Spill 	} else {
298*4cc286f3SDominic Spill 		type = (tvb_get_guint8(tvb, 6) >> 3) & 0x0f;
299*4cc286f3SDominic Spill 		info = val_to_str(type, packet_types, "Unknown type: 0x%x");
300*4cc286f3SDominic Spill 	}
301*4cc286f3SDominic Spill 
302*4cc286f3SDominic Spill 	col_clear(pinfo->cinfo, COL_INFO);
303*4cc286f3SDominic Spill 	col_add_str(pinfo->cinfo, COL_INFO, info);
304*4cc286f3SDominic Spill 
305*4cc286f3SDominic Spill 	/* see if we are being asked for details */
306*4cc286f3SDominic Spill 	if (tree) {
307*4cc286f3SDominic Spill 
308*4cc286f3SDominic Spill 		/* create display subtree for the protocol */
309*4cc286f3SDominic Spill 		offset = 0;
310*4cc286f3SDominic Spill 		btbb_item = proto_tree_add_item(tree, proto_btbb, tvb, offset, -1, ENC_NA);
311*4cc286f3SDominic Spill 		btbb_tree = proto_item_add_subtree(btbb_item, ett_btbb);
312*4cc286f3SDominic Spill 
313*4cc286f3SDominic Spill 		/* ID packets have no header, no payload */
314*4cc286f3SDominic Spill 		if (tvb_reported_length(tvb) == 0)
315*4cc286f3SDominic Spill 			return 1;
316*4cc286f3SDominic Spill 
317*4cc286f3SDominic Spill 		/* meta data */
318*4cc286f3SDominic Spill 		meta_item = proto_tree_add_item(btbb_tree, hf_btbb_meta, tvb, offset, 3, ENC_NA);
319*4cc286f3SDominic Spill 		meta_tree = proto_item_add_subtree(meta_item, ett_btbb_meta);
320*4cc286f3SDominic Spill 
321*4cc286f3SDominic Spill 		proto_tree_add_item(meta_tree, hf_btbb_dir, tvb, offset, 1, ENC_NA);
322*4cc286f3SDominic Spill 		proto_tree_add_item(meta_tree, hf_btbb_clk, tvb, offset, 4, ENC_LITTLE_ENDIAN);
323*4cc286f3SDominic Spill 		offset += 4;
324*4cc286f3SDominic Spill 
325*4cc286f3SDominic Spill 		proto_tree_add_item(meta_tree, hf_btbb_channel, tvb, offset, 1, ENC_NA);
326*4cc286f3SDominic Spill 		offset += 1;
327*4cc286f3SDominic Spill 
328*4cc286f3SDominic Spill 		proto_tree_add_item(meta_tree, hf_btbb_clkbits, tvb, offset, 1, ENC_NA);
329*4cc286f3SDominic Spill 		proto_tree_add_item(meta_tree, hf_btbb_addrbits, tvb, offset, 1, ENC_NA);
330*4cc286f3SDominic Spill 		offset += 1;
331*4cc286f3SDominic Spill 
332*4cc286f3SDominic Spill 		/* packet header */
333*4cc286f3SDominic Spill 		pkthdr_item = proto_tree_add_item(btbb_tree, hf_btbb_pkthdr, tvb, offset, 3, ENC_NA);
334*4cc286f3SDominic Spill 		pkthdr_tree = proto_item_add_subtree(pkthdr_item, ett_btbb_pkthdr);
335*4cc286f3SDominic Spill 
336*4cc286f3SDominic Spill 		proto_tree_add_item(pkthdr_tree, hf_btbb_ltaddr, tvb, offset, 1, ENC_NA);
337*4cc286f3SDominic Spill 		proto_tree_add_item(pkthdr_tree, hf_btbb_type, tvb, offset, 1, ENC_NA);
338*4cc286f3SDominic Spill 		offset += 1;
339*4cc286f3SDominic Spill 		proto_tree_add_bitmask(pkthdr_tree, tvb, offset, hf_btbb_flags,
340*4cc286f3SDominic Spill 			ett_btbb_flags, flag_fields, ENC_NA);
341*4cc286f3SDominic Spill 		offset += 1;
342*4cc286f3SDominic Spill 		proto_tree_add_item(pkthdr_tree, hf_btbb_hec, tvb, offset, 1, ENC_NA);
343*4cc286f3SDominic Spill 		offset += 1;
344*4cc286f3SDominic Spill 
345*4cc286f3SDominic Spill 		/* payload */
346*4cc286f3SDominic Spill 		switch (type) {
347*4cc286f3SDominic Spill 		case 0x0: /* NULL */
348*4cc286f3SDominic Spill 		case 0x1: /* POLL */
349*4cc286f3SDominic Spill 			break;
350*4cc286f3SDominic Spill 		case 0x2: /* FHS */
351*4cc286f3SDominic Spill 			dissect_fhs(btbb_tree, tvb, offset);
352*4cc286f3SDominic Spill 			break;
353*4cc286f3SDominic Spill 		case 0x3: /* DM1 */
354*4cc286f3SDominic Spill 			dissect_dm1(btbb_tree, tvb, pinfo, offset);
355*4cc286f3SDominic Spill 			break;
356*4cc286f3SDominic Spill 		case 0x4: /* DH1/2-DH1 */
357*4cc286f3SDominic Spill 			dissect_dm1(btbb_tree, tvb, pinfo, offset);
358*4cc286f3SDominic Spill 			break;
359*4cc286f3SDominic Spill 		case 0x5: /* HV1 */
360*4cc286f3SDominic Spill 		case 0x6: /* HV2/2-EV3 */
361*4cc286f3SDominic Spill 		case 0x7: /* HV3/EV3/3-EV3 */
362*4cc286f3SDominic Spill 		case 0x8: /* DV/3-DH1 */
363*4cc286f3SDominic Spill 		case 0x9: /* AUX1 */
364*4cc286f3SDominic Spill 		case 0xa: /* DM3/2-DH3 */
365*4cc286f3SDominic Spill 		case 0xb: /* DH3/3-DH3 */
366*4cc286f3SDominic Spill 		case 0xc: /* EV4/2-EV5 */
367*4cc286f3SDominic Spill 		case 0xd: /* EV5/3-EV5 */
368*4cc286f3SDominic Spill 		case 0xe: /* DM5/2-DH5 */
369*4cc286f3SDominic Spill 		case 0xf: /* DH5/3-DH5 */
370*4cc286f3SDominic Spill 			proto_tree_add_item(btbb_tree, hf_btbb_payload, tvb, offset, -1, ENC_NA);
371*4cc286f3SDominic Spill 			break;
372*4cc286f3SDominic Spill 		default:
373*4cc286f3SDominic Spill 			break;
374*4cc286f3SDominic Spill 		}
375*4cc286f3SDominic Spill 	}
376*4cc286f3SDominic Spill 
377*4cc286f3SDominic Spill 	/* Return the amount of data this dissector was able to dissect */
378*4cc286f3SDominic Spill 	return tvb_reported_length(tvb);
379*4cc286f3SDominic Spill }
380*4cc286f3SDominic Spill 
381*4cc286f3SDominic Spill /* register the protocol with Wireshark */
382*4cc286f3SDominic Spill void
proto_register_btbb(void)383*4cc286f3SDominic Spill proto_register_btbb(void)
384*4cc286f3SDominic Spill {
385*4cc286f3SDominic Spill 	/* list of fields */
386*4cc286f3SDominic Spill 	static hf_register_info hf[] = {
387*4cc286f3SDominic Spill 		{ &hf_btbb_meta,
388*4cc286f3SDominic Spill 			{ "Meta Data", "btbb.meta",
389*4cc286f3SDominic Spill 			FT_NONE, BASE_NONE, NULL, 0x0,
390*4cc286f3SDominic Spill 			"Meta Data About the Packet", HFILL }
391*4cc286f3SDominic Spill 		},
392*4cc286f3SDominic Spill 		{ &hf_btbb_dir,
393*4cc286f3SDominic Spill 			{ "Direction", "btbb.dir",
394*4cc286f3SDominic Spill 			FT_BOOLEAN, 8, TFS(&direction), 0x01,
395*4cc286f3SDominic Spill 			"Direction of Transmission", HFILL }
396*4cc286f3SDominic Spill 		},
397*4cc286f3SDominic Spill 		{ &hf_btbb_clk,
398*4cc286f3SDominic Spill 			{ "CLK", "btbb.clk",
399*4cc286f3SDominic Spill 			FT_UINT32, BASE_HEX, NULL, 0x0,
400*4cc286f3SDominic Spill 			"Clock bits 1 through 27", HFILL }
401*4cc286f3SDominic Spill 		},
402*4cc286f3SDominic Spill 		{ &hf_btbb_channel,
403*4cc286f3SDominic Spill 			{ "Channel", "btbb.channel",
404*4cc286f3SDominic Spill 			FT_UINT8, BASE_DEC, NULL, 0x0,
405*4cc286f3SDominic Spill 			"Channel (0-78)", HFILL }
406*4cc286f3SDominic Spill 		},
407*4cc286f3SDominic Spill 		{ &hf_btbb_clkbits,
408*4cc286f3SDominic Spill 			{ "Known Clock Bits", "btbb.clkbits",
409*4cc286f3SDominic Spill 			FT_BOOLEAN, 8, TFS(&clock_bits), 0x01,
410*4cc286f3SDominic Spill 			"Number of Known Master CLK Bits (6 or 27)", HFILL }
411*4cc286f3SDominic Spill 		},
412*4cc286f3SDominic Spill 		{ &hf_btbb_addrbits,
413*4cc286f3SDominic Spill 			{ "Known Address Bits", "btbb.addrbits",
414*4cc286f3SDominic Spill 			FT_BOOLEAN, 8, TFS(&address_bits), 0x02,
415*4cc286f3SDominic Spill 			"Number of Known Bits of BD_ADDR (32 or 48)", HFILL }
416*4cc286f3SDominic Spill 		},
417*4cc286f3SDominic Spill 		{ &hf_btbb_pkthdr,
418*4cc286f3SDominic Spill 			{ "Packet Header", "btbb.pkthdr",
419*4cc286f3SDominic Spill 			FT_NONE, BASE_NONE, NULL, 0x0,
420*4cc286f3SDominic Spill 			"Bluetooth Baseband Packet Header", HFILL }
421*4cc286f3SDominic Spill 		},
422*4cc286f3SDominic Spill 		{ &hf_btbb_ltaddr,
423*4cc286f3SDominic Spill 			{ "LT_ADDR", "btbb.lt_addr",
424*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, NULL, 0x07,
425*4cc286f3SDominic Spill 			"Logical Transport Address", HFILL }
426*4cc286f3SDominic Spill 		},
427*4cc286f3SDominic Spill 		{ &hf_btbb_type,
428*4cc286f3SDominic Spill 			{ "TYPE", "btbb.type",
429*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, VALS(packet_types), 0x78,
430*4cc286f3SDominic Spill 			"Packet Type", HFILL }
431*4cc286f3SDominic Spill 		},
432*4cc286f3SDominic Spill 		{ &hf_btbb_flags,
433*4cc286f3SDominic Spill 			{ "Flags", "btbb.flags",
434*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, NULL, 0x0,
435*4cc286f3SDominic Spill 			"Packet Header Flags", HFILL }
436*4cc286f3SDominic Spill 		},
437*4cc286f3SDominic Spill 		{ &hf_btbb_flow,
438*4cc286f3SDominic Spill 			{ "FLOW", "btbb.flow",
439*4cc286f3SDominic Spill 			FT_BOOLEAN, 8, NULL, 0x01,
440*4cc286f3SDominic Spill 			"Flow control indication", HFILL }
441*4cc286f3SDominic Spill 		},
442*4cc286f3SDominic Spill 		{ &hf_btbb_arqn,
443*4cc286f3SDominic Spill 			{ "ARQN", "btbb.arqn",
444*4cc286f3SDominic Spill 			FT_BOOLEAN, 8, NULL, 0x02,
445*4cc286f3SDominic Spill 			"Acknowledgment indication", HFILL }
446*4cc286f3SDominic Spill 		},
447*4cc286f3SDominic Spill 		{ &hf_btbb_seqn,
448*4cc286f3SDominic Spill 			{ "SEQN", "btbb.seqn",
449*4cc286f3SDominic Spill 			FT_BOOLEAN, 8, NULL, 0x04,
450*4cc286f3SDominic Spill 			"Sequence number", HFILL }
451*4cc286f3SDominic Spill 		},
452*4cc286f3SDominic Spill 		{ &hf_btbb_hec,
453*4cc286f3SDominic Spill 			{ "HEC", "btbb.lt_addr",
454*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, NULL, 0x0,
455*4cc286f3SDominic Spill 			"Header Error Check", HFILL }
456*4cc286f3SDominic Spill 		},
457*4cc286f3SDominic Spill 		{ &hf_btbb_payload,
458*4cc286f3SDominic Spill 			{ "Payload", "btbb.payload",
459*4cc286f3SDominic Spill 			FT_NONE, BASE_NONE, NULL, 0x0,
460*4cc286f3SDominic Spill 			NULL, HFILL }
461*4cc286f3SDominic Spill 		},
462*4cc286f3SDominic Spill 		{ &hf_btbb_llid,
463*4cc286f3SDominic Spill 			{ "LLID", "btbb.llid",
464*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, VALS(llid_codes), 0x03,
465*4cc286f3SDominic Spill 			"Logical Link ID", HFILL }
466*4cc286f3SDominic Spill 		},
467*4cc286f3SDominic Spill 		{ &hf_btbb_pldflow,
468*4cc286f3SDominic Spill 			{ "Flow", "btbb.flow",
469*4cc286f3SDominic Spill 			FT_BOOLEAN, 8, NULL, 0x04,
470*4cc286f3SDominic Spill 			"Payload Flow indication", HFILL }
471*4cc286f3SDominic Spill 		},
472*4cc286f3SDominic Spill 		{ &hf_btbb_length,
473*4cc286f3SDominic Spill 			{ "Length", "btbb.length",
474*4cc286f3SDominic Spill 			FT_UINT8, BASE_DEC, NULL, 0xf8,
475*4cc286f3SDominic Spill 			"Payload Length", HFILL }
476*4cc286f3SDominic Spill 		},
477*4cc286f3SDominic Spill 		{ &hf_btbb_pldhdr,
478*4cc286f3SDominic Spill 			{ "Payload Header", "btbb.pldhdr",
479*4cc286f3SDominic Spill 			FT_NONE, BASE_NONE, NULL, 0x0,
480*4cc286f3SDominic Spill 			NULL, HFILL }
481*4cc286f3SDominic Spill 		},
482*4cc286f3SDominic Spill 		{ &hf_btbb_pldbody,
483*4cc286f3SDominic Spill 			{ "Payload Body", "btbb.pldbody",
484*4cc286f3SDominic Spill 			FT_BYTES, BASE_NONE, NULL, 0x0,
485*4cc286f3SDominic Spill 			NULL, HFILL }
486*4cc286f3SDominic Spill 		},
487*4cc286f3SDominic Spill 		{ &hf_btbb_crc,
488*4cc286f3SDominic Spill 			{ "CRC", "btbb.crc",
489*4cc286f3SDominic Spill 			FT_UINT16, BASE_HEX, NULL, 0x0,
490*4cc286f3SDominic Spill 			"Payload CRC", HFILL }
491*4cc286f3SDominic Spill 		},
492*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_parity,
493*4cc286f3SDominic Spill 			{ "Parity", "btbb.parity",
494*4cc286f3SDominic Spill 			/* FIXME this doesn't work because bitmasks can only be 32 bits */
495*4cc286f3SDominic Spill 			FT_UINT64, BASE_HEX, NULL, /*0x00000003ffffffffULL,*/ 0x0,
496*4cc286f3SDominic Spill 			"LAP parity", HFILL }
497*4cc286f3SDominic Spill 		},
498*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_lap,
499*4cc286f3SDominic Spill 			{ "LAP", "btbb.lap",
500*4cc286f3SDominic Spill 			FT_UINT24, BASE_HEX, NULL, 0x03fffffc,
501*4cc286f3SDominic Spill 			"Lower Address Part", HFILL }
502*4cc286f3SDominic Spill 		},
503*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_eir,
504*4cc286f3SDominic Spill 			{ "EIR", "btbb.eir",
505*4cc286f3SDominic Spill 			FT_BOOLEAN, 8, NULL, 0x04,
506*4cc286f3SDominic Spill 			"Extended Inquiry Response packet may follow", HFILL }
507*4cc286f3SDominic Spill 		},
508*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_sr,
509*4cc286f3SDominic Spill 			{ "SR", "btbb.sr",
510*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, VALS(sr_modes), 0x30,
511*4cc286f3SDominic Spill 			"Scan Repetition", HFILL }
512*4cc286f3SDominic Spill 		},
513*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_uap,
514*4cc286f3SDominic Spill 			{ "UAP", "btbb.uap",
515*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, NULL, 0x0,
516*4cc286f3SDominic Spill 			"Upper Address Part", HFILL }
517*4cc286f3SDominic Spill 		},
518*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_nap,
519*4cc286f3SDominic Spill 			{ "NAP", "btbb.nap",
520*4cc286f3SDominic Spill 			FT_UINT16, BASE_HEX, NULL, 0x0,
521*4cc286f3SDominic Spill 			"Non-Significant Address Part", HFILL }
522*4cc286f3SDominic Spill 		},
523*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_class, /* FIXME break out further */
524*4cc286f3SDominic Spill 			{ "Class of Device", "btbb.class",
525*4cc286f3SDominic Spill 			FT_UINT24, BASE_HEX, NULL, 0x0,
526*4cc286f3SDominic Spill 			NULL, HFILL }
527*4cc286f3SDominic Spill 		},
528*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_ltaddr,
529*4cc286f3SDominic Spill 			{ "LT_ADDR", "btbb.lt_addr",
530*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, NULL, 0x07,
531*4cc286f3SDominic Spill 			"Logical Transport Address", HFILL }
532*4cc286f3SDominic Spill 		},
533*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_clk,
534*4cc286f3SDominic Spill 			{ "CLK", "btbb.clk",
535*4cc286f3SDominic Spill 			FT_UINT32, BASE_HEX, NULL, 0x1ffffff8,
536*4cc286f3SDominic Spill 			"Clock bits 2 through 27", HFILL }
537*4cc286f3SDominic Spill 		},
538*4cc286f3SDominic Spill 		{ &hf_btbb_fhs_psmode,
539*4cc286f3SDominic Spill 			{ "Page Scan Mode", "btbb.psmode",
540*4cc286f3SDominic Spill 			FT_UINT8, BASE_HEX, NULL, 0xe0,
541*4cc286f3SDominic Spill 			NULL, HFILL }
542*4cc286f3SDominic Spill 		},
543*4cc286f3SDominic Spill 	};
544*4cc286f3SDominic Spill 
545*4cc286f3SDominic Spill 	/* protocol subtree arrays */
546*4cc286f3SDominic Spill 	static gint *ett[] = {
547*4cc286f3SDominic Spill 		&ett_btbb,
548*4cc286f3SDominic Spill 		&ett_btbb_meta,
549*4cc286f3SDominic Spill 		&ett_btbb_pkthdr,
550*4cc286f3SDominic Spill 		&ett_btbb_flags,
551*4cc286f3SDominic Spill 		&ett_btbb_payload,
552*4cc286f3SDominic Spill 		&ett_btbb_pldhdr,
553*4cc286f3SDominic Spill 	};
554*4cc286f3SDominic Spill 
555*4cc286f3SDominic Spill 	/* register the protocol name and description */
556*4cc286f3SDominic Spill 	proto_btbb = proto_register_protocol(
557*4cc286f3SDominic Spill 		"Bluetooth Baseband",	/* full name */
558*4cc286f3SDominic Spill 		"BT Baseband",			/* short name */
559*4cc286f3SDominic Spill 		"btbb"			/* abbreviation (e.g. for filters) */
560*4cc286f3SDominic Spill 		);
561*4cc286f3SDominic Spill 
562*4cc286f3SDominic Spill 	/* register the header fields and subtrees used */
563*4cc286f3SDominic Spill 	proto_register_field_array(proto_btbb, hf, array_length(hf));
564*4cc286f3SDominic Spill 	proto_register_subtree_array(ett, array_length(ett));
565*4cc286f3SDominic Spill }
566*4cc286f3SDominic Spill 
567*4cc286f3SDominic Spill void
proto_reg_handoff_btbb(void)568*4cc286f3SDominic Spill proto_reg_handoff_btbb(void)
569*4cc286f3SDominic Spill {
570*4cc286f3SDominic Spill 	dissector_handle_t btbb_handle;
571*4cc286f3SDominic Spill 
572*4cc286f3SDominic Spill 	btbb_handle = new_create_dissector_handle(dissect_btbb, proto_btbb);
573*4cc286f3SDominic Spill 	/* hijacking this ethertype */
574*4cc286f3SDominic Spill 	dissector_add_uint("ethertype", 0xFFF0, btbb_handle);
575*4cc286f3SDominic Spill 	/* dissector_add_uint("wtap_encap", WTAP_ENCAP_BLUETOOTH_BASEBAND, btbb_handle); */
576*4cc286f3SDominic Spill 
577*4cc286f3SDominic Spill 	btbrlmp_handle = find_dissector("btbrlmp");
578*4cc286f3SDominic Spill 	btl2cap_handle = find_dissector("btl2cap");
579*4cc286f3SDominic Spill }
580*4cc286f3SDominic Spill 
581*4cc286f3SDominic Spill /*
582*4cc286f3SDominic Spill  * Editor modelines  -  http://www.wireshark.org/tools/modelines.html
583*4cc286f3SDominic Spill  *
584*4cc286f3SDominic Spill  * Local variables:
585*4cc286f3SDominic Spill  * c-basic-offset: 4
586*4cc286f3SDominic Spill  * tab-width: 8
587*4cc286f3SDominic Spill  * indent-tabs-mode: nil
588*4cc286f3SDominic Spill  * End:
589*4cc286f3SDominic Spill  *
590*4cc286f3SDominic Spill  * vi: set shiftwidth=4 tabstop=8 expandtab:
591*4cc286f3SDominic Spill  * :indentSize=4:tabSize=8:noTabs=true:
592*4cc286f3SDominic Spill  */
593