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