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