Add fd param to check fd is valid
[platform/core/connectivity/bluetooth-frwk.git] / bt-api / bt-common.c
1 /*
2  * Copyright (c) 2011 Samsung Electronics Co., Ltd All Rights Reserved
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *              http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  *
16  */
17
18 #include <string.h>
19 #include <fcntl.h>
20 #include <unistd.h>
21 #include <sys/types.h>
22 #include <stdlib.h>
23 #include <sys/socket.h>
24 #include<stdio.h>
25
26 #include <gio/gunixfdlist.h>
27
28
29 #include "bluetooth-api.h"
30 #include "bluetooth-audio-api.h"
31 #include "bluetooth-hid-api.h"
32 #include "bluetooth-media-control.h"
33 #include "bt-internal-types.h"
34 #include "bluetooth-ipsp-api.h"
35
36 #ifndef GATT_DIRECT
37 #include "bluetooth-gatt-server-api.h"
38 #endif
39
40 #ifdef TIZEN_GATT_CLIENT
41 #include "bluetooth-gatt-client-api.h"
42 #endif
43
44 #include "bluetooth-mesh-api.h"
45
46 #include "bt-common.h"
47 #include "bt-request-sender.h"
48 #include "bt-event-handler.h"
49
50 #ifdef TIZEN_FEATURE_BT_DPM
51 #include "bt-dpm.h"
52 #endif
53
54 static bt_user_info_t user_info[BT_MAX_USER_INFO];
55 static GDBusConnection *system_shared_conn = NULL;
56 static GDBusConnection *system_private_conn = NULL;
57
58 static guint bus_id;
59
60 static gboolean bt_enabled = FALSE;
61
62 #define DBUS_TIMEOUT 20 * 1000 /* 20 Seconds */
63
64 static GDBusConnection *g_bus_get_private_conn(void)
65 {
66         GError *error = NULL;
67         char *address;
68         GDBusConnection *private_gconn = NULL;
69
70         address = g_dbus_address_get_for_bus_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
71         if (address == NULL) {
72                 if (error) {
73                         BT_ERR("Failed to get bus address: %s", error->message);
74                         g_clear_error(&error);
75                 }
76                 return NULL;
77         }
78
79         private_gconn = g_dbus_connection_new_for_address_sync(address,
80                                 G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_CLIENT |
81                                 G_DBUS_CONNECTION_FLAGS_MESSAGE_BUS_CONNECTION,
82                                 NULL, /* GDBusAuthObserver */
83                                 NULL,
84                                 &error);
85         g_free(address);
86         if (!private_gconn) {
87                 if (error) {
88                         BT_ERR("Unable to connect to dbus: %s", error->message);
89                         g_clear_error(&error);
90                 }
91                 return NULL;
92         }
93
94         return private_gconn;
95 }
96
97 GDBusConnection *_bt_get_system_private_conn(void)
98 {
99         if (system_private_conn == NULL)
100                 system_private_conn = g_bus_get_private_conn();
101         else if (g_dbus_connection_is_closed(system_private_conn))
102                 system_private_conn = g_bus_get_private_conn();
103
104         return system_private_conn;
105 }
106
107 GDBusConnection *_bt_get_system_shared_conn(void)
108 {
109         if (system_shared_conn == NULL) {
110                 GError *error = NULL;
111                 system_shared_conn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
112                 if (error) {
113                         BT_ERR("GDBus connection Error : %s \n", error->message);
114                         g_clear_error(&error);
115                         return NULL;
116                 }
117         }
118         return system_shared_conn;
119 }
120
121 void _bt_print_device_address_t(const bluetooth_device_address_t *addr)
122 {
123         BT_DBG("%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X\n", addr->addr[0], addr->addr[1], addr->addr[2],
124                                 addr->addr[3], addr->addr[4], addr->addr[5]);
125 }
126
127 void _bt_set_user_data(int type, void *callback, void *user_data)
128 {
129         user_info[type].cb = callback;
130         user_info[type].user_data = user_data;
131 }
132
133 bt_user_info_t *_bt_get_user_data(int type)
134 {
135         return &user_info[type];
136 }
137
138 int _bt_common_send_rfcomm_rx_details(unsigned int len)
139 {
140         uid_t uid = getuid();
141         pid_t pid = getpid();
142         unsigned int size = len;
143         int ret = BLUETOOTH_ERROR_NONE;
144
145         BT_INFO("+");
146         BT_INIT_PARAMS();
147         BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
148
149         g_array_append_vals(in_param1, &uid, sizeof(uid_t));
150         g_array_append_vals(in_param2, &pid, sizeof(pid_t));
151         g_array_append_vals(in_param3, &size, sizeof(unsigned int));
152
153         ret = _bt_send_request(BT_BLUEZ_SERVICE, BT_RFCOMM_SEND_RX_DETAILS,
154                 in_param1, in_param2, in_param3, in_param4, &out_param);
155
156         BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
157         BT_INFO("-");
158         return ret;
159 }
160
161 int _bt_common_send_rfcomm_tx_details(unsigned int len)
162 {
163         uid_t uid = getuid();
164         pid_t pid = getpid();
165         unsigned int size = len;
166         int ret = BLUETOOTH_ERROR_NONE;
167
168         BT_INFO("+");
169         BT_INIT_PARAMS();
170         BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
171
172         g_array_append_vals(in_param1, &uid, sizeof(uid_t));
173         g_array_append_vals(in_param2, &pid, sizeof(pid_t));
174         g_array_append_vals(in_param3, &size, sizeof(unsigned int));
175
176         ret = _bt_send_request(BT_BLUEZ_SERVICE, BT_RFCOMM_SEND_TX_DETAILS,
177                 in_param1, in_param2, in_param3, in_param4, &out_param);
178
179         BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
180         BT_INFO("-");
181         return ret;
182 }
183
184 int _bt_common_send_rfcomm_conn_info(int role, gboolean connected, int socket_fd)
185 {
186         int result = BLUETOOTH_ERROR_NONE;
187
188         if (bluetooth_get_battery_monitor_state() == 0) {
189                 BT_DBG("Battery monitor state is off");
190                 return BLUETOOTH_ERROR_NONE;
191         }
192
193         BT_INIT_PARAMS();
194         BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
195
196         g_array_append_vals(in_param1, &role, sizeof(int));
197         g_array_append_vals(in_param2, &connected, sizeof(gboolean));
198         g_array_append_vals(in_param3, &socket_fd, sizeof(int));
199
200         result = _bt_send_request(BT_BLUEZ_SERVICE, BT_RFCOMM_UPDATE_CONNECTION_INFO,
201                 in_param1, in_param2, in_param3, in_param4, &out_param);
202
203         BT_DBG("result: %x", result);
204
205         BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
206
207         return result;
208 }
209
210 void _bt_common_event_cb(int event, int result, void *param,
211                                         void *callback, void *user_data)
212 {
213         bluetooth_event_param_t bt_event = { 0, };
214         bt_event.event = event;
215         bt_event.result = result;
216         bt_event.param_data = param;
217
218         if (callback)
219                 ((bluetooth_cb_func_ptr)callback)(bt_event.event, &bt_event,
220                                         user_data);
221 }
222
223 void _bt_input_event_cb(int event, int result, void *param,
224                                         void *callback, void *user_data)
225 {
226         hid_event_param_t bt_event = { 0, };
227         bt_event.event = event;
228         bt_event.result = result;
229         bt_event.param_data = param;
230
231         if (callback)
232                 ((hid_cb_func_ptr)callback)(bt_event.event, &bt_event,
233                                         user_data);
234 }
235
236 void _bt_headset_event_cb(int event, int result, void *param,
237                                         void *callback, void *user_data)
238 {
239         bt_audio_event_param_t bt_event = { 0, };
240         bt_event.event = event;
241         bt_event.result = result;
242         bt_event.param_data = param;
243
244         if (callback)
245                 ((bt_audio_func_ptr)callback)(bt_event.event, &bt_event,
246                                         user_data);
247 }
248
249 void _bt_a2dp_source_event_cb(int event, int result, void *param,
250                                         void *callback, void *user_data)
251 {
252         bt_audio_event_param_t bt_event = { 0, };
253         bt_event.event = event;
254         bt_event.result = result;
255         bt_event.param_data = param;
256         if (callback)
257                 ((bt_audio_func_ptr)callback)(bt_event.event, &bt_event,
258                                         user_data);
259 }
260
261 void _bt_hf_event_cb(int event, int result, void *param,
262                                         void *callback, void *user_data)
263 {
264         bt_hf_event_param_t bt_event = { 0, };
265         bt_event.event = event;
266         bt_event.result = result;
267         bt_event.param_data = param;
268
269         if (callback)
270                 ((bt_hf_func_ptr)callback)(bt_event.event, &bt_event,
271                                         user_data);
272 }
273
274
275 void _bt_avrcp_event_cb(int event, int result, void *param,
276                                         void *callback, void *user_data)
277 {
278         media_event_param_t bt_event = { 0, };
279         bt_event.event = event;
280         bt_event.result = result;
281         bt_event.param_data = param;
282
283         if (callback)
284                 ((media_cb_func_ptr)callback)(bt_event.event, &bt_event,
285                                         user_data);
286 }
287
288 #ifdef TIZEN_GATT_CLIENT
289 void _bt_gatt_client_event_cb(int event, int result, void *param,
290                 void *callback, void *user_data)
291 {
292         BT_DBG("__bt_gatt_client_event_cb");
293         gatt_client_event_param_t bt_event = { 0, 0, NULL, NULL };
294         bt_event.event = event;
295         bt_event.result = result;
296         bt_event.param_data = param;
297         if (callback) {
298                 BT_DBG("GATT Client event callback is registered");
299                 ((gatt_client_cb_func_ptr)callback)(bt_event.event, &bt_event,
300                         user_data);
301         } else {
302                 BT_ERR("GATT Client event callback is not registered!!!");
303         }
304 }
305 #endif
306
307 #ifndef GATT_DIRECT
308 void _bt_gatt_server_event_cb(int event, int result, void *param,
309                 void *callback, void *user_data)
310 {
311         gatt_server_event_param_t bt_event = { 0, 0, NULL, NULL };
312         bt_event.event = event;
313         bt_event.result = result;
314         bt_event.param_data = param;
315         if (callback) {
316                 ((gatt_server_cb_func_ptr)callback)(bt_event.event, &bt_event,
317                         user_data);
318         } else {
319                 BT_ERR("GATT Server event callback is not registered!!!");
320         }
321 }
322 #endif
323
324 void _bt_mesh_event_cb(int event, int result, void *param,
325                 void *callback, void *user_data)
326 {
327         BT_DBG("_bt_mesh_event_cb");
328         mesh_event_param_t bt_event = { 0, 0, NULL, NULL };
329         bt_event.event = event;
330         bt_event.result = result;
331         bt_event.param_data = param;
332         if (callback) {
333                 BT_DBG("Mesh event callback is registered");
334                 ((mesh_cb_func_ptr)callback)(bt_event.event, &bt_event,
335                         user_data);
336         } else {
337                 BT_ERR("Mesh event callback is not registered!!!");
338         }
339 }
340
341 void _bt_divide_device_class(bluetooth_device_class_t *device_class,
342                                 unsigned int cod)
343 {
344         ret_if(device_class == NULL);
345
346         device_class->major_class = (unsigned short)(cod & 0x00001F00) >> 8;
347         device_class->minor_class = (unsigned short)((cod & 0x000000FC));
348         device_class->service_class = (unsigned long)((cod & 0x00FF0000));
349
350         if (cod & 0x002000) {
351                 device_class->service_class |=
352                 BLUETOOTH_DEVICE_SERVICE_CLASS_LIMITED_DISCOVERABLE_MODE;
353         }
354 }
355
356 void _bt_convert_addr_string_to_type(unsigned char *addr, const char *address)
357 {
358         int i;
359         char *ptr = NULL;
360
361         ret_if(address == NULL);
362         ret_if(addr == NULL);
363
364         for (i = 0; i < BT_ADDRESS_LENGTH_MAX; i++) {
365                 addr[i] = strtol(address, &ptr, 16);
366                 if (ptr[0] != '\0') {
367                         if (ptr[0] != ':')
368                                 return;
369
370                         address = ptr + 1;
371                 }
372         }
373 }
374
375 void  _bt_convert_uuid_type_to_string(char *str, const unsigned char *uuid)
376 {
377         if (!str) {
378                 BT_ERR("str == NULL");
379                 return;
380         }
381
382         if (!uuid) {
383                 BT_ERR("uuid == NULL");
384                 return;
385         }
386
387         snprintf(str, 37,
388                         "%2.2X%2.2X%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X%2.2X%2.2X%2.2X%2.2X",
389                         uuid[0], uuid[1], uuid[2], uuid[3], uuid[4], uuid[5], uuid[6], uuid[7],
390                         uuid[8], uuid[9], uuid[10], uuid[11], uuid[12], uuid[13], uuid[14], uuid[15]);
391 }
392
393 void _bt_convert_addr_string_to_secure_string(char *addr, const char *address)
394 {
395         int len;
396
397         ret_if(address == NULL);
398         ret_if(addr == NULL);
399
400         len = strlen(address);
401         ret_if(len != BT_ADDRESS_STRING_SIZE - 1);
402
403         strncpy(addr, address, len);
404         addr[len] = '\0';
405
406         addr[len-7] = 'X';
407         addr[len-8] = 'X';
408         addr[len-10] = 'X';
409         addr[len-11] = 'X';
410         addr[len-13] = 'X';
411         addr[len-14] = 'X';
412         addr[len-16] = 'X';
413         addr[len-17] = 'X';
414
415         return;
416 }
417
418 void _bt_convert_addr_type_to_string(char *address, unsigned char *addr)
419 {
420         ret_if(address == NULL);
421         ret_if(addr == NULL);
422
423         g_snprintf(address, BT_ADDRESS_STRING_SIZE,
424                         "%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X",
425                         addr[0], addr[1], addr[2],
426                         addr[3], addr[4], addr[5]);
427 }
428
429 void _bt_convert_addr_type_to_secure_string(char *address, unsigned char *addr)
430 {
431         ret_if(address == NULL);
432         ret_if(addr == NULL);
433
434         g_snprintf(address, BT_ADDRESS_STRING_SIZE,
435                         "XX:XX:XX:XX:%2.2X:%2.2X", addr[4], addr[5]);
436 }
437
438 const char *_bt_convert_error_to_string(int error)
439 {
440         switch (error) {
441         case BLUETOOTH_ERROR_CANCEL:
442                 return "CANCELLED";
443         case BLUETOOTH_ERROR_INVALID_PARAM:
444                 return "INVALID_PARAMETER";
445         case BLUETOOTH_ERROR_INVALID_DATA:
446                 return "INVALID DATA";
447         case BLUETOOTH_ERROR_MEMORY_ALLOCATION:
448         case BLUETOOTH_ERROR_OUT_OF_MEMORY:
449                 return "OUT_OF_MEMORY";
450         case BLUETOOTH_ERROR_TIMEOUT:
451                 return "TIMEOUT";
452         case BLUETOOTH_ERROR_NO_RESOURCES:
453                 return "NO_RESOURCES";
454         case BLUETOOTH_ERROR_INTERNAL:
455                 return "INTERNAL";
456         case BLUETOOTH_ERROR_NOT_SUPPORT:
457                 return "NOT_SUPPORT";
458         case BLUETOOTH_ERROR_DEVICE_NOT_ENABLED:
459                 return "NOT_ENABLED";
460         case BLUETOOTH_ERROR_DEVICE_ALREADY_ENABLED:
461                 return "ALREADY_ENABLED";
462         case BLUETOOTH_ERROR_DEVICE_BUSY:
463                 return "DEVICE_BUSY";
464         case BLUETOOTH_ERROR_ACCESS_DENIED:
465                 return "ACCESS_DENIED";
466         case BLUETOOTH_ERROR_MAX_CLIENT:
467                 return "MAX_CLIENT";
468         case BLUETOOTH_ERROR_NOT_FOUND:
469                 return "NOT_FOUND";
470         case BLUETOOTH_ERROR_SERVICE_SEARCH_ERROR:
471                 return "SERVICE_SEARCH_ERROR";
472         case BLUETOOTH_ERROR_PARING_FAILED:
473                 return "PARING_FAILED";
474         case BLUETOOTH_ERROR_NOT_PAIRED:
475                 return "NOT_PAIRED";
476         case BLUETOOTH_ERROR_SERVICE_NOT_FOUND:
477                 return "SERVICE_NOT_FOUND";
478         case BLUETOOTH_ERROR_NOT_CONNECTED:
479                 return "NOT_CONNECTED";
480         case BLUETOOTH_ERROR_ALREADY_CONNECT:
481                 return "ALREADY_CONNECT";
482         case BLUETOOTH_ERROR_CONNECTION_BUSY:
483                 return "CONNECTION_BUSY";
484         case BLUETOOTH_ERROR_CONNECTION_ERROR:
485                 return "CONNECTION_ERROR";
486         case BLUETOOTH_ERROR_MAX_CONNECTION:
487                 return "MAX_CONNECTION";
488         case BLUETOOTH_ERROR_NOT_IN_OPERATION:
489                 return "NOT_IN_OPERATION";
490         case BLUETOOTH_ERROR_CANCEL_BY_USER:
491                 return "CANCEL_BY_USER";
492         case BLUETOOTH_ERROR_REGISTRATION_FAILED:
493                 return "REGISTRATION_FAILED";
494         case BLUETOOTH_ERROR_IN_PROGRESS:
495                 return "IN_PROGRESS";
496         case BLUETOOTH_ERROR_AUTHENTICATION_FAILED:
497                 return "AUTHENTICATION_FAILED";
498         case BLUETOOTH_ERROR_HOST_DOWN:
499                 return "HOST_DOWN";
500         case BLUETOOTH_ERROR_END_OF_DEVICE_LIST:
501                 return "END_OF_DEVICE_LIST";
502         case BLUETOOTH_ERROR_AGENT_ALREADY_EXIST:
503                 return "AGENT_ALREADY_EXIST";
504         case BLUETOOTH_ERROR_AGENT_DOES_NOT_EXIST:
505                 return "AGENT_DOES_NOT_EXIST";
506         case BLUETOOTH_ERROR_ALREADY_INITIALIZED:
507                 return "ALREADY_INITIALIZED";
508         case BLUETOOTH_ERROR_PERMISSION_DEINED:
509                 return "PERMISSION_DEINED";
510         case BLUETOOTH_ERROR_ALREADY_DEACTIVATED:
511                 return "ALREADY_DEACTIVATED";
512         case BLUETOOTH_ERROR_NOT_INITIALIZED:
513                 return "NOT_INITIALIZED";
514         case BLUETOOTH_ERROR_AUTHENTICATION_REJECTED:
515                 return "AUTHENTICATION REJECTED";
516         case BLUETOOTH_ERROR_AUTHORIZATION_REJECTED:
517                 return "AUTHORIZATION REJECTED";
518         default:
519                 return "UNKNOWN";
520         }
521 }
522
523 const char *_bt_convert_service_function_to_string(int function)
524 {
525         int i;
526
527         typedef struct {
528                 int function;
529                 const char *function_name;
530         } bt_function_name_t;
531
532         const bt_function_name_t bt_functions[] = {
533                 {BT_CHECK_ADAPTER, "BT_CHECK_ADAPTER"},
534                 {BT_ENABLE_ADAPTER, "BT_ENABLE_ADAPTER"},
535                 {BT_DISABLE_ADAPTER, "BT_DISABLE_ADAPTER"},
536                 {BT_RECOVER_ADAPTER, "BT_RECOVER_ADAPTER"},
537                 {BT_SET_DISCOVERABLE_TIME, "BT_SET_DISCOVERABLE_TIME"},
538                 {BT_GET_DISCOVERABLE_TIME, "BT_GET_DISCOVERABLE_TIME"},
539                 {BT_IGNORE_AUTO_PAIRING, "BT_IGNORE_AUTO_PAIRING"},
540                 {BT_GET_LOCAL_ADDRESS, "BT_GET_LOCAL_ADDRESS"},
541                 {BT_GET_LOCAL_VERSION, "BT_GET_LOCAL_VERSION"},
542                 {BT_GET_LOCAL_NAME, "BT_GET_LOCAL_NAME"},
543                 {BT_SET_LOCAL_NAME, "BT_SET_LOCAL_NAME"},
544                 {BT_IS_SERVICE_USED, "BT_IS_SERVICE_USED"},
545                 {BT_GET_DISCOVERABLE_MODE, "BT_GET_DISCOVERABLE_MODE"},
546                 {BT_SET_DISCOVERABLE_MODE, "BT_SET_DISCOVERABLE_MODE"},
547                 {BT_START_DISCOVERY, "BT_START_DISCOVERY"},
548                 {BT_START_CUSTOM_DISCOVERY, "BT_START_CUSTOM_DISCOVERY"},
549                 {BT_CANCEL_DISCOVERY, "BT_CANCEL_DISCOVERY"},
550                 {BT_START_LE_DISCOVERY, "BT_START_LE_DISCOVERY"},
551                 {BT_STOP_LE_DISCOVERY, "BT_STOP_LE_DISCOVERY"},
552                 {BT_IS_DISCOVERYING, "BT_IS_DISCOVERYING"},
553                 {BT_IS_LE_DISCOVERYING, "BT_IS_LE_DISCOVERYING"},
554                 {BT_ENABLE_RSSI, "BT_ENABLE_RSSI"},
555                 {BT_GET_RSSI, "BT_GET_RSSI"},
556                 {BT_IS_CONNECTABLE, "BT_IS_CONNECTABLE"},
557                 {BT_SET_CONNECTABLE, "BT_SET_CONNECTABLE"},
558                 {BT_GET_BONDED_DEVICES, "BT_GET_BONDED_DEVICES"},
559                 {BT_RESET_ADAPTER, "BT_RESET_ADAPTER"},
560                 {BT_SET_ADVERTISING, "BT_SET_ADVERTISING"},
561                 {BT_SET_CUSTOM_ADVERTISING, "BT_SET_CUSTOM_ADVERTISING"},
562                 {BT_SET_ADVERTISING_PARAMETERS, "BT_SET_ADVERTISING_PARAMETERS"},
563                 {BT_GET_ADVERTISING_DATA, "BT_GET_ADVERTISING_DATA"},
564                 {BT_SET_ADVERTISING_DATA, "BT_SET_ADVERTISING_DATA"},
565                 {BT_SET_SCAN_PARAMETERS, "BT_SET_SCAN_PARAMETERS"},
566                 {BT_SET_SCAN_TYPE, "BT_SET_SCAN_TYPE"},
567                 {BT_GET_SCAN_RESPONSE_DATA, "BT_GET_SCAN_RESPONSE_DATA"},
568                 {BT_SET_SCAN_RESPONSE_DATA, "BT_SET_SCAN_RESPONSE_DATA"},
569                 {BT_IS_ADVERTISING, "BT_IS_ADVERTISING"},
570                 {BT_SET_MANUFACTURER_DATA, "BT_SET_MANUFACTURER_DATA"},
571                 {BT_LE_CONN_UPDATE, "BT_LE_CONN_UPDATE"},
572                 {BT_LE_READ_MAXIMUM_DATA_LENGTH, "BT_LE_READ_MAXIMUM_DATA_LENGTH"},
573                 {BT_LE_WRITE_HOST_SUGGESTED_DATA_LENGTH, "BT_LE_WRITE_HOST_SUGGESTED_DATA_LENGTH"},
574                 {BT_LE_READ_HOST_SUGGESTED_DATA_LENGTH, "BT_LE_READ_HOST_SUGGESTED_DATA_LENGTH"},
575                 {BT_LE_SET_DATA_LENGTH, "BT_LE_SET_DATA_LENGTH"},
576                 {BT_ADD_WHITE_LIST, "BT_ADD_WHITE_LIST"},
577                 {BT_REMOVE_WHITE_LIST, "BT_REMOVE_WHITE_LIST"},
578                 {BT_CLEAR_WHITE_LIST, "BT_CLEAR_WHITE_LIST"},
579                 {BT_REGISTER_SCAN_FILTER, "BT_REGISTER_SCAN_FILTER"},
580                 {BT_IS_SCAN_FILTER_SUPPORTED, "BT_IS_SCAN_FILTER_SUPPORTED"},
581                 {BT_GET_PROFILE_CONNECTED_DEVICES, "BT_GET_PROFILE_CONNECTED_DEVICES"},
582                 {BT_ENABLE_FORCE_HCI_DUMP, "BT_ENABLE_FORCE_HCI_DUMP"},
583                 {BT_SET_PASSKEY_NOTIFICATION, "BT_SET_PASSKEY_NOTIFICATION"},
584                 {BT_BOND_DEVICE, "BT_BOND_DEVICE"},
585                 {BT_BOND_DEVICE_BY_TYPE, "BT_BOND_DEVICE_BY_TYPE"},
586                 {BT_CANCEL_BONDING, "BT_CANCEL_BONDING"},
587                 {BT_PASSKEY_REPLY, "BT_PASSKEY_REPLY"},
588                 {BT_PASSKEY_CONFIRMATION_REPLY, "BT_PASSKEY_CONFIRMATION_REPLY"},
589                 {BT_UNBOND_DEVICE, "BT_UNBOND_DEVICE"},
590                 {BT_SEARCH_SERVICE, "BT_SEARCH_SERVICE"},
591                 {BT_CANCEL_SEARCH_SERVICE, "BT_CANCEL_SEARCH_SERVICE"},
592                 {BT_GET_BONDED_DEVICE, "BT_GET_BONDED_DEVICE"},
593                 {BT_GET_IS_ALIAS_SET, "BT_GET_IS_ALIAS_SET"},
594                 {BT_SET_ALIAS, "BT_SET_ALIAS"},
595                 {BT_SET_AUTHORIZATION, "BT_SET_AUTHORIZATION"},
596                 {BT_UNSET_AUTHORIZATION, "BT_UNSET_AUTHORIZATION"},
597                 {BT_IS_DEVICE_CONNECTED, "BT_IS_DEVICE_CONNECTED"},
598                 {BT_GET_CONNECTED_LINK_TYPE, "BT_GET_CONNECTED_LINK_TYPE"},
599                 {BT_SET_PIN_CODE, "BT_SET_PIN_CODE"},
600                 {BT_UNSET_PIN_CODE, "BT_UNSET_PIN_CODE"},
601                 {BT_UPDATE_LE_CONNECTION_MODE, "BT_UPDATE_LE_CONNECTION_MODE"},
602                 {BT_SET_PROFILE_TRUSTED, "BT_SET_PROFILE_TRUSTED"},
603                 {BT_GET_PROFILE_TRUSTED, "BT_GET_PROFILE_TRUSTED"},
604                 {BT_DISCONNECT_DEVICE, "BT_DISCONNECT_DEVICE"},
605                 {BT_HID_CONNECT, "BT_HID_CONNECT"},
606                 {BT_HID_DISCONNECT, "BT_HID_DISCONNECT"},
607                 {BT_HID_DEVICE_ACTIVATE, "BT_HID_DEVICE_ACTIVATE"},
608                 {BT_HID_DEVICE_DEACTIVATE, "BT_HID_DEVICE_DEACTIVATE"},
609                 {BT_HID_DEVICE_CONNECT, "BT_HID_DEVICE_CONNECT"},
610                 {BT_HID_DEVICE_DISCONNECT, "BT_HID_DEVICE_DISCONNECT"},
611                 {BT_HID_DEVICE_SEND_MOUSE_EVENT, "BT_HID_DEVICE_SEND_MOUSE_EVENT"},
612                 {BT_HID_DEVICE_SEND_KEY_EVENT, "BT_HID_DEVICE_SEND_KEY_EVENT"},
613                 {BT_HID_DEVICE_SEND_CUSTOM_EVENT, "BT_HID_DEVICE_SEND_CUSTOM_EVENT"},
614                 {BT_HID_DEVICE_SEND_REPLY_TO_REPORT, "BT_HID_DEVICE_SEND_REPLY_TO_REPORT"},
615                 {BT_HID_ENABLE_BARCODE_FEATURE, "BT_HID_ENABLE_BARCODE_FEATURE"},
616                 {BT_NETWORK_ACTIVATE, "BT_NETWORK_ACTIVATE"},
617                 {BT_NETWORK_DEACTIVATE, "BT_NETWORK_DEACTIVATE"},
618                 {BT_NETWORK_CONNECT, "BT_NETWORK_CONNECT"},
619                 {BT_NETWORK_DISCONNECT, "BT_NETWORK_DISCONNECT"},
620                 {BT_NETWORK_SERVER_DISCONNECT, "BT_NETWORK_SERVER_DISCONNECT"},
621                 {BT_AUDIO_CONNECT, "BT_AUDIO_CONNECT"},
622                 {BT_AUDIO_DISCONNECT, "BT_AUDIO_DISCONNECT"},
623                 {BT_AG_CONNECT, "BT_AG_CONNECT"},
624                 {BT_AG_DISCONNECT, "BT_AG_DISCONNECT"},
625                 {BT_AV_CONNECT, "BT_AV_CONNECT"},
626                 {BT_AV_DISCONNECT, "BT_AV_DISCONNECT"},
627                 {BT_AV_SOURCE_CONNECT, "BT_AV_SOURCE_CONNECT"},
628                 {BT_AV_SOURCE_DISCONNECT, "BT_AV_SOURCE_DISCONNECT"},
629                 {BT_HF_CONNECT, "BT_HF_CONNECT"},
630                 {BT_HF_DISCONNECT, "BT_HF_DISCONNECT"},
631                 {BT_GET_SPEAKER_GAIN, "BT_GET_SPEAKER_GAIN"},
632                 {BT_SET_SPEAKER_GAIN, "BT_SET_SPEAKER_GAIN"},
633                 {BT_SET_CONTENT_PROTECT, "BT_SET_CONTENT_PROTECT"},
634                 {BT_AUDIO_SELECT_ROLE, "BT_AUDIO_SELECT_ROLE"},
635                 {BT_AUDIO_SET_ABSOLUTE_VOLUME, "BT_AUDIO_SET_ABSOLUTE_VOLUME"},
636                 {BT_AUDIO_GET_ABSOLUTE_VOLUME, "BT_AUDIO_GET_ABSOLUTE_VOLUME"},
637                 {BT_AUDIO_IS_AVC_ACTIVATED, "BT_AUDIO_IS_AVC_ACTIVATED"},
638                 {BT_OOB_READ_LOCAL_DATA, "BT_OOB_READ_LOCAL_DATA"},
639                 {BT_OOB_ADD_REMOTE_DATA, "BT_OOB_ADD_REMOTE_DATA"},
640                 {BT_OOB_REMOVE_REMOTE_DATA, "BT_OOB_REMOVE_REMOTE_DATA"},
641                 {BT_LE_OOB_READ_LOCAL_DATA, "BT_LE_OOB_READ_LOCAL_DATA"},
642                 {BT_LE_OOB_ADD_REMOTE_DATA, "BT_LE_OOB_ADD_REMOTE_DATA"},
643                 {BT_AVRCP_SET_TRACK_INFO, "BT_AVRCP_SET_TRACK_INFO"},
644                 {BT_AVRCP_SET_PROPERTY, "BT_AVRCP_SET_PROPERTY"},
645                 {BT_AVRCP_SET_PROPERTIES, "BT_AVRCP_SET_PROPERTIES"},
646                 {BT_AVRCP_CONTROL_CONNECT, "BT_AVRCP_CONTROL_CONNECT"},
647                 {BT_AVRCP_CONTROL_DISCONNECT, "BT_AVRCP_CONTROL_DISCONNECT"},
648                 {BT_AVRCP_TARGET_CONNECT, "BT_AVRCP_TARGET_CONNECT"},
649                 {BT_AVRCP_TARGET_DISCONNECT, "BT_AVRCP_TARGET_DISCONNECT"},
650                 {BT_AVRCP_HANDLE_CONTROL, "BT_AVRCP_HANDLE_CONTROL"},
651                 {BT_AVRCP_HANDLE_CONTROL_TO_DEST, "BT_AVRCP_HANDLE_CONTROL_TO_DEST"},
652                 {BT_AVRCP_CONTROL_SET_PROPERTY, "BT_AVRCP_CONTROL_SET_PROPERTY"},
653                 {BT_AVRCP_CONTROL_GET_PROPERTY, "BT_AVRCP_CONTROL_GET_PROPERTY"},
654                 {BT_AVRCP_GET_TRACK_INFO, "BT_AVRCP_GET_TRACK_INFO"},
655                 {BT_OPP_PUSH_FILES, "BT_OPP_PUSH_FILES"},
656                 {BT_OPP_CANCEL_PUSH, "BT_OBT_OPP_IS_PUSHING_FILESPP_CANCEL_PUSH"},
657                 {BT_OPP_IS_PUSHING_FILES, "BT_OPP_IS_PUSHING_FILES"},
658                 {BT_OPP_GET_TRANSFER_PROGRESS, "BT_OPP_GET_TRANSFER_PROGRESS"},
659                 {BT_MAP_CREATE_SESSION, "BT_MAP_CREATE_SESSION"},
660                 {BT_MAP_DESTROY_SESSION, "BT_MAP_DESTROY_SESSION"},
661                 {BT_MAP_SET_FOLDER, "BT_MAP_SET_FOLDER"},
662                 {BT_MAP_LIST_FOLDERS, "BT_MAP_LIST_FOLDERS"},
663                 {BT_MAP_LIST_FILTER_FIELDS, "BT_MAP_LIST_FILTER_FIELDS"},
664                 {BT_MAP_LIST_MESSAGES, "BT_MAP_LIST_MESSAGES"},
665                 {BT_MAP_UPDATE_INBOX, "BT_MAP_UPDATE_INBOX"},
666                 {BT_MAP_PUSH_MESSAGE, "BT_MAP_PUSH_MESSAGE"},
667                 {BT_MAP_GET_MESSAGE, "BT_MAP_GET_MESSAGE"},
668                 {BT_OBEX_SERVER_ALLOCATE, "BT_OBEX_SERVER_ALLOCATE"},
669                 {BT_OBEX_SERVER_DEALLOCATE, "BT_OBEX_SERVER_DEALLOCATE"},
670                 {BT_OBEX_SERVER_IS_ACTIVATED, "BT_OBEX_SERVER_IS_ACTIVATED"},
671                 {BT_OBEX_SERVER_ACCEPT_CONNECTION, "BT_OBEX_SERVER_ACCEPT_CONNECTION"},
672                 {BT_OBEX_SERVER_REJECT_CONNECTION, "BT_OBEX_SERVER_REJECT_CONNECTION"},
673                 {BT_OBEX_SERVER_ACCEPT_FILE, "BT_OBEX_SERVER_ACCEPT_FILE"},
674                 {BT_OBEX_SERVER_REJECT_FILE, "BT_OBEX_SERVER_REJECT_FILE"},
675                 {BT_OBEX_SERVER_SET_PATH, "BT_OBEX_SERVER_SET_PATH"},
676                 {BT_OBEX_SERVER_SET_ROOT, "BT_OBEX_SERVER_SET_ROOT"},
677                 {BT_OBEX_SERVER_CANCEL_TRANSFER, "BT_OBEX_SERVER_CANCEL_TRANSFER"},
678                 {BT_OBEX_SERVER_CANCEL_ALL_TRANSFERS, "BT_OBEX_SERVER_CANCEL_ALL_TRANSFERS"},
679                 {BT_OBEX_SERVER_IS_RECEIVING, "BT_OBEX_SERVER_IS_RECEIVING"},
680                 {BT_RFCOMM_CLIENT_CONNECT, "BT_RFCOMM_CLIENT_CONNECT"},
681                 {BT_RFCOMM_CLIENT_CANCEL_CONNECT, "BT_RFCOMM_CLIENT_CANCEL_CONNECT"},
682                 {BT_RFCOMM_CLIENT_IS_CONNECTED, "BT_RFCOMM_CLIENT_IS_CONNECTED"},
683                 {BT_RFCOMM_SOCKET_DISCONNECT, "BT_RFCOMM_SOCKET_DISCONNECT"},
684                 {BT_RFCOMM_SOCKET_WRITE, "BT_RFCOMM_SOCKET_WRITE"},
685                 {BT_RFCOMM_CREATE_SOCKET, "BT_RFCOMM_CREATE_SOCKET"},
686                 {BT_RFCOMM_REMOVE_SOCKET, "BT_RFCOMM_REMOVE_SOCKET"},
687                 {BT_RFCOMM_LISTEN_AND_ACCEPT, "BT_RFCOMM_LISTEN_AND_ACCEPT"},
688                 {BT_RFCOMM_LISTEN, "BT_RFCOMM_LISTEN"},
689                 {BT_RFCOMM_IS_UUID_AVAILABLE, "BT_RFCOMM_IS_UUID_AVAILABLE"},
690                 {BT_RFCOMM_ACCEPT_CONNECTION, "BT_RFCOMM_ACCEPT_CONNECTION"},
691                 {BT_RFCOMM_REJECT_CONNECTION, "BT_RFCOMM_REJECT_CONNECTION"},
692                 {BT_RFCOMM_CREATE_SOCKET_EX, "BT_RFCOMM_CREATE_SOCKET_EX"},
693                 {BT_RFCOMM_REMOVE_SOCKET_EX, "BT_RFCOMM_REMOVE_SOCKET_EX"},
694                 {BT_PBAP_CONNECT, "BT_PBAP_CONNECT"},
695                 {BT_PBAP_DISCONNECT, "BT_PBAP_DISCONNECT"},
696                 {BT_PBAP_GET_PHONEBOOK_SIZE, "BT_PBAP_GET_PHONEBOOK_SIZE"},
697                 {BT_PBAP_GET_PHONEBOOK, "BT_PBAP_GET_PHONEBOOK"},
698                 {BT_PBAP_GET_LIST, "BT_PBAP_GET_LIST"},
699                 {BT_PBAP_PULL_VCARD, "BT_PBAP_PULL_VCARD"},
700                 {BT_PBAP_PHONEBOOK_SEARCH, "BT_PBAP_PHONEBOOK_SEARCH"},
701                 {BT_ENABLE_ADAPTER_LE, "BT_ENABLE_ADAPTER_LE"},
702                 {BT_DISABLE_ADAPTER_LE, "BT_DISABLE_ADAPTER_LE"},
703                 {BT_CONNECT_LE, "BT_CONNECT_LE"},
704                 {BT_DISCONNECT_LE, "BT_DISCONNECT_LE"},
705                 {BT_SET_LE_PRIVACY, "BT_SET_LE_PRIVACY"},
706                 {BT_REQ_ATT_MTU, "BT_REQ_ATT_MTU"},
707                 {BT_GET_ATT_MTU, "BT_GET_ATT_MTU"},
708                 {BT_GATT_ACQUIRE_WRITE, "BT_GATT_ACQUIRE_WRITE"},
709                 {BT_GET_DEVICE_IDA, "BT_GET_DEVICE_IDA"},
710                 {BT_SET_LE_STATIC_RANDOM_ADDRESS, "BT_SET_LE_STATIC_RANDOM_ADDRESS"},
711                 {BT_HDP_CONNECT, "BT_HDP_CONNECT"},
712                 {BT_HDP_DISCONNECT, "BT_HDP_DISCONNECT"},
713                 {BT_HDP_SEND_DATA, "BT_HDP_SEND_DATA"},
714                 {BT_HDP_REGISTER_SINK_APP, "BT_HDP_REGISTER_SINK_APP"},
715                 {BT_HDP_UNREGISTER_SINK_APP, "BT_HDP_UNREGISTER_SINK_APP"},
716                 {BT_GATT_GET_PRIMARY_SERVICES, "BT_GATT_GET_PRIMARY_SERVICES"},
717                 {BT_GATT_DISCOVER_CHARACTERISTICS, "BT_GATT_DISCOVER_CHARACTERISTICS"},
718                 {BT_GATT_SET_PROPERTY_REQUEST, "BT_GATT_SET_PROPERTY_REQUEST"},
719                 {BT_GATT_READ_CHARACTERISTIC, "BT_GATT_READ_CHARACTERISTIC"},
720                 {BT_GATT_DISCOVER_CHARACTERISTICS_DESCRIPTOR, "BT_GATT_DISCOVER_CHARACTERISTICS_DESCRIPTOR"},
721                 {BT_GATT_REGISTER_APPLICATION, "BT_GATT_REGISTER_APPLICATION"},
722                 {BT_GATT_REGISTER_SERVICE, "BT_GATT_REGISTER_SERVICE"},
723                 {BT_GATT_SEND_RESPONSE, "BT_GATT_SEND_RESPONSE"},
724                 {BT_LE_IPSP_INIT, "BT_LE_IPSP_INIT"},
725                 {BT_LE_IPSP_DEINIT, "BT_LE_IPSP_DEINIT"},
726                 {BT_LE_IPSP_CONNECT, "BT_LE_IPSP_CONNECT"},
727                 {BT_LE_IPSP_DISCONNECT, "BT_LE_IPSP_DISCONNECT"},
728                 {BT_DPM_SET_ALLOW_BT_MODE, "BT_DPM_SET_ALLOW_BT_MODE"},
729                 {BT_DPM_GET_ALLOW_BT_MODE, "BT_DPM_GET_ALLOW_BT_MODE"},
730                 {BT_DPM_SET_DEVICE_RESTRITION, "BT_DPM_SET_DEVICE_RESTRITION"},
731                 {BT_DPM_GET_DEVICE_RESTRITION, "BT_DPM_GET_DEVICE_RESTRITION"},
732                 {BT_DPM_SET_UUID_RESTRITION, "BT_DPM_SET_UUID_RESTRITION"},
733                 {BT_DPM_GET_UUID_RESTRITION, "BT_DPM_GET_UUID_RESTRITION"},
734                 {BT_DPM_ADD_DEVICES_BLACKLIST, "BT_DPM_ADD_DEVICES_BLACKLIST"},
735                 {BT_DPM_ADD_DEVICES_WHITELIST, "BT_DPM_ADD_DEVICES_WHITELIST"},
736                 {BT_DPM_ADD_UUIDS_BLACKLIST, "BT_DPM_ADD_UUIDS_BLACKLIST"},
737                 {BT_DPM_ADD_UUIDS_WHITELIST, "BT_DPM_ADD_UUIDS_WHITELIST"},
738                 {BT_DPM_CLEAR_DEVICES_BLACKLIST, "BT_DPM_CLEAR_DEVICES_BLACKLIST"},
739                 {BT_DPM_CLEAR_DEVICES_WHITELIST, "BT_DPM_CLEAR_DEVICES_WHITELIST"},
740                 {BT_DPM_CLEAR_UUIDS_BLACKLIST, "BT_DPM_CLEAR_UUIDS_BLACKLIST"},
741                 {BT_DPM_CLEAR_UUIDS_WHITELIST, "BT_DPM_CLEAR_UUIDS_WHITELIST"},
742                 {BT_DPM_REMOVE_DEVICE_BLACKLIST, "BT_DPM_REMOVE_DEVICE_BLACKLIST"},
743                 {BT_DPM_REMOVE_DEVICE_WHITELIST, "BT_DPM_REMOVE_DEVICE_WHITELIST"},
744                 {BT_DPM_REMOVE_UUID_BLACKLIST, "BT_DPM_REMOVE_UUID_BLACKLIST"},
745                 {BT_DPM_REMOVE_UUID_WHITELIST, "BT_DPM_REMOVE_UUID_WHITELIST"},
746                 {BT_DPM_GET_DEVICES_BLACKLIST, "BT_DPM_GET_DEVICES_BLACKLIST"},
747                 {BT_DPM_GET_DEVICES_WHITELIST, "BT_DPM_GET_DEVICES_WHITELIST"},
748                 {BT_DPM_GET_UUIDS_BLACKLIST, "BT_DPM_GET_UUIDS_BLACKLIST"},
749                 {BT_DPM_GET_UUIDS_WHITELIST, "BT_DPM_GET_UUIDS_WHITELIST"},
750                 {BT_DPM_SET_ALLOW_OUTGOING_CALL, "BT_DPM_SET_ALLOW_OUTGOING_CALL"},
751                 {BT_DPM_GET_ALLOW_OUTGOING_CALL, "BT_DPM_GET_ALLOW_OUTGOING_CALL"},
752                 {BT_DPM_SET_PAIRING_STATE, "BT_DPM_SET_PAIRING_STATE"},
753                 {BT_DPM_GET_PAIRING_STATE, "BT_DPM_GET_PAIRING_STATE"},
754                 {BT_DPM_SET_PROFILE_STATE, "BT_DPM_SET_PROFILE_STATE"},
755                 {BT_DPM_GET_PROFILE_STATE, "BT_DPM_GET_PROFILE_STATE"},
756                 {BT_DPM_SET_DESKROP_CONNECTIVITY_STATE, "BT_DPM_SET_DESKROP_CONNECTIVITY_STATE"},
757                 {BT_DPM_GET_DESKROP_CONNECTIVITY_STATE, "BT_DPM_GET_DESKROP_CONNECTIVITY_STATE"},
758                 {BT_DPM_SET_DISCOVERABLE_STATE, "BT_DPM_SET_DISCOVERABLE_STATE"},
759                 {BT_DPM_GET_DISCOVERABLE_STATE, "BT_DPM_GET_DISCOVERABLE_STATE"},
760                 {BT_DPM_SET_LIMITED_DISCOVERABLE_STATE, "BT_DPM_SET_LIMITED_DISCOVERABLE_STATE"},
761                 {BT_DPM_GET_LIMITED_DISCOVERABLE_STATE, "BT_DPM_GET_LIMITED_DISCOVERABLE_STATE"},
762                 {BT_DPM_SET_DATA_TRANSFER_STATE, "BT_DPM_SET_DATA_TRANSFER_STATE"},
763                 {BT_DPM_GET_DATA_TRANSFER_STATE, "BT_DPM_GET_DATA_TRANSFER_STATE"},
764                 {BT_PXP_MONITOR_SET_PROPERTY, "BT_PXP_MONITOR_SET_PROPERTY"},
765                 {BT_PXP_MONITOR_GET_PROPERTY, "BT_PXP_MONITOR_GET_PROPERTY"},
766                 {BT_PXP_MONITOR_GET_SUPPORTED_SERIVCES, "BT_PXP_MONITOR_GET_SUPPORTED_SERIVCES"},
767                 {BT_PXP_REPORTER_REGISTER, "BT_PXP_REPORTER_REGISTER"},
768                 {BT_PXP_REPORTER_UNREGISTER, "BT_PXP_REPORTER_UNREGISTER"},
769                 {BT_PXP_REPORTER_GET_PROPERTY, "BT_PXP_REPORTER_GET_PROPERTY"},
770                 {BT_TDS_PROVIDER_REGISTER, "BT_TDS_PROVIDER_REGISTER"},
771                 {BT_TDS_PROVIDER_UNREGISTER, "BT_TDS_PROVIDER_UNREGISTER"},
772                 {BT_TDS_PROVIDER_SET_MANUF_DATA, "BT_TDS_PROVIDER_SET_MANUF_DATA"},
773                 {BT_TDS_PROVIDER_CREATE, "BT_TDS_PROVIDER_CREATE"},
774                 {BT_TDS_PROVIDER_DESTROY, "BT_TDS_PROVIDER_DESTROY"},
775                 {BT_TDS_PROVIDER_SET_TRANSPORT_DATA, "BT_TDS_PROVIDER_SET_TRANSPORT_DATA"},
776                 {BT_TDS_SEND_ACTIVATION_RESPONSE, "BT_TDS_SEND_ACTIVATION_RESPONSE"},
777                 {BT_TDS_READ_TRANSPORT_DATA, "BT_TDS_READ_TRANSPORT_DATA"},
778                 {BT_TDS_ENABLE_CONTROL_POINT, "BT_TDS_ENABLE_CONTROL_POINT"},
779                 {BT_TDS_ACTIVATE_CONTROL_POINT, "BT_TDS_ACTIVATE_CONTROL_POINT"},
780                 {BT_OTP_READ_VALUE, "BT_OTP_READ_VALUE"},
781                 {BT_OTP_ENABLE_NOTIFICATION, "BT_OTP_ENABLE_NOTIFICATION"},
782                 {BT_OTP_WRITE_VALUE, "BT_OTP_WRITE_VALUE"},
783                 {BT_LE_OTC_CONNECT, "BT_LE_OTC_CONNECT"},
784                 {BT_LE_OTC_DISCONNECT, "BT_LE_OTC_DISCONNECT"},
785                 {-1, ""},
786         };
787
788         for (i = 0; bt_functions[i].function != -1; i++) {
789                 if (bt_functions[i].function == function)
790                         return bt_functions[i].function_name;
791         }
792
793         return NULL;
794 }
795
796 int _bt_copy_utf8_string(char *dest, const char *src, unsigned int length)
797 {
798         int i;
799         const char *p = src;
800         char *next;
801         int count;
802
803         if (dest == NULL || src == NULL)
804                 return BLUETOOTH_ERROR_INVALID_PARAM;
805
806         i = 0;
807         while (*p != '\0' && i < length) {
808                 next = g_utf8_next_char(p);
809                 count = next - p;
810
811                 while (count > 0 && ((i + count) < length)) {
812                         dest[i++] = *p;
813                         p++;
814                         count--;
815                 }
816                 p = next;
817         }
818         return BLUETOOTH_ERROR_NONE;
819 }
820
821 gboolean _bt_utf8_validate(char *name)
822 {
823         BT_DBG("+");
824         gunichar2 *u16;
825         glong items_written = 0;
826
827         if (FALSE == g_utf8_validate(name, -1, NULL))
828                 return FALSE;
829
830         u16 = g_utf8_to_utf16(name, -1, NULL, &items_written, NULL);
831         if (u16 == NULL)
832                 return FALSE;
833
834         g_free(u16);
835
836         if (items_written != g_utf8_strlen(name, -1))
837                 return FALSE;
838
839         BT_DBG("-");
840         return TRUE;
841 }
842
843 static GDBusProxy *profile_gproxy;
844 static int latest_id = -1;
845 #define BT_RFCOMM_ID_MAX 245
846 static gboolean id_used[BT_RFCOMM_ID_MAX];
847
848 static const gchar rfcomm_agent_xml[] =
849 "<node name='/'>"
850 " <interface name='org.bluez.Profile1'>"
851 "     <method name='NewConnection'>"
852 "          <arg type='o' name='object' direction='in'/>"
853 "          <arg type='h' name='fd' direction='in'/>"
854 "          <arg type='a{sv}' name='properties' direction='in'/>"
855 "     </method>"
856 "     <method name='RequestDisconnection'>"
857 "          <arg type='o' name='device' direction='in'/>"
858 "     </method>"
859 "  </interface>"
860 "</node>";
861
862 static void __new_connection_method(GDBusConnection *connection,
863                 const gchar *sender,
864                 const gchar *object_path,
865                 const gchar *interface_name,
866                 const gchar *method_name,
867                 GVariant *parameters,
868                 GDBusMethodInvocation *invocation,
869                 gpointer user_data)
870 {
871         BT_DBG("method %s", method_name);
872         if (g_strcmp0(method_name, "NewConnection") == 0) {
873                 int index;
874                 GDBusMessage *msg;
875                 GUnixFDList *fd_list;
876                 GVariantBuilder *properties;
877                 char *obj_path;
878                 char addr[20];
879                 bluetooth_device_address_t  remote_addr1;
880                 bt_new_connection_cb cb = user_data;
881                 char secure_address[BT_ADDRESS_STRING_SIZE] = { 0 };
882                 int fd;
883
884                 g_variant_get(parameters, "(oha{sv})", &obj_path, &index,
885                                                                 &properties);
886
887                 msg = g_dbus_method_invocation_get_message(invocation);
888                 fd_list = g_dbus_message_get_unix_fd_list(msg);
889                 if (fd_list == NULL) {
890                         BT_ERR("fd_list is NULL");
891                         GQuark quark = g_quark_from_string("rfcomm-app");
892                         GError *err = g_error_new(quark, 0, "No fd in message");
893                         g_dbus_method_invocation_return_gerror(invocation, err);
894                         g_error_free(err);
895                         return;
896                 }
897
898
899                 fd = g_unix_fd_list_get(fd_list, index, NULL);
900                 if (fd == -1) {
901                         BT_ERR("Invalid fd return");
902                         GQuark quark = g_quark_from_string("rfcomm-app");
903                         GError *err = g_error_new(quark, 0, "Invalid FD return");
904                         g_dbus_method_invocation_return_gerror(invocation, err);
905                         g_error_free(err);
906                         return;
907                 }
908
909                 _bt_convert_device_path_to_address(obj_path, addr);
910                 _bt_convert_addr_string_to_type(remote_addr1.addr, (const char *)addr);
911
912                 _bt_convert_addr_string_to_secure_string(secure_address, addr);
913                 BT_INFO("fd: %d, address %s", fd, secure_address);
914
915                 g_dbus_method_invocation_return_value(invocation, NULL);
916
917                 if (cb)
918                         cb(object_path, fd, &remote_addr1);
919         } else if (g_strcmp0(method_name, "RequestDisconnection") == 0) {
920                 g_dbus_method_invocation_return_value(invocation, NULL);
921         }
922 }
923
924
925 static const GDBusInterfaceVTable method_table = {
926         __new_connection_method,
927         NULL,
928         NULL,
929 };
930
931 void _bt_swap_addr(unsigned char *dst, const unsigned char *src)
932 {
933         int i;
934
935         for (i = 0; i < 6; i++)
936                 dst[i] = src[5-i];
937 }
938
939 int __rfcomm_assign_id(void)
940 {
941         int index;
942
943         BT_DBG("latest_id: %d", latest_id);
944
945         index = latest_id + 1;
946
947         if (index >= BT_RFCOMM_ID_MAX)
948                 index = 0;
949
950         BT_DBG("index: %d", index);
951
952         while (id_used[index] == TRUE) {
953                 if (index == latest_id) {
954                         /* No available ID */
955                         BT_ERR("All request ID is used");
956                         return -1;
957                 }
958
959                 index++;
960
961                 if (index >= BT_RFCOMM_ID_MAX)
962                         index = 0;
963         }
964
965         latest_id = index;
966         id_used[index] = TRUE;
967
968         BT_DBG("Assigned Id: %d", latest_id);
969
970         return latest_id;
971 }
972
973 void __rfcomm_delete_id(int id)
974 {
975         ret_if(id >= BT_RFCOMM_ID_MAX);
976         ret_if(id < 0);
977
978         id_used[id] = FALSE;
979
980         /* Next server will use this ID */
981         latest_id = id - 1;
982 }
983
984 static GDBusProxy *__bt_gdbus_get_profile_proxy(void)
985 {
986         GDBusConnection *gconn;
987         GError *err = NULL;
988
989         if (profile_gproxy)
990                 return profile_gproxy;
991
992         /* Shared connection should be used because rfcomm interface was registered
993          * on shared connection not private. Otherwise, dbus rejection is occured
994          * because dbus policy is only applied on shared connection not private. */
995         gconn = _bt_get_system_shared_conn();
996         if (gconn == NULL)
997                 return NULL;
998
999         profile_gproxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES,
1000                                                 NULL, BT_BLUEZ_NAME,
1001                                                 "/org/bluez",
1002                                                 "org.bluez.ProfileManager1",
1003                                                 NULL, &err);
1004         if (err) {
1005                 BT_ERR("Unable to create proxy: %s", err->message);
1006                 g_clear_error(&err);
1007                 return NULL;
1008         }
1009
1010         return profile_gproxy;
1011 }
1012
1013 static GDBusProxy *__bt_gdbus_get_device_proxy(char *object_path)
1014 {
1015         GDBusConnection *gconn;
1016         GError *err = NULL;
1017         GDBusProxy *device_gproxy;
1018
1019         gconn = _bt_get_system_private_conn();
1020         if (gconn == NULL)
1021                 return NULL;
1022
1023         device_gproxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES,
1024                                                 NULL, BT_BLUEZ_NAME,
1025                                                 object_path,
1026                                                 BT_DEVICE_INTERFACE,
1027                                                 NULL, &err);
1028
1029         if (device_gproxy == NULL && err) {
1030                 BT_ERR("Unable to create proxy: %s", err->message);
1031                 g_clear_error(&err);
1032                 return NULL;
1033         }
1034
1035         return device_gproxy;
1036 }
1037
1038 void _bt_unregister_gdbus(int object_id)
1039 {
1040         GDBusConnection *gconn;
1041
1042         gconn = _bt_get_system_shared_conn();
1043         if (gconn == NULL)
1044                 return;
1045
1046         g_dbus_connection_unregister_object(gconn, object_id);
1047
1048         /* TODO: bus_id needs to be un-owned if all of path is unregistered */
1049 }
1050
1051 static GDBusNodeInfo *_bt_get_gdbus_node(GDBusConnection *gconn, const gchar *xml_data, const char *bus_name)
1052 {
1053         if (bus_id == 0) {
1054                 char *name = g_strdup(bus_name);
1055                 BT_DBG("well-known name: %s", name);
1056                 bus_id = g_bus_own_name_on_connection(gconn,
1057                                 name, G_BUS_NAME_OWNER_FLAGS_NONE,
1058                                 NULL, NULL, NULL, NULL);
1059                 BT_DBG("Got bus id %d", bus_id);
1060                 g_free(name);
1061         }
1062
1063         return g_dbus_node_info_new_for_xml(xml_data, NULL);
1064 }
1065
1066 int _bt_register_new_conn(const char *path, bt_new_connection_cb cb)
1067 {
1068         GDBusConnection *gconn;
1069         GDBusNodeInfo *node_info;
1070         int id;
1071         GError *error = NULL;
1072         char *bus_name;
1073
1074         gconn = _bt_get_system_shared_conn();
1075         if (gconn == NULL)
1076                 return -1;
1077
1078         bus_name = g_strdup_printf("org.bt.frwk.p%d", getpid());
1079
1080         node_info = _bt_get_gdbus_node(gconn, rfcomm_agent_xml, bus_name);
1081         if (node_info == NULL) {
1082                 g_free(bus_name);
1083                 return -1;
1084         }
1085         g_free(bus_name);
1086
1087         id = g_dbus_connection_register_object(gconn, path,
1088                                                 node_info->interfaces[0],
1089                                                 &method_table,
1090                                                 cb, NULL, &error);
1091         g_dbus_node_info_unref(node_info);
1092         if (id == 0) {
1093                 BT_ERR("Failed to register: %s", error->message);
1094                 g_error_free(error);
1095                 return -1;
1096         }
1097
1098         BT_DBG("NEW CONNECTION ID %d", id);
1099
1100         return id;
1101 }
1102
1103 static GDBusProxy * __bt_gdbus_get_adapter_proxy()
1104 {
1105         GError *err = NULL;
1106         GDBusProxy *manager_proxy = NULL;
1107         GDBusProxy *adapter_proxy = NULL;
1108         GDBusConnection *conn;
1109         GVariant *result = NULL;
1110         char *adapter_path = NULL;
1111
1112         conn = _bt_get_system_private_conn();
1113         retv_if(conn == NULL, NULL);
1114
1115         manager_proxy =  g_dbus_proxy_new_sync(conn,
1116                         G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL,
1117                         BT_BLUEZ_NAME,
1118                         BT_MANAGER_PATH,
1119                         BT_MANAGER_INTERFACE,
1120                         NULL, &err);
1121
1122         if (!manager_proxy) {
1123                 BT_ERR("Unable to create proxy: %s", err->message);
1124                 goto fail;
1125         }
1126
1127         result = g_dbus_proxy_call_sync(manager_proxy, "DefaultAdapter", NULL,
1128                         G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
1129         if (!result) {
1130                 if (err != NULL)
1131                         BT_ERR("Fail to get DefaultAdapter (Error: %s)", err->message);
1132                 else
1133                         BT_ERR("Fail to get DefaultAdapter");
1134
1135                 goto fail;
1136         }
1137
1138         if (g_strcmp0(g_variant_get_type_string(result), "(o)")) {
1139                 BT_ERR("Incorrect result\n");
1140                 goto fail;
1141         }
1142
1143         g_variant_get(result, "(&o)", &adapter_path);
1144
1145         if (adapter_path == NULL ||
1146                 strlen(adapter_path) >= BT_ADAPTER_OBJECT_PATH_MAX) {
1147                 BT_ERR("Adapter path is inproper\n");
1148                 goto fail;
1149         }
1150
1151         BT_INFO("Adapter Path %s", adapter_path);
1152
1153         adapter_proxy = g_dbus_proxy_new_sync(conn,
1154                                         G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL,
1155                                         BT_BLUEZ_NAME,
1156                                         adapter_path,
1157                                         BT_ADAPTER_INTERFACE,
1158                                         NULL, &err);
1159         if (err) {
1160                 BT_ERR("DBus Error message: [%s]", err->message);
1161                 g_clear_error(&err);
1162         }
1163
1164 fail:
1165         if (manager_proxy)
1166                 g_object_unref(manager_proxy);
1167         if (result)
1168                 g_variant_unref(result);
1169         return adapter_proxy;
1170 }
1171
1172 int _bt_register_new_conn_ex(const char *path, const char *bus_name, bt_new_connection_cb cb)
1173 {
1174         GDBusConnection *gconn;
1175         GDBusNodeInfo *node_info;
1176         int id;
1177         GError *error = NULL;
1178
1179         gconn = _bt_get_system_shared_conn();
1180         if (gconn == NULL)
1181                 return -1;
1182
1183         node_info = _bt_get_gdbus_node(gconn, rfcomm_agent_xml, bus_name);
1184         if (node_info == NULL)
1185                 return -1;
1186
1187         id = g_dbus_connection_register_object(gconn, path,
1188                                                 node_info->interfaces[0],
1189                                                 &method_table,
1190                                                 cb, NULL, &error);
1191         g_dbus_node_info_unref(node_info);
1192         if (id == 0) {
1193                 BT_ERR("Failed to register: %s", error->message);
1194                 g_error_free(error);
1195                 return -1;
1196         }
1197
1198         BT_DBG("NEW CONNECTION ID %d", id);
1199
1200         return id;
1201 }
1202
1203 int _bt_register_profile(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
1204 {
1205         GVariantBuilder *option_builder;
1206         GVariant *ret;
1207         GDBusProxy *proxy;
1208         GError *err = NULL;
1209         int result = BLUETOOTH_ERROR_NONE;
1210
1211         proxy = __bt_gdbus_get_profile_proxy();
1212         if (proxy == NULL) {
1213                 BT_ERR("Getting profile proxy failed");
1214                 return BLUETOOTH_ERROR_INTERNAL;
1215         }
1216
1217         option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
1218         if (info->authentication)
1219                 g_variant_builder_add(option_builder, "{sv}",
1220                                                 "RequireAuthentication",
1221                                                 g_variant_new_boolean(TRUE));
1222         if (info->authorization)
1223                 g_variant_builder_add(option_builder, "{sv}",
1224                                                 "RequireAuthorization",
1225                                                 g_variant_new_boolean(TRUE));
1226         if (info->role)
1227                 g_variant_builder_add(option_builder, "{sv}",
1228                                                 "Role",
1229                                                 g_variant_new_string(info->role));
1230
1231         /* Setting RFCOMM channel to default value 0; would allow bluez to assign
1232          * RFCOMM channels based on the availability when two services want
1233          * to use the RFCOMM along with SPP. Hence bluez makes sure that no
1234          * two services use the same SPP RFCOMM channel. */
1235         if (use_default_rfcomm)
1236                 g_variant_builder_add(option_builder, "{sv}",
1237                                                 "Channel",
1238                                                 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
1239         if (info->service)
1240                 g_variant_builder_add(option_builder, "{sv}",
1241                                                 "Service",
1242                                                 g_variant_new_string(info->service));
1243
1244         ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile",
1245                                         g_variant_new("(osa{sv})", info->obj_path,
1246                                                                 info->uuid,
1247                                                                 option_builder),
1248                                         G_DBUS_CALL_FLAGS_NONE, -1,
1249                                         NULL, &err);
1250         if (err) {
1251                 g_dbus_error_strip_remote_error(err);
1252                 BT_ERR("RegisterProfile failed: %s", err->message);
1253
1254                 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
1255                         result = BLUETOOTH_ERROR_ACCESS_DENIED;
1256                 else
1257                         result = BLUETOOTH_ERROR_INTERNAL;
1258
1259                 g_clear_error(&err);
1260         }
1261
1262         g_variant_builder_unref(option_builder);
1263
1264         if (ret)
1265                 g_variant_unref(ret);
1266
1267         return result;
1268 }
1269
1270 int _bt_register_profile_ex(bt_register_profile_info_t *info, gboolean use_default_rfcomm, const char *name, const char *path)
1271 {
1272         GVariantBuilder *option_builder;
1273         GVariant *ret;
1274         GDBusProxy *proxy;
1275         GError *err = NULL;
1276         int result = BLUETOOTH_ERROR_NONE;
1277
1278         proxy = __bt_gdbus_get_profile_proxy();
1279         if (proxy == NULL) {
1280                 BT_ERR("Getting profile proxy failed");
1281                 return BLUETOOTH_ERROR_INTERNAL;
1282         }
1283
1284         option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
1285         if (info->authentication)
1286                 g_variant_builder_add(option_builder, "{sv}",
1287                                                 "RequireAuthentication",
1288                                                 g_variant_new_boolean(TRUE));
1289         if (info->authorization)
1290                 g_variant_builder_add(option_builder, "{sv}",
1291                                                 "RequireAuthorization",
1292                                                 g_variant_new_boolean(TRUE));
1293         if (info->role)
1294                 g_variant_builder_add(option_builder, "{sv}",
1295                                                 "Role",
1296                                                 g_variant_new_string(info->role));
1297
1298         /* Setting RFCOMM channel to default value 0; would allow bluez to assign
1299          * RFCOMM channels based on the availability when two services want
1300          * to use the RFCOMM along with SPP. Hence bluez makes sure that no
1301          * two services use the same SPP RFCOMM channel. */
1302         if (use_default_rfcomm)
1303                 g_variant_builder_add(option_builder, "{sv}",
1304                                                 "Channel",
1305                                                 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
1306         if (info->service)
1307                 g_variant_builder_add(option_builder, "{sv}",
1308                                                 "Service",
1309                                                 g_variant_new_string(info->service));
1310
1311         ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile2",
1312                                         g_variant_new("(osssa{sv})", info->obj_path,
1313                                                                 info->uuid,
1314                                                                 name,
1315                                                                 path,
1316                                                                 option_builder),
1317                                         G_DBUS_CALL_FLAGS_NONE, -1,
1318                                         NULL, &err);
1319         if (err) {
1320                 g_dbus_error_strip_remote_error(err);
1321                 BT_ERR("RegisterProfile2 failed: %s", err->message);
1322
1323                 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
1324                         result = BLUETOOTH_ERROR_ACCESS_DENIED;
1325                 else
1326                         result = BLUETOOTH_ERROR_INTERNAL;
1327
1328                 g_clear_error(&err);
1329         }
1330
1331         g_variant_builder_unref(option_builder);
1332
1333         if (ret)
1334                 g_variant_unref(ret);
1335
1336         return result;
1337 }
1338
1339 int _bt_register_profile_platform(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
1340 {
1341         GVariantBuilder *option_builder;
1342         GVariant *ret;
1343         GDBusProxy *proxy;
1344         GError *err = NULL;
1345         int result = BLUETOOTH_ERROR_NONE;
1346
1347         proxy = __bt_gdbus_get_profile_proxy();
1348         if (proxy == NULL) {
1349                 BT_ERR("Getting profile proxy failed");
1350                 return BLUETOOTH_ERROR_INTERNAL;
1351         }
1352
1353         option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
1354         if (info->authentication)
1355                 g_variant_builder_add(option_builder, "{sv}",
1356                                                 "RequireAuthentication",
1357                                                 g_variant_new_boolean(TRUE));
1358         if (info->authorization)
1359                 g_variant_builder_add(option_builder, "{sv}",
1360                                                 "RequireAuthorization",
1361                                                 g_variant_new_boolean(TRUE));
1362         if (info->role)
1363                 g_variant_builder_add(option_builder, "{sv}",
1364                                                 "Role",
1365                                                 g_variant_new_string(info->role));
1366
1367         /* Setting RFCOMM channel to default value 0; would allow bluez to assign
1368          * RFCOMM channels based on the availability when two services want
1369          * to use the RFCOMM along with SPP. Hence bluez makes sure that no
1370          * two services use the same SPP RFCOMM channel. */
1371         if (use_default_rfcomm)
1372                 g_variant_builder_add(option_builder, "{sv}",
1373                                                 "Channel",
1374                                                 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
1375         if (info->service)
1376                 g_variant_builder_add(option_builder, "{sv}",
1377                                                 "Service",
1378                                                 g_variant_new_string(info->service));
1379
1380         ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile1",
1381                                         g_variant_new("(osa{sv})", info->obj_path,
1382                                                                 info->uuid,
1383                                                                 option_builder),
1384                                         G_DBUS_CALL_FLAGS_NONE, -1,
1385                                         NULL, &err);
1386
1387         if (err) {
1388                 g_dbus_error_strip_remote_error(err);
1389                 BT_ERR("RegisterProfile1 failed: %s", err->message);
1390
1391                 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
1392                         result = BLUETOOTH_ERROR_ACCESS_DENIED;
1393                 else
1394                         result = BLUETOOTH_ERROR_INTERNAL;
1395
1396                 g_clear_error(&err);
1397         }
1398
1399         g_variant_builder_unref(option_builder);
1400
1401         if (ret)
1402                 g_variant_unref(ret);
1403
1404         return result;
1405 }
1406
1407
1408 void _bt_unregister_profile(char *path)
1409 {
1410         GVariant *ret;
1411         GDBusProxy *proxy;
1412         GError *err = NULL;
1413
1414         proxy = __bt_gdbus_get_profile_proxy();
1415         if (proxy == NULL) {
1416                 BT_ERR("Getting profile proxy failed");
1417                 return;
1418         }
1419
1420         ret = g_dbus_proxy_call_sync(proxy, "UnregisterProfile",
1421                         g_variant_new("(o)", path),
1422                         G_DBUS_CALL_FLAGS_NONE, -1,
1423                         NULL, &err);
1424         if (err) {
1425                 BT_ERR("UnregisterProfile failed : %s", err->message);
1426                 g_clear_error(&err);
1427         }
1428
1429         if (ret)
1430                 g_variant_unref(ret);
1431
1432         return;
1433 }
1434
1435 void _bt_print_api_caller_name(void)
1436 {
1437         FILE *fp = NULL;
1438         char *path = NULL;
1439         char buf[256] = {0, };
1440         char **str_list = NULL;
1441
1442         path = g_strdup_printf("/proc/%d/cmdline", getpid());
1443         fp = fopen(path, "r");
1444         if (fp == NULL) {
1445                 g_free(path);
1446                 return;
1447         }
1448
1449         if (fgets(buf, 256, fp) != NULL) {
1450                 str_list = g_strsplit(buf, " ", -1);
1451                 if (str_list[0] != NULL)
1452                         BT_INFO("Caller : %s", str_list[0]);
1453                 g_strfreev(str_list);
1454         }
1455
1456         fclose(fp);
1457         g_free(path);
1458         return;
1459 }
1460
1461 int _bt_connect_profile(char *address, char *uuid, void *cb,
1462                                                         gpointer func_data)
1463 {
1464         GDBusProxy *proxy;
1465         GDBusProxy *adapter_proxy;
1466         char *object_path;
1467         GError *err = NULL;
1468
1469         object_path = _bt_get_device_object_path(address);
1470
1471         if (object_path == NULL) {
1472                 GVariant *ret = NULL;
1473                 BT_ERR("No searched device");
1474                 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1475
1476                 if (adapter_proxy == NULL) {
1477                         BT_ERR("adapter proxy is NULL");
1478                         return BLUETOOTH_ERROR_INTERNAL;
1479                 }
1480
1481                 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1482                                 g_variant_new("(s)", address),
1483                                 G_DBUS_CALL_FLAGS_NONE,
1484                                 DBUS_TIMEOUT, NULL,
1485                                 &err);
1486
1487                 if (err != NULL) {
1488                         BT_ERR("CreateDevice Failed: %s", err->message);
1489                         g_clear_error(&err);
1490                 }
1491                 if (ret)
1492                         g_variant_unref(ret);
1493                 g_object_unref(adapter_proxy);
1494                 object_path = _bt_get_device_object_path(address);
1495
1496                 if (object_path == NULL)
1497                         return BLUETOOTH_ERROR_INTERNAL;
1498         }
1499
1500         proxy = __bt_gdbus_get_device_proxy(object_path);
1501         g_free(object_path);
1502
1503         if (proxy == NULL) {
1504                 BT_ERR("Error while getting proxy");
1505                 return BLUETOOTH_ERROR_INTERNAL;
1506         }
1507
1508         g_dbus_proxy_call(proxy, "ConnectProfile",
1509                         g_variant_new("(s)", uuid),
1510                         G_DBUS_CALL_FLAGS_NONE,
1511                         DBUS_TIMEOUT, NULL,
1512                         (GAsyncReadyCallback)cb,
1513                         func_data);
1514         if (cb == NULL)
1515                 g_object_unref(proxy);
1516         BT_DBG("-");
1517         return BLUETOOTH_ERROR_NONE;
1518 }
1519
1520 int _bt_discover_services(char *address, char *uuid, void *cb,
1521                 gpointer func_data)
1522 {
1523         char *object_path;
1524         GDBusProxy *proxy;
1525         GDBusProxy *adapter_proxy;
1526         GError *err = NULL;
1527         object_path = _bt_get_device_object_path(address);
1528         if (object_path == NULL) {
1529                 GVariant *ret = NULL;
1530                 BT_ERR("No searched device");
1531                 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1532                 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1533                 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1534                                 g_variant_new("(s)", address),
1535                                 G_DBUS_CALL_FLAGS_NONE,
1536                                 DBUS_TIMEOUT, NULL,
1537                                 &err);
1538                 if (err != NULL) {
1539                         BT_ERR("CreateDevice Failed: %s", err->message);
1540                         g_clear_error(&err);
1541                 }
1542                 if (ret)
1543                         g_variant_unref(ret);
1544                 g_object_unref(adapter_proxy);
1545                 object_path = _bt_get_device_object_path(address);
1546                 if (object_path == NULL)
1547                         return BLUETOOTH_ERROR_INTERNAL;
1548         }
1549         proxy = __bt_gdbus_get_device_proxy(object_path);
1550         g_free(object_path);
1551         if (proxy == NULL) {
1552                 BT_ERR("Error while getting proxy");
1553                 return BLUETOOTH_ERROR_INTERNAL;
1554         }
1555         g_dbus_proxy_call(proxy, "DiscoverServices",
1556                         g_variant_new("(s)", uuid),
1557                         G_DBUS_CALL_FLAGS_NONE,
1558                         DBUS_TIMEOUT, NULL,
1559                         (GAsyncReadyCallback)cb,
1560                         func_data);
1561         if (cb == NULL)
1562                 g_object_unref(proxy);
1563         BT_DBG("-");
1564         return BLUETOOTH_ERROR_NONE;
1565 }
1566
1567 int _bt_cancel_discovers(char *address)
1568 {
1569         char *object_path;
1570         GDBusProxy *proxy;
1571         GDBusProxy *adapter_proxy;
1572         GVariant *ret = NULL;
1573         GError *err = NULL;
1574         object_path = _bt_get_device_object_path(address);
1575         if (object_path == NULL) {
1576                 GVariant *ret = NULL;
1577                 BT_ERR("No searched device");
1578                 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1579                 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1580                 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1581                                 g_variant_new("(s)", address),
1582                                 G_DBUS_CALL_FLAGS_NONE,
1583                                 DBUS_TIMEOUT, NULL,
1584                                 &err);
1585                 if (err != NULL) {
1586                         BT_ERR("CreateDevice Failed: %s", err->message);
1587                         g_clear_error(&err);
1588                 }
1589                 if (ret)
1590                         g_variant_unref(ret);
1591                 g_object_unref(adapter_proxy);
1592                 object_path = _bt_get_device_object_path(address);
1593                 if (object_path == NULL)
1594                         return BLUETOOTH_ERROR_INTERNAL;
1595         }
1596         proxy = __bt_gdbus_get_device_proxy(object_path);
1597         g_free(object_path);
1598         ret = g_dbus_proxy_call_sync(proxy, "CancelDiscovery",
1599                 NULL,
1600                 G_DBUS_CALL_FLAGS_NONE,
1601                 DBUS_TIMEOUT, NULL,
1602                 &err);
1603         if (err) {
1604                 BT_ERR("DBus Error message: [%s]", err->message);
1605                 g_clear_error(&err);
1606                 return BLUETOOTH_ERROR_INTERNAL;
1607         }
1608         if (ret)
1609                 g_variant_unref(ret);
1610         if (proxy)
1611                 g_object_unref(proxy);
1612         return BLUETOOTH_ERROR_NONE;
1613 }
1614
1615 int _bt_discover_service_uuids(char *address, char *remote_uuid)
1616 {
1617         char *object_path;
1618         GDBusProxy *proxy;
1619         GDBusConnection *gconn;
1620         GError *err = NULL;
1621         char **uuid_value = NULL;
1622         gsize size = 0;
1623         int i = 0;
1624         GVariant *value = NULL;
1625         GVariant *ret = NULL;
1626         int result = BLUETOOTH_ERROR_INTERNAL;
1627
1628         retv_if(remote_uuid == NULL, BLUETOOTH_ERROR_INTERNAL);
1629         gconn = _bt_get_system_private_conn();
1630         retv_if(gconn == NULL, BLUETOOTH_ERROR_INTERNAL);
1631         object_path = _bt_get_device_object_path(address);
1632         retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
1633
1634         proxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL,
1635                                 BT_BLUEZ_NAME, object_path, BT_PROPERTIES_INTERFACE, NULL,
1636                                 &err);
1637         g_free(object_path);
1638         if (proxy == NULL) {
1639                 if (err) {
1640                         BT_ERR("DBus Error: [%s]", err->message);
1641                         g_clear_error(&err);
1642                 }
1643                 return BLUETOOTH_ERROR_INTERNAL;
1644         }
1645         ret = g_dbus_proxy_call_sync(proxy, "Get",
1646                         g_variant_new("(ss)", BT_DEVICE_INTERFACE, "UUIDs"),
1647                         G_DBUS_CALL_FLAGS_NONE, DBUS_TIMEOUT, NULL, &err);
1648         if (err) {
1649                 result = BLUETOOTH_ERROR_INTERNAL;
1650                 BT_ERR("DBus Error : %s", err->message);
1651                 g_clear_error(&err);
1652                 goto done;
1653         } else {
1654                 g_variant_get(ret, "(v)", &value);
1655                 uuid_value = (char **)g_variant_get_strv(value, &size);
1656                 BT_DBG("Size items %zd", size);
1657         }
1658
1659         if (uuid_value) {
1660                 for (i = 0; uuid_value[i] != NULL; i++) {
1661                         BT_DBG("Remote uuids %s", uuid_value[i]);
1662                         if (uuid_value[i] == NULL) {
1663                                 BT_ERR("_bt_discover_service_uuids Error Parameter are NULL..\n");
1664                                 continue;
1665                         } else if (g_ascii_strcasecmp(uuid_value[i], remote_uuid) == 0) {
1666                                 result = BLUETOOTH_ERROR_NONE;
1667                                 goto done;
1668                         }
1669                 }
1670         }
1671
1672 done:
1673         if (proxy)
1674                 g_object_unref(proxy);
1675         if (uuid_value)
1676                 g_free(uuid_value);
1677
1678         if (value)
1679                 g_variant_unref(value);
1680
1681         if (ret)
1682                 g_variant_unref(ret);
1683
1684         BT_DBG("-");
1685         return result;
1686 }
1687
1688 int _bt_get_cod_by_address(char *address, bluetooth_device_class_t *dev_class)
1689 {
1690         char *object_path;
1691         GDBusProxy *proxy;
1692         GDBusConnection *gconn;
1693         GError *err = NULL;
1694         GVariant *value = NULL;
1695         GVariant *result = NULL;
1696         unsigned int  class = 0x00;
1697         int ret = BLUETOOTH_ERROR_NONE;
1698
1699         gconn = _bt_get_system_private_conn();
1700         retv_if(gconn == NULL, BLUETOOTH_ERROR_INTERNAL);
1701
1702         object_path = _bt_get_device_object_path(address);
1703         retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
1704
1705         proxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL,
1706                                 BT_BLUEZ_NAME, object_path, BT_PROPERTIES_INTERFACE, NULL,
1707                                 &err);
1708         if (proxy == NULL) {
1709                 ret = BLUETOOTH_ERROR_INTERNAL;
1710                 if (err) {
1711                         BT_ERR("DBus Error: [%s]", err->message);
1712                         g_clear_error(&err);
1713                 }
1714                 goto done;
1715         }
1716
1717         result = g_dbus_proxy_call_sync(proxy, "Get",
1718                         g_variant_new("(ss)", BT_DEVICE_INTERFACE, "Class"),
1719                         G_DBUS_CALL_FLAGS_NONE, DBUS_TIMEOUT, NULL, &err);
1720         if (err) {
1721                 ret = BLUETOOTH_ERROR_INTERNAL;
1722                 BT_ERR("DBus Error : %s", err->message);
1723                 g_clear_error(&err);
1724                 goto done;
1725         } else {
1726                 g_variant_get(result, "(v)", &value);
1727                 class = g_variant_get_uint32(value);
1728                 _bt_divide_device_class(dev_class, class);
1729                 g_variant_unref(value);
1730                 g_variant_unref(result);
1731         }
1732
1733 done:
1734         g_free(object_path);
1735         if (proxy)
1736                 g_object_unref(proxy);
1737
1738         BT_DBG("-");
1739         return ret;
1740 }
1741
1742 int _bt_disconnect_profile(char *address, char *uuid, void *cb,
1743                                                         gpointer func_data)
1744 {
1745         GDBusProxy *proxy;
1746         char *object_path;
1747         GError *err = NULL;
1748         GDBusProxy *adapter_proxy;
1749         object_path = _bt_get_device_object_path(address);
1750         if (object_path == NULL) {
1751                 GVariant *ret = NULL;
1752                 BT_ERR("No searched device");
1753                 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1754                 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1755                 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1756                                 g_variant_new("(s)", address),
1757                                 G_DBUS_CALL_FLAGS_NONE,
1758                                 DBUS_TIMEOUT, NULL,
1759                                 &err);
1760                 if (err != NULL) {
1761                         BT_ERR("CreateDevice Failed: %s", err->message);
1762                         g_error_free(err);
1763                 }
1764                 if (ret)
1765                         g_variant_unref(ret);
1766                 g_object_unref(adapter_proxy);
1767                 object_path = _bt_get_device_object_path(address);
1768                 if (object_path == NULL)
1769                         return BLUETOOTH_ERROR_INTERNAL;
1770         }
1771         proxy = __bt_gdbus_get_device_proxy(object_path);
1772         g_free(object_path);
1773         if (proxy == NULL) {
1774                 BT_ERR("Error while getting proxy");
1775                 return BLUETOOTH_ERROR_INTERNAL;
1776         }
1777         g_dbus_proxy_call(proxy, "DisconnectProfile",
1778                         g_variant_new("(s)", uuid),
1779                         G_DBUS_CALL_FLAGS_NONE,
1780                         DBUS_TIMEOUT, NULL,
1781                         (GAsyncReadyCallback)cb,
1782                         func_data);
1783         if (cb == NULL)
1784                 g_object_unref(proxy);
1785         BT_DBG("-");
1786         return BLUETOOTH_ERROR_NONE;
1787 }
1788
1789 int _bt_disconnect_ext_profile(char *address, char *path)
1790 {
1791         GDBusProxy *proxy;
1792         char *object_path;
1793         GDBusConnection *gconn;
1794         GError *err = NULL;
1795
1796         object_path = _bt_get_device_object_path(address);
1797         if (object_path == NULL)
1798                 return BLUETOOTH_ERROR_INTERNAL;
1799
1800         gconn = _bt_get_system_shared_conn();
1801         if (gconn == NULL) {
1802                 g_free(object_path);
1803                 return BLUETOOTH_ERROR_INTERNAL;
1804         }
1805
1806         proxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES,
1807                                                 NULL, BT_BLUEZ_NAME,
1808                                                 object_path,
1809                                                 BT_DEVICE_INTERFACE,
1810                                                 NULL, &err);
1811
1812         g_free(object_path);
1813
1814         if (proxy == NULL && err) {
1815                 BT_ERR("Unable to create proxy: %s", err->message);
1816                 g_clear_error(&err);
1817                 return BLUETOOTH_ERROR_INTERNAL;
1818         }
1819
1820         g_dbus_proxy_call(proxy, "DisconnectExtProfile",
1821                         g_variant_new("(o)", path),
1822                         G_DBUS_CALL_FLAGS_NONE,
1823                         DBUS_TIMEOUT, NULL, NULL, NULL);
1824         g_object_unref(proxy);
1825         BT_DBG("-");
1826         return BLUETOOTH_ERROR_NONE;
1827 }
1828
1829 int _bt_get_adapter_path(GDBusConnection *conn, char *path)
1830 {
1831         GError *err = NULL;
1832         GDBusProxy *manager_proxy = NULL;
1833         GVariant *result = NULL;
1834         char *adapter_path = NULL;
1835
1836         retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
1837
1838         manager_proxy =  g_dbus_proxy_new_sync(conn,
1839                         G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL,
1840                         BT_BLUEZ_NAME,
1841                         BT_MANAGER_PATH,
1842                         BT_MANAGER_INTERFACE,
1843                         NULL, &err);
1844
1845         if (!manager_proxy) {
1846                 BT_ERR("Unable to create proxy: %s", err->message);
1847                 goto fail;
1848         }
1849
1850         result = g_dbus_proxy_call_sync(manager_proxy, "DefaultAdapter", NULL,
1851                         G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
1852         if (!result) {
1853                 if (err != NULL) {
1854                         if (!g_strrstr(err->message, "ServiceUnknown"))
1855                                 BT_ERR("Fail to get DefaultAdapter (Error: %s)", err->message);
1856                 } else {
1857                         BT_ERR("Fail to get DefaultAdapter");
1858                 }
1859
1860                 goto fail;
1861         }
1862
1863         if (g_strcmp0(g_variant_get_type_string(result), "(o)")) {
1864                 BT_ERR("Incorrect result\n");
1865                 goto fail;
1866         }
1867
1868         g_variant_get(result, "(&o)", &adapter_path);
1869
1870         if (adapter_path == NULL ||
1871                 strlen(adapter_path) >= BT_ADAPTER_OBJECT_PATH_MAX) {
1872                 BT_ERR("Adapter path is inproper\n");
1873                 goto fail;
1874         }
1875
1876         if (path)
1877                 g_strlcpy(path, adapter_path, BT_ADAPTER_OBJECT_PATH_MAX);
1878
1879         g_variant_unref(result);
1880         g_object_unref(manager_proxy);
1881
1882         return BLUETOOTH_ERROR_NONE;
1883
1884 fail:
1885         g_clear_error(&err);
1886
1887         if (result)
1888                 g_variant_unref(result);
1889
1890         if (manager_proxy)
1891                 g_object_unref(manager_proxy);
1892
1893         return BLUETOOTH_ERROR_INTERNAL;
1894
1895 }
1896
1897 void _bt_convert_device_path_to_addr_type(const char *device_path,
1898                                           unsigned char *addr)
1899 {
1900         char *dev_addr;
1901         char address[BT_ADDRESS_STRING_SIZE] = { 0 };
1902         int i;
1903         char *addr_ptr = address;
1904         char *ptr = NULL;
1905
1906         ret_if(device_path == NULL);
1907         ret_if(addr == NULL);
1908
1909         dev_addr = strstr(device_path, "dev_");
1910         if (dev_addr != NULL) {
1911                 dev_addr += 4;
1912                 g_strlcpy(addr_ptr, dev_addr, BT_ADDRESS_STRING_SIZE);
1913
1914                 for (i = 0; i < BT_ADDRESS_LENGTH_MAX; i++) {
1915                         addr[i] = strtol(addr_ptr, &ptr, 16);
1916                         if (ptr[0] != '\0') {
1917                                 if (ptr[0] != '_')
1918                                         return;
1919
1920                                 addr_ptr = ptr + 1;
1921                         }
1922                 }
1923         }
1924 }
1925
1926 void _bt_convert_device_path_to_address(const char *device_path,
1927                                                 char *device_address)
1928 {
1929         char address[BT_ADDRESS_STRING_SIZE] = { 0 };
1930         char *dev_addr;
1931
1932         ret_if(device_path == NULL);
1933         ret_if(device_address == NULL);
1934
1935         dev_addr = strstr(device_path, "dev_");
1936         if (dev_addr != NULL) {
1937                 char *pos = NULL;
1938                 dev_addr += 4;
1939                 g_strlcpy(address, dev_addr, sizeof(address));
1940
1941                 while ((pos = strchr(address, '_')) != NULL)
1942                         *pos = ':';
1943
1944                 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
1945         }
1946 }
1947
1948 static char *__bt_extract_device_path(GVariantIter *iter, char *address)
1949 {
1950         char *object_path = NULL;
1951         char device_address[BT_ADDRESS_STRING_SIZE] = { 0 };
1952         /* Parse the signature:  oa{sa{sv}}} */
1953         while (g_variant_iter_loop(iter, "{&oa{sa{sv}}}", &object_path,
1954                         NULL)) {
1955                 retv_if(object_path == NULL, NULL);
1956                 _bt_convert_device_path_to_address(object_path, device_address);
1957
1958                 if (g_strcmp0(address, device_address) == 0)
1959                         return g_strdup(object_path);
1960         }
1961         return NULL;
1962 }
1963
1964 char *_bt_get_device_object_path(char *address)
1965 {
1966         GError *err = NULL;
1967         GDBusProxy *proxy = NULL;
1968         GVariant *result = NULL;
1969         GVariantIter *iter = NULL;
1970         GDBusConnection *conn = NULL;
1971         char *object_path = NULL;
1972
1973         conn = _bt_get_system_private_conn();
1974         retv_if(conn == NULL, NULL);
1975
1976         proxy =  g_dbus_proxy_new_sync(conn,
1977                         G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL,
1978                         BT_BLUEZ_NAME,
1979                         BT_MANAGER_PATH,
1980                         BT_MANAGER_INTERFACE,
1981                         NULL, &err);
1982
1983         if (!proxy) {
1984                 BT_ERR("Unable to create proxy: %s", err->message);
1985                 goto fail;
1986         }
1987
1988         result = g_dbus_proxy_call_sync(proxy, "GetManagedObjects", NULL,
1989                         G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
1990         if (!result) {
1991                 if (err != NULL)
1992                         BT_ERR("Fail to get GetManagedObjects (Error: %s)", err->message);
1993                 else
1994                         BT_ERR("Fail to get GetManagedObjects");
1995
1996                 goto fail;
1997         }
1998
1999         g_variant_get(result, "(a{oa{sa{sv}}})", &iter);
2000         object_path = __bt_extract_device_path(iter, address);
2001
2002         g_variant_unref(result);
2003         g_object_unref(proxy);
2004         g_variant_iter_free(iter);
2005         return object_path;
2006
2007 fail:
2008         g_clear_error(&err);
2009
2010         if (proxy)
2011                 g_object_unref(proxy);
2012
2013         return object_path;
2014 }
2015
2016 int _bt_register_osp_server_in_agent(int type, char *uuid, char *path, int fd)
2017 {
2018         int ret;
2019         char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
2020         char path_str[BLUETOOTH_PATH_STRING] = { 0, };
2021
2022         BT_DBG("+");
2023         BT_INIT_PARAMS();
2024         BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
2025
2026         g_array_append_vals(in_param1, &type, sizeof(int));
2027         g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
2028         g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
2029         g_strlcpy(path_str, path, sizeof(path_str));
2030         g_array_append_vals(in_param3, &path_str, BLUETOOTH_PATH_STRING);
2031         g_array_append_vals(in_param4, &fd, sizeof(int));
2032
2033         ret =  _bt_send_request(BT_AGENT_SERVICE, BT_SET_AUTHORIZATION,
2034                 in_param1, in_param2, in_param3, in_param4, &out_param);
2035
2036         BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
2037         BT_DBG("-");
2038         return ret;
2039 }
2040
2041 int _bt_unregister_osp_server_in_agent(int type, char *uuid)
2042 {
2043         int ret;
2044         char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
2045
2046         BT_DBG("+");
2047         BT_INIT_PARAMS();
2048         BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
2049
2050         g_array_append_vals(in_param1, &type, sizeof(int));
2051         g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
2052         g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
2053
2054         ret =  _bt_send_request(BT_AGENT_SERVICE, BT_UNSET_AUTHORIZATION,
2055                 in_param1, in_param2, in_param3, in_param4, &out_param);
2056
2057         BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
2058         BT_DBG("-");
2059         return ret;
2060 }
2061
2062 int _bt_check_privilege(int service_type, int service_function)
2063 {
2064         int result;
2065
2066         BT_CHECK_ENABLED(return);
2067
2068         BT_INIT_PARAMS();
2069         BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
2070
2071         result = _bt_sync_send_request(service_type, service_function,
2072                 in_param1, in_param2, in_param3, in_param4, &out_param);
2073
2074         BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
2075
2076         return result;
2077 }
2078
2079 GVariant *_bt_get_managed_objects(void)
2080 {
2081         GDBusConnection *g_conn;
2082         GDBusProxy *manager_proxy = NULL;
2083         GVariant *result = NULL;
2084         GError *error = NULL;
2085
2086         BT_DBG("+");
2087
2088         g_conn = _bt_get_system_private_conn();
2089         retv_if(g_conn == NULL, NULL);
2090
2091         manager_proxy = g_dbus_proxy_new_sync(g_conn,
2092                         G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL,
2093                         BT_BLUEZ_NAME,
2094                         BT_MANAGER_PATH,
2095                         BT_MANAGER_INTERFACE,
2096                         NULL, &error);
2097
2098         if (error) {
2099                 BT_ERR("Unable to create proxy: %s", error->message);
2100                 g_clear_error(&error);
2101                 if (manager_proxy)
2102                         g_object_unref(manager_proxy);
2103                 return NULL;
2104         }
2105
2106         if (!manager_proxy)
2107                 return NULL;
2108
2109         result = g_dbus_proxy_call_sync(manager_proxy,
2110                         "GetManagedObjects", NULL,
2111                         G_DBUS_CALL_FLAGS_NONE, -1,
2112                         NULL, &error);
2113
2114         if (error) {
2115                 BT_ERR("Fail to get ManagedObjects (Error: %s)", error->message);
2116                 g_clear_error(&error);
2117         }
2118
2119         g_object_unref(manager_proxy);
2120
2121         BT_DBG("-");
2122         return result;
2123 }
2124
2125 gboolean _bt_check_enabled_internal(void)
2126 {
2127         if (bt_enabled == TRUE)
2128                 return TRUE;
2129
2130         if (bluetooth_check_adapter() == BLUETOOTH_ADAPTER_ENABLED)
2131                 bt_enabled = TRUE;
2132
2133         return bt_enabled;
2134 }
2135
2136 void _bt_set_adapter_internal_status(gboolean enabled)
2137 {
2138         bt_enabled = enabled;
2139 }
2140
2141 BT_EXPORT_API int bluetooth_get_uuid_name(const char *uuid, char **name)
2142 {
2143 #define SHORT_UUID_COUNT        199
2144 #define LONG_UUID_COUNT         17
2145
2146         int offset = 0;
2147         int uuid_len = 4;
2148         typedef struct {
2149                 const char *uuid;
2150                 const char *specification_name;
2151         } uuid_name_s;
2152         static uuid_name_s short_uuid_name[SHORT_UUID_COUNT] = {
2153                 // List should be sorted by UUID
2154                 /* BT Classic Services */
2155                 {"1101", "Serial Port Service"},
2156                 {"1102", "LAN Access Using PPP Service"},
2157                 {"1103", "Dialup Networking Service"},
2158                 {"1104", "IrMCSync Service"},
2159                 {"1105", "OBEX Object Push Service"},
2160                 {"1106", "OBEX File Transfer Service"},
2161                 {"1107", "IrMC Sync Command Service"},
2162                 {"1108", "Headset Service"},
2163                 {"1109", "Cordless Telephony Service"},
2164                 {"110A", "Audio Source Service"},
2165                 {"110B", "Audio Sink Service"},
2166                 {"110C", "AV Remote Control Target Service"},
2167                 {"110D", "Advanced Audio Distribution Profile"},
2168                 {"110E", "AV Remote Control Service"},
2169                 {"110F", "Video Conferencing Service"},
2170                 {"1110", "Intercom Service"},
2171                 {"1111", "Fax Service"},
2172                 {"1112", "Headset Audio Gateway Service"},
2173                 {"1113", "WAP Service"},
2174                 {"1114", "WAP Client Service"},
2175                 {"1115", "PANU Service"},
2176                 {"1116", "NAP Service"},
2177                 {"1117", "GN Service"},
2178                 {"1118", "Direct Printing Service"},
2179                 {"1119", "Reference Printing Service"},
2180                 {"111A", "Basic Imaging Profile"},
2181                 {"111B", "Imaging Responder Service"},
2182                 {"111C", "Imaging Automatic Archive Service"},
2183                 {"111D", "Imaging Reference Objects Service"},
2184                 {"111E", "Handsfree Service"},
2185                 {"111F", "Handsfree Audio Gateway Service"},
2186                 {"1120", "Direct Printing Reference Objects Service"},
2187                 {"1121", "Reflected UI Service"},
2188                 {"1122", "Basic Printing Profile"},
2189                 {"1123", "Printing Status Service"},
2190                 {"1124", "Human Interface Device Service"},
2191                 {"1125", "Hardcopy Cable Replacement Profile"},
2192                 {"1126", "HCR Print Service"},
2193                 {"1127", "HCR Scan Service"},
2194                 {"112D", "SIM Access Service"},
2195                 {"112E", "Phonebook Access PCE Service"},
2196                 {"112F", "Phonebook Access PSE Service"},
2197                 {"1130", "Phonebook Access Profile"},
2198                 {"1132", "Message Access Server Service"},
2199                 {"1133", "Message Notification Server Service"},
2200                 {"1134", "Message Access Profile"},
2201                 {"1200", "PnP Information Service"},
2202                 {"1201", "Generic Networking Service"},
2203                 {"1202", "Generic File Transfer Service"},
2204                 {"1203", "Generic Audio Service"},
2205                 {"1204", "Generic Telephony Service"},
2206                 {"1205", "UPnP Service"},
2207                 {"1206", "UPnP Ip Service"},
2208                 {"1303", "Video Source Service"},
2209                 {"1304", "Video Sink Service"},
2210                 {"1305", "Video Distribution Profile"},
2211                 {"1400", "Health Device Profile"},
2212                 {"1401", "HDP Source Service"},
2213                 {"1402", "HDP Sink Service"},
2214
2215                 /* GATT Services */
2216                 {"1800", "Generic Access"},
2217                 {"1801", "Generic Attribute"},
2218                 {"1802", "Immediate Alert"},
2219                 {"1803", "Link Loss"},
2220                 {"1804", "Tx Power"},
2221                 {"1805", "Current Time Service"},
2222                 {"1806", "Reference Time Update Service"},
2223                 {"1807", "Next DST Change Service"},
2224                 {"1808", "Glucose"},
2225                 {"1809", "Health Thermometer"},
2226                 {"180A", "Device Information"},
2227                 {"180D", "Heart Rate"},
2228                 {"180F", "Battery Service"},
2229                 {"1810", "Blood Pressure"},
2230                 {"1811", "Alert Notification Service"},
2231                 {"1812", "Human Interface Device"},
2232                 {"1813", "Scan Parameters"},
2233                 {"1814", "Running Speed and Cadence"},
2234                 {"1815", "Automation IO"},
2235                 {"1816", "Cycling Speed and Cadence"},
2236                 {"1818", "Cycling Power"},
2237                 {"1819", "Location and Navigation"},
2238                 {"181A", "Environmental Sensing"},
2239                 {"181B", "Body Composition"},
2240                 {"181C", "User Data"},
2241                 {"181D", "Weight Scale"},
2242                 {"181E", "Bond Management"},
2243                 {"181F", "Continuous Glucose Monitoring"},
2244                 {"1820", "Internet Protocol Support Service"},
2245                 {"1821", "Indoor Positioning"},
2246                 {"1822", "Pulse Oximeter Service"},
2247                 {"1823", "HTTP Proxy"},
2248                 {"1824", "Transport Discovery"},
2249                 {"1825", "Object Transfer Service"},
2250                 {"1826", "Fitness Machine"},
2251                 {"1827", "Mesh Provisioning Service"},
2252                 {"1828", "Mesh Proxy Service"},
2253                 {"1829", "Reconnection Configuration"},
2254                 {"183A", "Insulin Delivery"},
2255
2256                 /* GATT Declarations */
2257                 {"2800", "Primary Service Declaration"},
2258                 {"2801", "Secondary Service Declaration"},
2259                 {"2802", "Include Declaration"},
2260                 {"2803", "Characteristic Declaration"},
2261
2262                 /* GATT Descriptors */
2263                 {"2900", "Characteristic Extended Properties"},
2264                 {"2901", "Characteristic User Description"},
2265                 {"2902", "Client Characteristic Configuration"},
2266                 {"2903", "Server Characteristic Configuration"},
2267                 {"2904", "Characteristic Format"},
2268                 {"2905", "Characteristic Aggregate Formate"},
2269                 {"2906", "Valid Range"},
2270                 {"2907", "External Report Reference"},
2271                 {"2908", "Report Reference"},
2272
2273                 /* GATT Characteristics */
2274                 {"2A00", "Device Name"},
2275                 {"2A01", "Appearance"},
2276                 {"2A02", "Peripheral Privacy Flag"},
2277                 {"2A03", "Reconnection Address"},
2278                 {"2A04", "Peripheral Preferred Connection Parameters"},
2279                 {"2A05", "Service Changed"},
2280                 {"2A06", "Alert Level"},
2281                 {"2A07", "Tx Power Level"},
2282                 {"2A08", "Date Time"},
2283                 {"2A09", "Day of Week"},
2284                 {"2A0A", "Day Date Time"},
2285                 {"2A11", "Time with DST"},
2286                 {"2A12", "Time Accuracy"},
2287                 {"2A13", "Time Source"},
2288                 {"2A14", "Reference Time Information"},
2289                 {"2A16", "Time Update Control Point"},
2290                 {"2A17", "Time Update State"},
2291                 {"2A18", "Glucose Measurement"},
2292                 {"2A19", "Battery Level"},
2293                 {"2A1C", "Temperature Measurement"},
2294                 {"2A1D", "Temperature Type"},
2295                 {"2A1E", "Intermediate Temperature"},
2296                 {"2A21", "Measurement Interval"},
2297                 {"2A23", "System ID"},
2298                 {"2A24", "Model Number String"},
2299                 {"2A25", "Serial Number String"},
2300                 {"2A26", "Firmware Revision String"},
2301                 {"2A27", "Hardware Revision String"},
2302                 {"2A28", "Software Revision String"},
2303                 {"2A29", "Manufacturer Name String"},
2304                 {"2A2A", "IEEE 11073-20601 Regulatory Certification Data List"},
2305                 {"2A2B", "Current Time"},
2306                 {"2A34", "Glucose Measurement Context"},
2307                 {"2A35", "Blood Pressure Measurement"},
2308                 {"2A37", "Heart Rate Measurement"},
2309                 {"2A38", "Body Sensor Location"},
2310                 {"2A39", "Heart Rate Control Point"},
2311                 {"2A3F", "Alert Status"},
2312                 {"2A46", "New Alert"},
2313                 {"2A49", "Blood Pressure Feature"},
2314                 {"2A4A", "HID Information"},
2315                 {"2A4C", "HID Control Point"},
2316                 {"2A50", "PnP ID"},
2317                 {"2A51", "Glucose Feature"},
2318                 {"2A52", "Record Access Control Point"},
2319                 {"2A53", "RSC Measurement"},
2320                 {"2A54", "RSC Feature"},
2321                 {"2A55", "SC Control Point"},
2322                 {"2A56", "Digital"},
2323                 {"2A58", "Analog"},
2324                 {"2A5A", "Aggregate"},
2325                 {"2A5B", "CSC Measurement"},
2326                 {"2A5C", "CSC Feature"},
2327                 {"2A5D", "Sensor Location"},
2328                 {"2A63", "Cycling Power Measurement"},
2329                 {"2A64", "Cycling Power Vector"},
2330                 {"2A65", "Cycling Power Feature"},
2331                 {"2A66", "Cycling Power Control Point"},
2332                 {"2A67", "Location and Speed"},
2333                 {"2A68", "Navigation"},
2334                 {"2A6D", "Pressure"},
2335                 {"2A6E", "Temperature"},
2336                 {"2A8E", "Height"},
2337                 {"2A90", "Last Name"},
2338                 {"2A91", "Maximum Recommended Heart Rate"},
2339                 {"2A92", "Resting Heart Rate"},
2340                 {"2A98", "Weight"},
2341                 {"2A9B", "Body Composition Feature"},
2342                 {"2A9C", "Body Composition Measurement"},
2343                 {"2A9D", "Weight Measurement"},
2344                 {"2AA2", "Language"},
2345                 {"2AA4", "Bond Management Control Point"},
2346                 {"2AA5", "Bond Management Features"},
2347                 {"2AA6", "Central Address Resolution"},
2348                 {"2AAD", "Indoor Positioning Configuration"},
2349                 {"2AB5", "Location Name"},
2350                 {"2AB6", "URI"},
2351                 {"2ABC", "TDS Control Point"},
2352                 {"2AC9", "Resolvable Private Address Only"},
2353                 {"2ACC", "Fitness Machine Feature"},
2354                 {"2ACE", "Cross Trainer Data"},
2355                 {"2AD3", "Training Status"},
2356                 {"2AD7", "Supported Heart Rate Range"},
2357                 {"2AD9", "Fitness Machine Control Point"},
2358                 {"2ADA", "Fitness Machine Status"},
2359                 {"2B1D", "RC Feature"},
2360                 {"2B1E", "RC Settings"},
2361                 {"2B1F", "Reconnection Configuration Control Point"},
2362         };
2363         static uuid_name_s long_uuid_name[LONG_UUID_COUNT] = {
2364                 // List should be sorted by UUID
2365                 /* Custom uuids */
2366                 {"1AB7C24D-185A-45B9-90D4-F7AB1A71949A", "Samsung Health Service"},
2367                 {"22EAC6E9-24D6-4BB5-BE44-B36ACE7C7BFB", "Data Source"},
2368                 {"2F7CABCE-808D-411F-9A0C-BB92BA96C102", "Entity Update"},
2369                 {"32D1955A-E5AA-4A96-9A49-08538DA8B8F6", "Samsung Gear Fit Manager Service"},
2370                 {"69D1D8F3-45E1-49A8-9821-9BBDFDAAD9D9", "Control Point"},
2371                 {"7905F431-B5CE-4E99-A40F-4B1E122D00D0", "Apple Notification Center Service"},
2372                 {"89D3502B-0F36-433A-8EF4-C502AD55F8DC", "Apple Media Service"},
2373                 {"9A3F68E0-86CE-11E5-A309-0002A5D5C51B", "Samsung Gear Manager Service"},
2374                 {"9B3C81D8-57B1-4A8A-B8DF-0E56F7CA51C2", "Remote Command"},
2375                 {"9FBF120D-6301-42D9-8C58-25E699A21DBD", "Notifications Source"},
2376                 {"A49EB41E-CB06-495C-9F4F-BB80A90CDF00", "Samsung Gear Manager Service"},
2377                 {"ADE3D529-C784-4F63-A987-EB69F70EE816", "IoT OIC Service"},
2378                 {"C2051EE0-804D-4D50-A12C-15E243852100", "Notifications Source"},
2379                 {"C2F2CC0F-C085-4DD4-BE5A-ACA3074BBC72", "Control Point"},
2380                 {"C6B2F38C-23AB-46D8-A6AB-A3A870BBD5D7", "Entity Attribute"},
2381                 {"CECE518B-28D7-4171-92D5-76A1E249A3B9", "Notifications Source"},
2382                 {"FE53FF98-B259-4337-B56A-0EC9F82C6BAD", "Control Point"},
2383         };
2384         const uuid_name_s *uuid_name = short_uuid_name;
2385         int start = 0;
2386         int end = SHORT_UUID_COUNT - 1;
2387         int p;
2388         int ret;
2389
2390         if (!uuid || !name)
2391                 return BLUETOOTH_ERROR_INVALID_PARAM;
2392         if (strlen(uuid) == 36) {
2393                 if (!g_ascii_strncasecmp(uuid + 9, "0000-1000-8000-00805F9B34FB", 27))
2394                         offset = 4;
2395                 else {
2396                         offset = 0;
2397                         uuid_len = 36;
2398                         end = LONG_UUID_COUNT - 1;
2399                         uuid_name = long_uuid_name;
2400                 }
2401         } else if (strlen(uuid) >= 8)
2402                 offset = 4;
2403
2404         while (start <= end) {
2405                 p = start + (end - start) / 2;
2406                 ret = g_ascii_strncasecmp(uuid + offset, uuid_name[p].uuid, uuid_len);
2407                 if (ret == 0) {
2408                         *name = g_strdup(uuid_name[p].specification_name);
2409                         return BLUETOOTH_ERROR_NONE;
2410                 } else if (ret < 0)
2411                         end = p - 1;
2412                 else
2413                         start = p + 1;
2414         }
2415
2416         *name = g_strdup("Unknown");
2417         return BLUETOOTH_ERROR_NONE;
2418 }
2419
2420 int _bt_get_error_value_from_message(const char *error_message)
2421 {
2422         if (error_message == NULL) {
2423                 BT_ERR("Error message NULL");
2424                 return BLUETOOTH_ERROR_INTERNAL;
2425         }
2426
2427         BT_ERR("Error message = %s", error_message);
2428
2429         if (g_strrstr(error_message, BT_ERROR_OPERATION_NOT_SUPPORTED))
2430                 return BLUETOOTH_ERROR_NOT_IN_OPERATION;
2431         else if (g_strrstr(error_message, BT_ERROR_OPERATION_NOT_ALLOWED))
2432                 return BLUETOOTH_ERROR_PERMISSION_DEINED;
2433         else if (g_strrstr(error_message, BT_ERROR_ACCESS_DENIED))
2434                 return BLUETOOTH_ERROR_PERMISSION_DEINED;
2435         else
2436                 return BLUETOOTH_ERROR_INTERNAL;
2437 }
2438
2439 BT_EXPORT_API int bluetooth_is_supported(void)
2440 {
2441         int is_supported = 0;
2442         int len = 0;
2443         int fd = -1;
2444         rfkill_event event;
2445
2446         fd = open(RFKILL_NODE, O_RDONLY);
2447         if (fd < 0) {
2448                 BT_ERR("Fail to open RFKILL node");
2449                 return BLUETOOTH_ERROR_INTERNAL;
2450         }
2451
2452         if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0) {
2453                 BT_ERR("Fail to set RFKILL node to non-blocking");
2454                 close(fd);
2455                 return BLUETOOTH_ERROR_INTERNAL;
2456         }
2457
2458         while (1) {
2459                 len = read(fd, &event, sizeof(event));
2460                 if (len < 0) {
2461                         BT_ERR("Fail to read events");
2462                         break;
2463                 }
2464
2465                 if (len != RFKILL_EVENT_SIZE) {
2466                         BT_ERR("The size is wrong\n");
2467                         continue;
2468                 }
2469
2470                 if (event.type == RFKILL_TYPE_BLUETOOTH) {
2471                         is_supported = 1;
2472                         break;
2473                 }
2474         }
2475
2476         close(fd);
2477
2478         BT_DBG("supported: %d", is_supported);
2479
2480         return is_supported;
2481 }
2482
2483 BT_EXPORT_API int bluetooth_register_callback(bluetooth_cb_func_ptr callback_ptr, void *user_data)
2484 {
2485         int ret;
2486
2487         _bt_get_system_private_conn();
2488
2489         ret = _bt_init_event_handler();
2490         if (ret != BLUETOOTH_ERROR_NONE &&
2491              ret != BLUETOOTH_ERROR_ALREADY_INITIALIZED) {
2492                 BT_ERR("Fail to init the event handler");
2493                 return ret;
2494         }
2495
2496
2497         _bt_set_user_data(BT_COMMON, (void *)callback_ptr, user_data);
2498
2499         /* Register All events */
2500         if (ret != BLUETOOTH_ERROR_ALREADY_INITIALIZED) {
2501                 ret = _bt_register_event(BT_ADAPTER_EVENT, (void *)callback_ptr, user_data);
2502                 if (ret != BLUETOOTH_ERROR_NONE)
2503                         goto fail;
2504                 ret = _bt_register_event(BT_DEVICE_EVENT, (void *)callback_ptr, user_data);
2505                 if (ret != BLUETOOTH_ERROR_NONE)
2506                         goto fail;
2507                 ret = _bt_register_event(BT_NETWORK_EVENT, (void *)callback_ptr, user_data);
2508                 if (ret != BLUETOOTH_ERROR_NONE)
2509                         goto fail;
2510                 ret = _bt_register_event(BT_RFCOMM_CLIENT_EVENT, (void *)callback_ptr, user_data);
2511                 if (ret != BLUETOOTH_ERROR_NONE)
2512                         goto fail;
2513                 ret = _bt_register_event(BT_RFCOMM_SERVER_EVENT, (void *)callback_ptr, user_data);
2514                 if (ret != BLUETOOTH_ERROR_NONE)
2515                         goto fail;
2516
2517 #ifdef GATT_NO_RELAY
2518                 ret = _bt_register_event(BT_GATT_BLUEZ_EVENT, (void *)callback_ptr, user_data);
2519                 if (ret != BLUETOOTH_ERROR_NONE)
2520                         goto fail;
2521 #endif
2522                 ret = _bt_register_event(BT_TDS_EVENT, (void *)callback_ptr, user_data);
2523                 if (ret != BLUETOOTH_ERROR_NONE)
2524                         goto fail;
2525                 ret = _bt_register_event(BT_OTP_EVENT, (void *)callback_ptr, user_data);
2526                 if (ret != BLUETOOTH_ERROR_NONE)
2527                         goto fail;
2528                 ret = _bt_register_event(BT_HDP_EVENT, (void *)callback_ptr, user_data);
2529                 if (ret != BLUETOOTH_ERROR_NONE)
2530                         goto fail;
2531         }
2532
2533         _bt_register_name_owner_changed();
2534
2535         return BLUETOOTH_ERROR_NONE;
2536 fail:
2537         BT_ERR("Fail to do _bt_register_event()");
2538         bluetooth_unregister_callback();
2539         return ret;
2540 }
2541
2542 BT_EXPORT_API int bluetooth_unregister_callback(void)
2543 {
2544         int ret;
2545
2546         ret = _bt_deinit_event_handler();
2547         if (ret != BLUETOOTH_ERROR_NONE)
2548                 BT_ERR("Fail to deinit the event handler");
2549
2550         _bt_unregister_name_owner_changed();
2551
2552         _bt_set_user_data(BT_COMMON, NULL, NULL);
2553
2554         _bt_set_obex_server_id(BT_NO_SERVER);
2555
2556         _bt_reset_battery_monitor_info();
2557
2558         _bt_gdbus_deinit_proxys();
2559
2560         if (profile_gproxy) {
2561                 g_object_unref(profile_gproxy);
2562                 profile_gproxy = NULL;
2563         }
2564
2565         if (system_private_conn) {
2566                 g_object_unref(system_private_conn);
2567                 system_private_conn = NULL;
2568         }
2569
2570         return BLUETOOTH_ERROR_NONE;
2571 }