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