1 /******************************************************************************
2  *
3  *  Copyright 2014 Google, Inc.
4  *
5  *  Licensed under the Apache License, Version 2.0 (the "License");
6  *  you may not use this file except in compliance with the License.
7  *  You may obtain a copy of the License at:
8  *
9  *  http://www.apache.org/licenses/LICENSE-2.0
10  *
11  *  Unless required by applicable law or agreed to in writing, software
12  *  distributed under the License is distributed on an "AS IS" BASIS,
13  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  *  See the License for the specific language governing permissions and
15  *  limitations under the License.
16  *
17  ******************************************************************************/
18 
19 #include "support/adapter.h"
20 #include "base.h"
21 #include "btcore/include/property.h"
22 #include "support/callbacks.h"
23 
24 static bt_state_t state;
25 static int property_count = 0;
26 static bt_property_t* properties = NULL;
27 static bt_discovery_state_t discovery_state;
28 static bt_acl_state_t acl_state;
29 static bt_bond_state_t bond_state;
30 
31 static void parse_properties(int num_properties, bt_property_t* property);
32 
33 // Returns the current adapter state.
adapter_get_state()34 bt_state_t adapter_get_state() { return state; }
35 
36 // Returns the number of adapter properties.
adapter_get_property_count()37 int adapter_get_property_count() { return property_count; }
38 
39 // Returns the specified property.
adapter_get_property(bt_property_type_t type)40 bt_property_t* adapter_get_property(bt_property_type_t type) {
41   for (int i = 0; i < property_count; ++i) {
42     if (properties[i].type == type) {
43       return &properties[i];
44     }
45   }
46 
47   return NULL;
48 }
49 
50 // Returns the device discovery state.
adapter_get_discovery_state()51 bt_discovery_state_t adapter_get_discovery_state() { return discovery_state; }
52 
53 // Returns the device acl state.
adapter_get_acl_state()54 bt_acl_state_t adapter_get_acl_state() { return acl_state; }
55 
56 // Returns the device bond state.
adapter_get_bond_state()57 bt_bond_state_t adapter_get_bond_state() { return bond_state; }
58 
59 // callback
acl_state_changed(bt_status_t status,RawAddress * remote_bd_addr,bt_acl_state_t state)60 void acl_state_changed(bt_status_t status, RawAddress* remote_bd_addr,
61                        bt_acl_state_t state) {
62   acl_state = state;
63   CALLBACK_RET();
64 }
65 
66 // callback
adapter_properties(bt_status_t status,int num_properties,bt_property_t * new_properties)67 void adapter_properties(bt_status_t status, int num_properties,
68                         bt_property_t* new_properties) {
69   property_free_array(properties, property_count);
70   properties = property_copy_array(new_properties, num_properties);
71   property_count = num_properties;
72 
73   CALLBACK_RET();
74 }
75 
76 // callback
adapter_state_changed(bt_state_t new_state)77 void adapter_state_changed(bt_state_t new_state) {
78   state = new_state;
79   CALLBACK_RET();
80 }
81 
82 // callback
bond_state_changed(bt_status_t status,RawAddress * bdaddr,bt_bond_state_t state)83 void bond_state_changed(bt_status_t status, RawAddress* bdaddr,
84                         bt_bond_state_t state) {
85   char buf[18];
86   bond_state = state;
87 
88   const char* state_name = "Bond state unknown";
89   switch (bond_state) {
90     case BT_BOND_STATE_NONE:
91       state_name = "Bond state none";
92       break;
93 
94     case BT_BOND_STATE_BONDING:
95       state_name = "Bond state bonding";
96       break;
97 
98     case BT_BOND_STATE_BONDED:
99       state_name = "Bond state bonded";
100       break;
101 
102       // default none
103   }
104   fprintf(stdout, "Bond state changed callback addr:%s state:%s\n",
105           bdaddr_to_string(bdaddr, buf, sizeof(buf)), state_name);
106 
107   CALLBACK_RET();
108 }
109 
110 // callback
device_found(int num_properties,bt_property_t * property)111 void device_found(int num_properties, bt_property_t* property) {
112   fprintf(stdout, "Device found num_properties:%d\n", num_properties);
113   parse_properties(num_properties, property);
114 
115   CALLBACK_RET();
116 }
117 
118 // callback
discovery_state_changed(bt_discovery_state_t state)119 void discovery_state_changed(bt_discovery_state_t state) {
120   const char* state_name = "Unknown";
121   discovery_state = state;
122 
123   switch (discovery_state) {
124     case BT_DISCOVERY_STOPPED:
125       state_name = "Discovery stopped";
126       break;
127 
128     case BT_DISCOVERY_STARTED:
129       state_name = "Discovery started";
130       break;
131 
132       // default omitted
133   }
134   fprintf(stdout, "Discover state %s\n", state_name);
135 
136   CALLBACK_RET();
137 }
138 
139 // callback
remote_device_properties(bt_status_t status,RawAddress * bdaddr,int num_properties,bt_property_t * properties)140 void remote_device_properties(bt_status_t status, RawAddress* bdaddr,
141                               int num_properties, bt_property_t* properties) {
142   char buf[18];
143   fprintf(stdout, "Device found bdaddr:%s num_properties:%d\n",
144           bdaddr_to_string(bdaddr, buf, sizeof(buf)), num_properties);
145 
146   parse_properties(num_properties, properties);
147 
148   CALLBACK_RET();
149 }
150 
151 // callback
ssp_request(RawAddress * remote_bd_addr,bt_bdname_t * bd_name,uint32_t cod,bt_ssp_variant_t pairing_variant,uint32_t pass_key)152 void ssp_request(RawAddress* remote_bd_addr, bt_bdname_t* bd_name, uint32_t cod,
153                  bt_ssp_variant_t pairing_variant, uint32_t pass_key) {
154   char* pairing_variant_name = "Unknown";
155 
156   switch (pairing_variant) {
157     case BT_SSP_VARIANT_PASSKEY_CONFIRMATION:
158       pairing_variant_name = "Passkey confirmation";
159       break;
160     case BT_SSP_VARIANT_PASSKEY_ENTRY:
161       pairing_variant_name = "Passkey entry";
162       break;
163 
164     case BT_SSP_VARIANT_CONSENT:
165       pairing_variant_name = "Passkey consent";
166       break;
167 
168     case BT_SSP_VARIANT_PASSKEY_NOTIFICATION:
169       pairing_variant_name = "Passkey notification";
170       break;
171   }
172 
173   fprintf(stdout,
174           "Got ssp request device_class:%u passkey:%x pairing_variant:%s\n",
175           cod, pass_key, pairing_variant_name);
176   char buf[18];
177   fprintf(stdout, "Device found:%s %s\n",
178           bdaddr_to_string(remote_bd_addr, buf, sizeof(buf)), bd_name->name);
179 
180   fprintf(stdout, "auto-accepting bond\n");
181   bool accept = true;
182   int rc = bt_interface->ssp_reply(remote_bd_addr, pairing_variant,
183                                    (uint8_t)accept, pass_key);
184   CALLBACK_RET();
185 }
186 
187 // callback
thread_evt(bt_cb_thread_evt evt)188 void thread_evt(bt_cb_thread_evt evt) { CALLBACK_RET(); }
189 
parse_properties(int num_properties,bt_property_t * property)190 static void parse_properties(int num_properties, bt_property_t* property) {
191   while (num_properties-- > 0) {
192     switch (property->type) {
193       case BT_PROPERTY_BDNAME: {
194         const bt_bdname_t* name = property_as_name(property);
195         if (name) fprintf(stdout, " name:%s\n", name->name);
196       } break;
197 
198       case BT_PROPERTY_BDADDR: {
199         char buf[18];
200         const RawAddress* addr = property_as_addr(property);
201         if (addr)
202           fprintf(stdout, " addr:%s\n",
203                   bdaddr_to_string(addr, buf, sizeof(buf)));
204       } break;
205 
206       case BT_PROPERTY_UUIDS: {
207         size_t num_uuid;
208         const Uuid* uuid = property_as_uuids(property, &num_uuid);
209         if (uuid) {
210           for (size_t i = 0; i < num_uuid; i++) {
211             fprintf(stdout, " uuid:%zd: ", i);
212             for (size_t j = 0; j < sizeof(uuid); j++) {
213               fprintf(stdout, "%02x", uuid->uu[j]);
214             }
215             fprintf(stdout, "\n");
216           }
217         }
218       } break;
219 
220       case BT_PROPERTY_TYPE_OF_DEVICE: {
221         bt_device_type_t device_type = property_as_device_type(property);
222         if (device_type) {
223           const struct {
224             const char* device_type;
225           } device_type_lookup[] = {
226               {"Unknown"},
227               {"Classic Only"},
228               {"BLE Only"},
229               {"Both Classic and BLE"},
230           };
231           int idx = (int)device_type;
232           if (idx > BT_DEVICE_DEVTYPE_DUAL) idx = 0;
233           fprintf(stdout, " device_type:%s\n",
234                   device_type_lookup[idx].device_type);
235         }
236       } break;
237 
238       case BT_PROPERTY_CLASS_OF_DEVICE: {
239         const bt_device_class_t* dc = property_as_device_class(property);
240         int dc_int = device_class_to_int(dc);
241         fprintf(stdout, " device_class:0x%x\n", dc_int);
242       } break;
243 
244       case BT_PROPERTY_REMOTE_RSSI: {
245         int8_t rssi = property_as_rssi(property);
246         fprintf(stdout, " rssi:%d\n", rssi);
247       } break;
248 
249       case BT_PROPERTY_REMOTE_FRIENDLY_NAME: {
250         const bt_bdname_t* name = property_as_name(property);
251         if (name) fprintf(stdout, " remote_name:%s\n", name->name);
252       } break;
253 
254       case BT_PROPERTY_SERVICE_RECORD:
255       case BT_PROPERTY_ADAPTER_SCAN_MODE:
256       case BT_PROPERTY_ADAPTER_BONDED_DEVICES:
257       case BT_PROPERTY_ADAPTER_DISCOVERY_TIMEOUT:
258       case BT_PROPERTY_REMOTE_VERSION_INFO:
259       case BT_PROPERTY_LOCAL_LE_FEATURES:
260       case BT_PROPERTY_REMOTE_DEVICE_TIMESTAMP:
261       default: {
262         fprintf(stderr, "Unhandled property type:%d len:%d\n", property->type,
263                 property->len);
264         uint8_t* p = (uint8_t*)property->val;
265         for (int i = 0; i < property->len; ++i, p++) {
266           fprintf(stderr, " %02x", *p);
267         }
268         if (property->len != 0) fprintf(stderr, "\n");
269       }
270     }
271     property++;
272   }
273 }
274