[Adapt] Fix _bt_get_adapter_visibility API
[platform/core/connectivity/bluetooth-frwk.git] / bt-oal / oal-adapter-mgr.c
1 /*
2  * Open Adaptation Layer (OAL)
3  *
4  * Copyright (c) 2014-2015 Samsung Electronics Co., Ltd.
5  *
6  * Licensed under the Apache License, Version 2.0 (the "License");
7  * you may not use this file except in compliance with the License.
8  * You may obtain a copy of the License at
9  *
10  * http://www.apache.org/licenses/LICENSE-2.0
11  *
12  * Unless required by applicable law or agreed to in writing, software
13  * distributed under the License is distributed on an "AS IS" BASIS,
14  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15  * See the License for the specific language governing permissions and
16  * limitations under the License.
17  *
18  */
19
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <dlog.h>
23 #include <string.h>
24 #include <vconf.h>
25 #include <sys/wait.h>
26
27 #include <bluetooth.h>
28
29 #include "oal-event.h"
30 #include "oal-internal.h"
31 #include "oal-manager.h"
32 #include "oal-hardware.h"
33 #include "oal-common.h"
34 #include "oal-utils.h"
35
36 #define CHECK_MAX(max, x) (((max) > (x)) ? (x) : (max))
37
38 static const bt_interface_t * blued_api;
39
40 static bt_address_t local_address;
41 static char local_name[BT_DEVICE_NAME_LENGTH_MAX + 1] = {'O', 'A', 'L', 0};
42 static char local_version[BT_VERSION_STR_LEN_MAX + 1];
43 static bt_scan_mode_t scan_mode = BT_SCAN_MODE_NONE;
44 static int discoverable_timeout = 0;
45
46 /* Forward declarations */
47 oal_status_t convert_to_oal_status(bt_status_t status);
48 static gboolean retry_enable_adapter(gpointer data);
49 oal_status_t oal_mgr_init_internal(void);
50
51
52 /* Callback registered with Stack */
53 static void cb_adapter_state_change(bt_state_t status);
54 static void cb_adapter_discovery_state_changed(bt_discovery_state_t state);
55 static void cb_adapter_device_found(int num_properties, bt_property_t *properties);
56 static void cb_adapter_properties (bt_status_t status,
57                 int num_properties, bt_property_t *properties);
58 extern void cb_device_properties(bt_status_t status, bt_bdaddr_t *bd_addr,
59                 int num_properties, bt_property_t *properties);
60 extern void cb_device_bond_state_changed(bt_status_t status, bt_bdaddr_t *bd_addr,
61                                         bt_bond_state_t state);
62 extern void cb_device_acl_state_changed(bt_status_t status, bt_bdaddr_t *remote_bd_addr,
63                                             bt_acl_state_t state);
64 extern void cb_device_pin_request(bt_bdaddr_t *bd_addr, bt_bdname_t *bdname, uint32_t device_class);
65 extern void cb_device_ssp_request(bt_bdaddr_t *bd_addr, bt_bdname_t *bdname, uint32_t device_class,
66                         bt_ssp_variant_t pairing_variant, uint32_t pass_key);
67 extern void cb_device_authorize_request(bt_bdaddr_t *remote_bd_addr, bt_service_id_t service_d);
68 extern void cb_device_trust_state_changed(bt_bdaddr_t *remote_bd_addr, bt_device_trust_state_t trusted);
69 #ifdef TIZEN_BT_HAL
70 extern void cb_socket_conn_authorize_request(bt_bdaddr_t *remote_bd_addr, bt_uuid_t *uuid);
71 #endif
72
73 static bt_callbacks_t callbacks = {
74         sizeof(callbacks),
75         cb_adapter_state_change,
76         cb_adapter_properties,
77         cb_device_properties,
78         cb_adapter_device_found,
79         cb_adapter_discovery_state_changed,
80         cb_device_pin_request,
81         cb_device_ssp_request,
82         cb_device_bond_state_changed,
83         cb_device_acl_state_changed,
84         NULL, /* callback_thread_event */
85         NULL, /* dut_mode_recv_callback */
86         NULL, /* le_test_mode_callback*/
87         NULL, /* energy_info_callback */
88         cb_device_authorize_request,
89         cb_device_trust_state_changed,
90 #ifdef TIZEN_BT_HAL
91         cb_socket_conn_authorize_request,
92 #endif
93 };
94
95 oal_status_t adapter_mgr_init(const bt_interface_t * stack_if)
96 {
97         int ret;
98         blued_api = stack_if;
99
100         ret = blued_api->init(&callbacks);
101
102         if (ret != BT_STATUS_SUCCESS) {
103                 BT_ERR("Adapter callback registration failed: [%s]", status2string(ret));
104                 blued_api->cleanup();
105                 return convert_to_oal_status(ret);
106         }
107
108         return OAL_STATUS_SUCCESS;
109 }
110
111 const bt_interface_t* adapter_get_stack_interface(void)
112 {
113         return blued_api;
114 }
115
116 void adapter_mgr_cleanup(void)
117 {
118         /* Nothing to clean yet , do not set blued_api NULL as it will be used to clean Bluedroid states */
119         BT_DBG();
120 }
121
122 oal_status_t adapter_enable(void)
123 {
124         int ret = BT_STATUS_SUCCESS;
125
126         API_TRACE();
127         CHECK_OAL_INITIALIZED();
128         if (OAL_STATUS_SUCCESS != hw_is_module_ready()) {
129                 g_timeout_add(200, retry_enable_adapter, NULL);
130                 return OAL_STATUS_PENDING;
131         }
132
133         ret = blued_api->enable();
134
135         if (ret != BT_STATUS_SUCCESS) {
136                 BT_ERR("Enable failed: [%s]", status2string(ret));
137                 return convert_to_oal_status(ret);
138         }
139
140         return OAL_STATUS_SUCCESS;
141 }
142
143 oal_status_t adapter_disable(void)
144 {
145         int ret;
146
147         API_TRACE();
148
149         CHECK_OAL_INITIALIZED();
150
151         ret = blued_api->disable();
152
153         if (ret != BT_STATUS_SUCCESS) {
154                 BT_ERR("Disable failed: [%s]", status2string(ret));
155                 return convert_to_oal_status(ret);
156         }
157         return OAL_STATUS_SUCCESS;
158 }
159
160 oal_status_t adapter_start_inquiry(void)
161 {
162         int ret;
163
164         API_TRACE();
165
166         CHECK_OAL_INITIALIZED();
167
168         ret = blued_api->start_discovery();
169         if (ret != BT_STATUS_SUCCESS) {
170                 BT_ERR("start_discovery failed: [%s]", status2string(ret));
171                 return convert_to_oal_status(ret);
172         }
173
174         return OAL_STATUS_SUCCESS;
175 }
176
177 oal_status_t adapter_stop_inquiry(void)
178 {
179         int ret;
180
181         API_TRACE();
182
183         CHECK_OAL_INITIALIZED();
184
185         ret = blued_api->cancel_discovery();
186         if (ret != BT_STATUS_SUCCESS) {
187                 BT_ERR("cancel_discovery failed: [%s]", status2string(ret));
188                 return convert_to_oal_status(ret);
189         }
190
191         return OAL_STATUS_SUCCESS;
192 }
193
194 /* Callbacks from Stack */
195 static void cb_adapter_state_change(bt_state_t status)
196 {
197         BT_DBG("+");
198         oal_event_t event;
199
200         event = (BT_STATE_ON == status)?OAL_EVENT_ADAPTER_ENABLED:OAL_EVENT_ADAPTER_DISABLED;
201
202         send_event(event, NULL, 0);
203 }
204
205 static gboolean retry_enable_adapter(gpointer data)
206 {
207         adapter_enable();
208         return FALSE;
209 }
210
211 oal_status_t adapter_get_properties(void)
212 {
213         int ret;
214
215         API_TRACE();
216         CHECK_OAL_INITIALIZED();
217
218         ret = blued_api->get_adapter_properties();
219         if (ret != BT_STATUS_SUCCESS) {
220                 BT_ERR("get_adapter_properties failed: [%s]", status2string(ret));
221                 return convert_to_oal_status(ret);
222         }
223
224         return OAL_STATUS_SUCCESS;
225 }
226
227 oal_status_t adapter_get_address(void)
228 {
229         int ret;
230
231         API_TRACE();
232         CHECK_OAL_INITIALIZED();
233
234         ret = blued_api->get_adapter_property(BT_PROPERTY_BDADDR);
235         if (ret != BT_STATUS_SUCCESS) {
236                 BT_ERR("get_adapter_property failed: [%s]", status2string(ret));
237                 return convert_to_oal_status(ret);
238         }
239
240         return OAL_STATUS_SUCCESS;
241 }
242
243 oal_status_t adapter_get_version(void)
244 {
245         int ret;
246
247         API_TRACE();
248         CHECK_OAL_INITIALIZED();
249
250         ret = blued_api->get_adapter_property(BT_PROPERTY_VERSION);
251         if (ret != BT_STATUS_SUCCESS) {
252                 BT_ERR("get_adapter_property failed: [%s]", status2string(ret));
253                 return convert_to_oal_status(ret);
254         }
255
256         return OAL_STATUS_SUCCESS;
257 }
258
259 oal_status_t adapter_get_name(void)
260 {
261         int ret;
262
263         CHECK_OAL_INITIALIZED();
264
265         API_TRACE();
266
267         ret = blued_api->get_adapter_property(BT_PROPERTY_BDNAME);
268         if (ret != BT_STATUS_SUCCESS) {
269                 BT_ERR("get_adapter_property failed: [%s]", status2string(ret));
270                 return convert_to_oal_status(ret);
271         }
272
273         return OAL_STATUS_SUCCESS;
274 }
275
276 oal_status_t adapter_set_name(char * name)
277 {
278         int ret;
279         bt_property_t prop;
280
281         CHECK_OAL_INITIALIZED();
282
283         OAL_CHECK_PARAMETER(name, return);
284         API_TRACE("Name: %s", name);
285
286         prop.type = BT_PROPERTY_BDNAME;
287         prop.len = strlen(name);
288         prop.val = name;
289
290         ret = blued_api->set_adapter_property(&prop);
291         if (ret != BT_STATUS_SUCCESS) {
292                 BT_ERR("set_adapter_property: [%s]", status2string(ret));
293                 ret = OAL_STATUS_INTERNAL_ERROR;
294         } else
295                 ret = OAL_STATUS_SUCCESS;
296
297         return ret;
298 }
299
300 oal_status_t adapter_is_discoverable(int *p_discoverable)
301 {
302         OAL_CHECK_PARAMETER(p_discoverable, return);
303
304         *p_discoverable = (scan_mode == BT_SCAN_MODE_CONNECTABLE_DISCOVERABLE);
305
306         API_TRACE("%d", *p_discoverable);
307
308         return OAL_STATUS_SUCCESS;
309 }
310
311 oal_status_t adapter_is_connectable(int *p_connectable)
312 {
313         OAL_CHECK_PARAMETER(p_connectable, return);
314
315         *p_connectable = (scan_mode == BT_SCAN_MODE_CONNECTABLE_DISCOVERABLE)
316                 ||(scan_mode == BT_SCAN_MODE_CONNECTABLE);
317
318         API_TRACE("%d", *p_connectable);
319
320         return OAL_STATUS_SUCCESS;
321 }
322
323 oal_status_t adapter_get_discoverable_timeout(int *p_timeout)
324 {
325         API_TRACE("%d", discoverable_timeout);
326
327         *p_timeout = discoverable_timeout;
328
329         return OAL_STATUS_SUCCESS;
330 }
331
332 oal_status_t adapter_get_service_uuids(void)
333 {
334         int ret;
335
336         CHECK_OAL_INITIALIZED();
337
338         API_TRACE();
339
340         ret = blued_api->get_adapter_property(BT_PROPERTY_UUIDS);
341         if (ret != BT_STATUS_SUCCESS) {
342                 BT_ERR("get_adapter_property failed: [%s]", status2string(ret));
343                 return convert_to_oal_status(ret);
344         }
345
346         return OAL_STATUS_SUCCESS;
347 }
348
349 oal_status_t adapter_get_bonded_devices(void)
350 {
351         int ret;
352
353         CHECK_OAL_INITIALIZED();
354
355         API_TRACE();
356
357         ret = blued_api->get_adapter_property(BT_PROPERTY_ADAPTER_BONDED_DEVICES);
358         if (ret != BT_STATUS_SUCCESS) {
359                 BT_ERR("get_adapter_property failed: [%s]", status2string(ret));
360                 return convert_to_oal_status(ret);
361         }
362
363         return OAL_STATUS_SUCCESS;
364 }
365
366 static oal_status_t set_scan_mode(bt_scan_mode_t mode)
367 {
368         bt_property_t prop;
369         int res;
370
371         BT_DBG("+");
372
373         CHECK_OAL_INITIALIZED();
374
375         prop.type = BT_PROPERTY_ADAPTER_SCAN_MODE;
376         prop.len = sizeof(bt_scan_mode_t);
377         prop.val = &mode;
378         res = blued_api->set_adapter_property(&prop);
379         if (res != BT_STATUS_SUCCESS) {
380                 BT_ERR("set scan mode failed [%s]", status2string(res));
381                 return convert_to_oal_status(res);
382         }
383
384         BT_DBG("-");
385         return OAL_STATUS_SUCCESS;
386 }
387
388 oal_status_t adapter_set_connectable(int connectable)
389 {
390         bt_scan_mode_t mode;
391
392         API_TRACE("%d", connectable);
393
394         CHECK_OAL_INITIALIZED();
395
396         mode = connectable ? BT_SCAN_MODE_CONNECTABLE : BT_SCAN_MODE_NONE;
397
398         return set_scan_mode(mode);
399 }
400
401 oal_status_t adapter_set_discoverable(void)
402 {
403         CHECK_OAL_INITIALIZED();
404         API_TRACE();
405
406         return set_scan_mode(BT_SCAN_MODE_CONNECTABLE_DISCOVERABLE);
407 }
408
409 oal_status_t adapter_set_discoverable_timeout(int timeout)
410 {
411         bt_property_t prop;
412         int res;
413         uint32_t prop_val = timeout;
414
415         CHECK_OAL_INITIALIZED();
416         API_TRACE("%d", timeout);
417
418         prop.type = BT_PROPERTY_ADAPTER_DISCOVERY_TIMEOUT;
419         prop.len = sizeof(prop_val);
420         prop.val = &prop_val;
421         res = blued_api->set_adapter_property(&prop);
422         if (res != BT_STATUS_SUCCESS) {
423                 BT_ERR("set_adapter_property failed [%s]", status2string(res));
424                 return convert_to_oal_status(res);
425         }
426         return OAL_STATUS_SUCCESS;
427 }
428
429 static void cb_adapter_properties(bt_status_t status,
430                                                int num_properties,
431                                                bt_property_t *properties)
432 {
433         int i;
434
435         BT_DBG("status: %d, count: %d", status, num_properties);
436
437         if (status != BT_STATUS_SUCCESS) {
438                 if (num_properties == 1) {
439                         BT_ERR("Adapter Prop failed: status: [%s], count: %d, prop: %d",
440                                 status2string(status), num_properties, properties[num_properties-1].type);
441                 } else {
442                         BT_ERR("Adapter Prop failed: status: [%s], count: %d", status2string(status), num_properties);
443                 }
444                 return;
445         }
446
447         for (i = 0; i < num_properties; i++) {
448                 BT_DBG("prop type %d, len %d", properties[i].type, properties[i].len);
449                 switch (properties[i].type) {
450                 case BT_PROPERTY_VERSION: {
451                         g_strlcpy(local_version, properties[i].val, BT_VERSION_STR_LEN_MAX);
452                         local_version[properties[i].len] = '\0';
453
454                         BT_DBG("Version: %s", local_version);
455                         /* Send event to application */
456                         if (num_properties == 1) {
457                                 char *adapter_ver = g_strdup(local_version);
458
459                                 /* Application has requested this property SET/GET hence send EVENT */
460                                 send_event(OAL_EVENT_ADAPTER_PROPERTY_VERSION, adapter_ver, strlen(adapter_ver));
461                         }
462                         break;
463                 }
464                 case BT_PROPERTY_BDNAME: {
465                         g_strlcpy(local_name, properties[i].val, BT_DEVICE_NAME_LENGTH_MAX);
466                         local_name[properties[i].len] = '\0';
467
468                         BT_DBG("Name: %s", local_name);
469                         /* Send event to application */
470                         if (num_properties == 1) {
471                                 char * adap_name = g_strdup(local_name);
472
473                                 /* Application has requested this property SET/GET hence send EVENT */
474                                 send_event(OAL_EVENT_ADAPTER_PROPERTY_NAME, adap_name, strlen(adap_name));
475                         }
476                         break;
477                 }
478                 case BT_PROPERTY_BDADDR: {
479                         bt_bdaddr_t * addr;
480
481                         addr =  properties[i].val;
482                         memcpy(local_address.addr, addr->address, 6);
483                         if (num_properties == 1) {
484                                 /* Application has requested this property SET/GET hence send EVENT */
485                                 send_event(OAL_EVENT_ADAPTER_PROPERTY_ADDRESS,
486                                                 g_memdup(&local_address, sizeof(local_address)),
487                                                 sizeof(local_address));
488                         }
489                         break;
490                 }
491                 case BT_PROPERTY_UUIDS: {
492                         int num_uuid;
493
494                         num_uuid = properties[i].len/sizeof(bt_uuid_t);
495
496                         BT_DBG("num_uuid: %d", num_uuid);
497
498                         /* Send event to application */
499                         if (num_properties == 1) {
500                                 event_adapter_services_t *uuids_event;
501
502                                 uuids_event = g_malloc(sizeof(event_adapter_services_t) + properties[i].len);
503                                 memcpy(uuids_event->service_list, properties[i].val, properties[i].len);
504                                 uuids_event->num = num_uuid;
505
506                                 /* Application has requested this property SET/GET hence send EVENT */
507                                 send_event(OAL_EVENT_ADAPTER_PROPERTY_SERVICES,
508                                                 uuids_event, (num_uuid * sizeof(bt_uuid_t)));
509                         }
510                         break;
511                 }
512                 case BT_PROPERTY_ADAPTER_SCAN_MODE: {
513                         bt_scan_mode_t cur_mode = *((bt_scan_mode_t *)properties[i].val);
514
515                         BT_INFO("Scan mode (%d)", cur_mode);
516
517                         scan_mode = cur_mode;
518
519                         /* Send event to application */
520                         if (num_properties == 1) {
521                                 oal_event_t event = OAL_EVENT_ADAPTER_MODE_NON_CONNECTABLE;
522
523                                 if (BT_SCAN_MODE_CONNECTABLE == cur_mode)
524                                         event = OAL_EVENT_ADAPTER_MODE_CONNECTABLE;
525                                 else if (BT_SCAN_MODE_CONNECTABLE_DISCOVERABLE == cur_mode)
526                                         event = OAL_EVENT_ADAPTER_MODE_DISCOVERABLE;
527
528                                 /* Application has requested this property SET/GET hence send EVENT */
529                                 send_event(event, NULL, 0);
530                         }
531                         break;
532                 }
533                 case BT_PROPERTY_ADAPTER_DISCOVERY_TIMEOUT: {
534                         int timeout;
535
536                         timeout = *((uint32_t*)properties[i].val);
537
538                         BT_INFO("Discoverability timeout: %d", timeout);
539                         discoverable_timeout = timeout;
540
541                         send_event(OAL_EVENT_ADAPTER_MODE_DISCOVERABLE_TIMEOUT,
542                                         g_memdup(properties[i].val, sizeof(uint32_t)),
543                                         sizeof(uint32_t));
544                         break;
545                 }
546                 case BT_PROPERTY_ADAPTER_BONDED_DEVICES: {
547                         int j;
548                         int num_bonded;
549                         bt_bdaddr_t *bonded_addr_list;
550                         event_device_list_t *event_data;
551
552                         num_bonded = properties[i].len/sizeof(bt_bdaddr_t);
553                         BT_DBG("num_bonded %d", num_bonded);
554
555                         if (num_properties > 1) /* No explicit req for this prop, ignore */
556                                 break;
557
558                         bonded_addr_list = properties[i].val;
559                         event_data = g_malloc(sizeof(event_device_list_t) + num_bonded*sizeof(bt_address_t));
560                         event_data->num = num_bonded;
561
562                         for (j = 0; j < num_bonded; j++)
563                                 memcpy(event_data->devices[j].addr, bonded_addr_list[j].address, 6);
564
565                         send_event(OAL_EVENT_ADAPTER_BONDED_DEVICE_LIST,
566                                         event_data, (num_bonded * sizeof(bt_bdaddr_t)));
567                         break;
568                 }
569                 default:
570                          BT_WARN("Unhandled property: %d", properties[i].type);
571                          break;
572                 }
573         }
574 }
575
576 static void cb_adapter_discovery_state_changed(bt_discovery_state_t state)
577 {
578         oal_event_t event;
579
580         event = (BT_DISCOVERY_STARTED == state)?OAL_EVENT_ADAPTER_INQUIRY_STARTED:OAL_EVENT_ADAPTER_INQUIRY_FINISHED;
581
582         BT_DBG("%d", state);
583         send_event(event, NULL, 0);
584 }
585
586 static void cb_adapter_device_found(int num_properties, bt_property_t *properties)
587 {
588         remote_device_t dev_info;
589         ble_adv_data_t adv_info;
590         oal_event_t event;
591         gpointer event_data;
592         gsize size = 0;
593         BT_DBG("+");
594
595         if (num_properties == 0) {
596                 BT_ERR("Unexpected, properties count is zero!!");
597                 return;
598         }
599
600         memset(&dev_info, 0x00, sizeof(remote_device_t));
601         memset(&adv_info, 0x00, sizeof(ble_adv_data_t));
602
603         print_bt_properties(num_properties, properties);
604         parse_device_properties(num_properties, properties, &dev_info, &adv_info);
605
606         BT_INFO("number of properties= [%d] ", num_properties, size);
607
608         if (dev_info.type != DEV_TYPE_BREDR) {
609                 /* BLE Single or DUAL mode found, so it should have Adv data */
610                 event_ble_dev_found_t * ble_dev_event = g_new0(event_ble_dev_found_t, 1);
611
612                 ble_dev_event->adv_len = adv_info.len;
613
614                 if (adv_info.len > 0 && adv_info.adv_data) {
615                         memcpy(ble_dev_event->adv_data, adv_info.adv_data, adv_info.len);
616                         ble_dev_event->adv_len = adv_info.len;
617                 } else
618                         ble_dev_event->adv_len = 0;
619
620                 ble_dev_event->device_info = dev_info;
621
622                 event_data = ble_dev_event;
623                 size = sizeof(event_ble_dev_found_t);
624                 event = OAL_EVENT_ADAPTER_INQUIRY_RESULT_BLE;
625         } else {
626                 /* BREDR device, so No Adv data */
627                 event_dev_found_t * dev_event = g_new0(event_dev_found_t, 1);
628
629                 memcpy(dev_event, &dev_info, sizeof(remote_device_t));
630                 event_data = dev_event;
631                 size = sizeof(remote_device_t);
632                 event = OAL_EVENT_ADAPTER_INQUIRY_RESULT_BREDR_ONLY;
633         }
634
635         send_event(event, event_data, size);
636
637         BT_DBG("-");
638 }