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