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