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