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