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