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