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