4 * Copyright (c) 2000 - 2011 Samsung Electronics Co., Ltd. All rights reserved.
6 * Contact: Hocheol Seo <hocheol.seo@samsung.com>
7 * Girishashok Joshi <girish.joshi@samsung.com>
8 * Chanyeol Park <chanyeol.park@samsung.com>
10 * Licensed under the Apache License, Version 2.0 (the "License");
11 * you may not use this file except in compliance with the License.
12 * You may obtain a copy of the License at
14 * http://www.apache.org/licenses/LICENSE-2.0
16 * Unless required by applicable law or agreed to in writing, software
17 * distributed under the License is distributed on an "AS IS" BASIS,
18 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
19 * See the License for the specific language governing permissions and
20 * limitations under the License.
26 #include <dbus/dbus-glib-lowlevel.h>
27 #include <dbus/dbus-glib.h>
28 #include <dbus/dbus.h>
34 #include <net_connection.h>
36 #include "bluetooth-api.h"
37 #include "bt-service-common.h"
38 #include "bt-service-agent.h"
40 static DBusGConnection *system_conn;
41 static DBusGConnection *session_conn;
42 static DBusGProxy *manager_proxy;
43 static DBusGProxy *adapter_proxy;
44 static void *net_conn;
46 static DBusGProxy *adapter_properties_proxy;
48 static DBusGProxy *__bt_init_manager_proxy(void)
54 if (system_conn == NULL) {
55 system_conn = dbus_g_bus_get(DBUS_BUS_SYSTEM, NULL);
56 retv_if(system_conn == NULL, NULL);
59 proxy = dbus_g_proxy_new_for_name(system_conn, BT_BLUEZ_NAME,
60 BT_MANAGER_PATH, BT_MANAGER_INTERFACE);
62 retv_if(proxy == NULL, NULL);
64 manager_proxy = proxy;
69 static DBusGProxy *__bt_init_adapter_proxy(void)
71 DBusGProxy *manager_proxy;
74 if (system_conn == NULL) {
75 system_conn = dbus_g_bus_get(DBUS_BUS_SYSTEM, NULL);
76 retv_if(system_conn == NULL, NULL);
79 manager_proxy = _bt_get_manager_proxy();
80 retv_if(manager_proxy == NULL, NULL);
82 proxy = dbus_g_proxy_new_for_name(system_conn, BT_BLUEZ_NAME,
83 BT_BLUEZ_HCI_PATH, BT_ADAPTER_INTERFACE);
85 retv_if(proxy == NULL, NULL);
87 adapter_proxy = proxy;
92 static DBusGProxy *__bt_init_adapter_properties_proxy(void)
94 DBusGProxy *manager_proxy;
99 if (system_conn == NULL) {
100 system_conn = dbus_g_bus_get(DBUS_BUS_SYSTEM, NULL);
101 retv_if(system_conn == NULL, NULL);
104 manager_proxy = _bt_get_manager_proxy();
105 retv_if(manager_proxy == NULL, NULL);
107 proxy = dbus_g_proxy_new_for_name(system_conn, BT_BLUEZ_NAME,
108 BT_BLUEZ_HCI_PATH, BT_PROPERTIES_INTERFACE);
110 retv_if(proxy == NULL, NULL);
112 adapter_properties_proxy = proxy;
117 DBusGConnection *__bt_init_system_gconn(void)
121 if (system_conn == NULL)
122 system_conn = dbus_g_bus_get(DBUS_BUS_SYSTEM, NULL);
127 DBusGConnection *__bt_init_session_conn(void)
129 if (session_conn == NULL)
130 session_conn = dbus_g_bus_get(DBUS_BUS_SESSION, NULL);
135 DBusGConnection *_bt_get_session_gconn(void)
137 return (session_conn) ? session_conn : __bt_init_session_conn();
140 DBusGConnection *_bt_get_system_gconn(void)
142 return (system_conn) ? system_conn : __bt_init_system_gconn();
145 DBusConnection *_bt_get_system_conn(void)
147 DBusGConnection *g_conn;
149 if (system_conn == NULL) {
150 g_conn = __bt_init_system_gconn();
152 g_conn = system_conn;
155 retv_if(g_conn == NULL, NULL);
157 return dbus_g_connection_get_connection(g_conn);
160 DBusGProxy *_bt_get_manager_proxy(void)
163 const char *path = dbus_g_proxy_get_path(manager_proxy);
165 BT_ERR("Already proxy released hence creating new proxy");
166 return __bt_init_manager_proxy();
168 return manager_proxy;
170 return __bt_init_manager_proxy();
173 static void *__bt_init_net_conn(void)
176 connection_h connection = NULL;
178 if (net_conn == NULL) {
179 result = connection_create(&connection);
181 if (result != CONNECTION_ERROR_NONE ||
182 connection == NULL) {
183 BT_DBG("connection_create() failed: %d", result);
187 net_conn = connection;
192 void *_bt_get_net_conn(void)
194 return (net_conn) ? net_conn : __bt_init_net_conn();
197 gboolean _bt_get_adapter_power(void)
199 DBusGProxy *proxy = NULL;
201 GValue powered_v = { 0 };
204 BT_DBG("_bt_check_adapter 4");
205 proxy = _bt_get_adapter_properties_proxy();
206 retv_if(proxy == NULL, FALSE);
208 if (!dbus_g_proxy_call(proxy, "Get", &err,
209 G_TYPE_STRING, BT_ADAPTER_INTERFACE,
210 G_TYPE_STRING, "Powered",
212 G_TYPE_VALUE, &powered_v,
215 BT_ERR("Getting property failed \n: [%s]\n", err->message);
221 powered = (gboolean)g_value_get_boolean(&powered_v);
223 BT_DBG("powered = %d", powered);
228 DBusGProxy *_bt_get_adapter_proxy(void)
231 const char *path = dbus_g_proxy_get_path(adapter_proxy);
233 BT_ERR("Already proxy released hence creating new proxy");
234 return __bt_init_adapter_proxy();
237 return adapter_proxy;
239 return __bt_init_adapter_proxy();
243 DBusGProxy *_bt_get_adapter_properties_proxy(void)
245 return (adapter_properties_proxy) ? adapter_properties_proxy :
246 __bt_init_adapter_properties_proxy();
249 static char *__bt_extract_adapter_path(DBusMessageIter *msg_iter)
251 char *object_path = NULL;
252 DBusMessageIter value_iter;
254 /* Parse the signature: oa{sa{sv}}} */
255 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
256 DBUS_TYPE_OBJECT_PATH, NULL);
258 dbus_message_iter_get_basic(msg_iter, &object_path);
259 retv_if(object_path == NULL, NULL);
261 /* object array (oa) */
262 retv_if(dbus_message_iter_next(msg_iter) == FALSE, NULL);
263 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
264 DBUS_TYPE_ARRAY, NULL);
266 dbus_message_iter_recurse(msg_iter, &value_iter);
268 /* string array (sa) */
269 while (dbus_message_iter_get_arg_type(&value_iter) ==
270 DBUS_TYPE_DICT_ENTRY) {
271 char *interface_name = NULL;
272 DBusMessageIter interface_iter;
274 dbus_message_iter_recurse(&value_iter, &interface_iter);
276 retv_if(dbus_message_iter_get_arg_type(&interface_iter) !=
277 DBUS_TYPE_STRING, NULL);
279 dbus_message_iter_get_basic(&interface_iter, &interface_name);
281 if (g_strcmp0(interface_name, "org.bluez.Adapter1") == 0) {
282 /* Tizen don't allow the multi-adapter */
283 BT_DBG("Found an adapter: %s", object_path);
284 return g_strdup(object_path);
287 dbus_message_iter_next(&value_iter);
290 BT_DBG("There is no adapter");
295 char *_bt_get_adapter_path(void)
299 DBusMessageIter reply_iter;
300 DBusMessageIter value_iter;
302 DBusConnection *conn;
303 char *adapter_path = NULL;
305 conn = _bt_get_system_conn();
306 retv_if(conn == NULL, NULL);
308 msg = dbus_message_new_method_call(BT_BLUEZ_NAME, BT_MANAGER_PATH,
309 BT_MANAGER_INTERFACE,
310 "GetManagedObjects");
312 retv_if(msg == NULL, NULL);
314 /* Synchronous call */
315 dbus_error_init(&err);
316 reply = dbus_connection_send_with_reply_and_block(
319 dbus_message_unref(msg);
322 BT_ERR("Can't get managed objects");
324 if (dbus_error_is_set(&err)) {
325 BT_ERR("%s", err.message);
326 dbus_error_free(&err);
331 if (dbus_message_iter_init(reply, &reply_iter) == FALSE) {
332 BT_ERR("Fail to iterate the reply");
333 dbus_message_unref(reply);
337 dbus_message_iter_recurse(&reply_iter, &value_iter);
339 /* signature of GetManagedObjects: a{oa{sa{sv}}} */
340 while (dbus_message_iter_get_arg_type(&value_iter) ==
341 DBUS_TYPE_DICT_ENTRY) {
342 DBusMessageIter msg_iter;
344 dbus_message_iter_recurse(&value_iter, &msg_iter);
346 adapter_path = __bt_extract_adapter_path(&msg_iter);
347 if (adapter_path != NULL) {
348 BT_DBG("Found the adapter path");
352 dbus_message_iter_next(&value_iter);
354 dbus_message_unref(reply);
359 void _bt_deinit_bluez_proxy(void)
362 g_object_unref(manager_proxy);
363 manager_proxy = NULL;
367 g_object_unref(adapter_proxy);
368 adapter_proxy = NULL;
370 if (adapter_properties_proxy) {
371 g_object_unref(adapter_properties_proxy);
372 adapter_properties_proxy = NULL;
376 void _bt_deinit_proxys(void)
379 _bt_deinit_bluez_proxy();
382 dbus_g_connection_unref(system_conn);
387 dbus_g_connection_unref(session_conn);
392 ret = connection_destroy(net_conn);
395 BT_ERR("connection_destroy failed : %d", ret);
399 void _bt_convert_device_path_to_address(const char *device_path,
400 char *device_address)
402 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
405 ret_if(device_path == NULL);
406 ret_if(device_address == NULL);
408 dev_addr = strstr(device_path, "dev_");
409 if (dev_addr != NULL) {
412 g_strlcpy(address, dev_addr, sizeof(address));
414 while ((pos = strchr(address, '_')) != NULL) {
418 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
423 void _bt_convert_addr_string_to_type(unsigned char *addr,
429 ret_if(address == NULL);
430 ret_if(addr == NULL);
432 for (i = 0; i < BT_ADDRESS_LENGTH_MAX; i++) {
433 addr[i] = strtol(address, &ptr, 16);
434 if (ptr[0] != '\0') {
443 void _bt_convert_addr_type_to_string(char *address,
446 ret_if(address == NULL);
447 ret_if(addr == NULL);
449 snprintf(address, BT_ADDRESS_STRING_SIZE,
450 "%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X",
451 addr[0], addr[1], addr[2],
452 addr[3], addr[4], addr[5]);
455 void _bt_print_device_address_t(const bluetooth_device_address_t *addr)
457 BT_DBG("%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X\n", addr->addr[0], addr->addr[1], addr->addr[2],
458 addr->addr[3], addr->addr[4], addr->addr[5]);
461 void _bt_divide_device_class(bluetooth_device_class_t *device_class,
464 ret_if(device_class == NULL);
466 device_class->major_class = (unsigned short)(cod & 0x00001F00) >> 8;
467 device_class->minor_class = (unsigned short)((cod & 0x000000FC));
468 device_class->service_class = (unsigned long)((cod & 0x00FF0000));
470 if (cod & 0x002000) {
471 device_class->service_class |=
472 BLUETOOTH_DEVICE_SERVICE_CLASS_LIMITED_DISCOVERABLE_MODE;
476 void _bt_free_device_info(bt_remote_dev_info_t *dev_info)
480 ret_if(dev_info == NULL);
482 g_free(dev_info->address);
483 g_free(dev_info->name);
484 g_free(dev_info->manufacturer_data);
486 if (dev_info->uuids) {
487 for (i = 0; dev_info->uuids[i] != NULL; i++)
488 g_free(dev_info->uuids[i]);
490 g_free(dev_info->uuids);
496 void _bt_free_le_device_info(bt_remote_le_dev_info_t *le_dev_info)
498 ret_if(le_dev_info == NULL);
500 g_free(le_dev_info->adv_data);
504 int _bt_copy_utf8_string(char *dest, const char *src, unsigned int length)
511 if (dest == NULL || src == NULL)
512 return BLUETOOTH_ERROR_INVALID_PARAM;
514 BT_DBG("+src : %s", src);
515 BT_DBG("+dest : %s", dest);
518 while (*p != '\0' && i < length) {
519 next = g_utf8_next_char(p);
522 while (count > 0 && ((i + count) < length)) {
529 return BLUETOOTH_ERROR_NONE;
532 gboolean _bt_utf8_validate(char *name)
536 glong items_written = 0;
538 if (FALSE == g_utf8_validate(name, -1, NULL))
541 u16 = g_utf8_to_utf16(name, -1, NULL, &items_written, NULL);
547 if (items_written != g_utf8_strlen(name, -1))
554 int _bt_register_osp_server_in_agent(int type, char *uuid, char *path, int fd)
556 if (!_bt_agent_register_osp_server( type, uuid, path, fd))
557 return BLUETOOTH_ERROR_INTERNAL;
559 return BLUETOOTH_ERROR_NONE;
562 int _bt_unregister_osp_server_in_agent(int type, char *uuid)
564 if (!_bt_agent_unregister_osp_server( type, uuid))
565 return BLUETOOTH_ERROR_INTERNAL;
567 return BLUETOOTH_ERROR_NONE;
570 int _bt_set_socket_non_blocking(int socket_fd)
572 /* Set Nonblocking */
575 arg = fcntl(socket_fd, F_GETFL);
580 if (arg & O_NONBLOCK) {
581 BT_ERR("Already Non-blocking \n");
586 if (fcntl(socket_fd, F_SETFL, arg) < 0)
589 return BLUETOOTH_ERROR_NONE;
592 int _bt_set_non_blocking_tty(int sk)
594 struct termios ti = {0,};
597 err = _bt_set_socket_non_blocking(sk);
600 BT_ERR("Error in set non blocking!\n");
604 tcflush(sk, TCIOFLUSH);
606 /* Switch tty to RAW mode */
608 tcsetattr(sk, TCSANOW, &ti);
610 return BLUETOOTH_ERROR_NONE;
613 static char *__bt_extract_device_path(DBusMessageIter *msg_iter, char *address)
615 char *object_path = NULL;
616 char device_address[BT_ADDRESS_STRING_SIZE] = { 0 };
618 /* Parse the signature: oa{sa{sv}}} */
619 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
620 DBUS_TYPE_OBJECT_PATH, NULL);
622 dbus_message_iter_get_basic(msg_iter, &object_path);
623 retv_if(object_path == NULL, NULL);
625 _bt_convert_device_path_to_address(object_path, device_address);
627 if (g_strcmp0(address, device_address) == 0) {
628 return g_strdup(object_path);
634 char *_bt_get_device_object_path(char *address)
638 DBusMessageIter reply_iter;
639 DBusMessageIter value_iter;
641 DBusConnection *conn;
642 char *object_path = NULL;
644 conn = _bt_get_system_conn();
645 retv_if(conn == NULL, NULL);
647 msg = dbus_message_new_method_call(BT_BLUEZ_NAME, BT_MANAGER_PATH,
648 BT_MANAGER_INTERFACE,
649 "GetManagedObjects");
651 retv_if(msg == NULL, NULL);
653 /* Synchronous call */
654 dbus_error_init(&err);
655 reply = dbus_connection_send_with_reply_and_block(
658 dbus_message_unref(msg);
661 BT_ERR("Can't get managed objects");
663 if (dbus_error_is_set(&err)) {
664 BT_ERR("%s", err.message);
665 dbus_error_free(&err);
670 if (dbus_message_iter_init(reply, &reply_iter) == FALSE) {
671 BT_ERR("Fail to iterate the reply");
672 dbus_message_unref(reply);
676 dbus_message_iter_recurse(&reply_iter, &value_iter);
678 /* signature of GetManagedObjects: a{oa{sa{sv}}} */
679 while (dbus_message_iter_get_arg_type(&value_iter) ==
680 DBUS_TYPE_DICT_ENTRY) {
681 DBusMessageIter msg_iter;
683 dbus_message_iter_recurse(&value_iter, &msg_iter);
685 object_path = __bt_extract_device_path(&msg_iter, address);
686 if (object_path != NULL) {
687 BT_DBG("Found the device path");
691 dbus_message_iter_next(&value_iter);
693 dbus_message_unref(reply);
698 char *_bt_get_profile_uuid128(bt_profile_type_t profile_type)
700 switch(profile_type) {
701 case BT_PROFILE_CONN_RFCOMM:
702 return strdup(RFCOMM_UUID_STR);
703 case BT_PROFILE_CONN_A2DP:
704 return strdup(A2DP_SINK_UUID);
705 case BT_PROFILE_CONN_HSP:
706 return strdup(HFP_HS_UUID);
707 case BT_PROFILE_CONN_HID:
708 return strdup(HID_UUID);
709 case BT_PROFILE_CONN_NAP:
710 return strdup(NAP_UUID);
711 case BT_PROFILE_CONN_HFG:
712 return strdup(HFP_AG_UUID);
713 case BT_PROFILE_CONN_GATT:
714 case BT_PROFILE_CONN_ALL: /* NULL UUID will connect to both the audio profiles*/
720 char *_bt_convert_error_to_string(int error)
723 case BLUETOOTH_ERROR_CANCEL:
725 case BLUETOOTH_ERROR_INVALID_PARAM:
726 return "INVALID_PARAMETER";
727 case BLUETOOTH_ERROR_INVALID_DATA:
728 return "INVALID DATA";
729 case BLUETOOTH_ERROR_MEMORY_ALLOCATION:
730 case BLUETOOTH_ERROR_OUT_OF_MEMORY:
731 return "OUT_OF_MEMORY";
732 case BLUETOOTH_ERROR_TIMEOUT:
734 case BLUETOOTH_ERROR_NO_RESOURCES:
735 return "NO_RESOURCES";
736 case BLUETOOTH_ERROR_INTERNAL:
738 case BLUETOOTH_ERROR_NOT_SUPPORT:
739 return "NOT_SUPPORT";
740 case BLUETOOTH_ERROR_DEVICE_NOT_ENABLED:
741 return "NOT_ENABLED";
742 case BLUETOOTH_ERROR_DEVICE_ALREADY_ENABLED:
743 return "ALREADY_ENABLED";
744 case BLUETOOTH_ERROR_DEVICE_BUSY:
745 return "DEVICE_BUSY";
746 case BLUETOOTH_ERROR_ACCESS_DENIED:
747 return "ACCESS_DENIED";
748 case BLUETOOTH_ERROR_MAX_CLIENT:
750 case BLUETOOTH_ERROR_NOT_FOUND:
752 case BLUETOOTH_ERROR_SERVICE_SEARCH_ERROR:
753 return "SERVICE_SEARCH_ERROR";
754 case BLUETOOTH_ERROR_PARING_FAILED:
755 return "PARING_FAILED";
756 case BLUETOOTH_ERROR_NOT_PAIRED:
758 case BLUETOOTH_ERROR_SERVICE_NOT_FOUND:
759 return "SERVICE_NOT_FOUND";
760 case BLUETOOTH_ERROR_NOT_CONNECTED:
761 return "NOT_CONNECTED";
762 case BLUETOOTH_ERROR_ALREADY_CONNECT:
763 return "ALREADY_CONNECT";
764 case BLUETOOTH_ERROR_CONNECTION_BUSY:
765 return "CONNECTION_BUSY";
766 case BLUETOOTH_ERROR_CONNECTION_ERROR:
767 return "CONNECTION_ERROR";
768 case BLUETOOTH_ERROR_MAX_CONNECTION:
769 return "MAX_CONNECTION";
770 case BLUETOOTH_ERROR_NOT_IN_OPERATION:
771 return "NOT_IN_OPERATION";
772 case BLUETOOTH_ERROR_CANCEL_BY_USER:
773 return "CANCEL_BY_USER";
774 case BLUETOOTH_ERROR_REGISTRATION_FAILED:
775 return "REGISTRATION_FAILED";
776 case BLUETOOTH_ERROR_IN_PROGRESS:
777 return "IN_PROGRESS";
778 case BLUETOOTH_ERROR_AUTHENTICATION_FAILED:
779 return "AUTHENTICATION_FAILED";
780 case BLUETOOTH_ERROR_HOST_DOWN:
782 case BLUETOOTH_ERROR_END_OF_DEVICE_LIST:
783 return "END_OF_DEVICE_LIST";
784 case BLUETOOTH_ERROR_AGENT_ALREADY_EXIST:
785 return "AGENT_ALREADY_EXIST";
786 case BLUETOOTH_ERROR_AGENT_DOES_NOT_EXIST:
787 return "AGENT_DOES_NOT_EXIST";
788 case BLUETOOTH_ERROR_ALREADY_INITIALIZED:
789 return "ALREADY_INITIALIZED";
790 case BLUETOOTH_ERROR_PERMISSION_DEINED:
791 return "PERMISSION_DEINED";
792 case BLUETOOTH_ERROR_ALREADY_DEACTIVATED:
793 return "ALREADY_DEACTIVATED";
794 case BLUETOOTH_ERROR_NOT_INITIALIZED:
795 return "NOT_INITIALIZED";
801 char * _bt_convert_disc_reason_to_string(int reason)
807 return "Connection terminated by local host";
809 return "Remote user terminated connection";
816 void _bt_logging_connection(gboolean connect, int addr_type)
818 static int le_conn = 0;
819 static int le_disc = 0;
820 static int edr_conn = 0;
821 static int edr_disc = 0;
835 BT_INFO("[PM] Number of LE conn: %d disc: %d, Number of BR/EDR conn: %d disc: %d",
836 le_conn, le_disc, edr_conn, edr_disc);