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.
27 #include <sys/types.h>
29 #include <sys/socket.h>
32 #include <gio/gunixfdlist.h>
35 #include "bluetooth-api.h"
36 #include "bluetooth-audio-api.h"
37 #include "bluetooth-hid-api.h"
38 #include "bluetooth-media-control.h"
39 #include "bt-internal-types.h"
41 #include "bt-common.h"
42 #include "bt-request-sender.h"
43 #include "bt-event-handler.h"
45 static bt_user_info_t user_info[BT_MAX_USER_INFO];
46 static DBusGConnection *system_conn = NULL;
47 static GDBusConnection *system_gdbus_conn = NULL;
51 static GDBusConnection *system_gconn = NULL;
53 GDBusConnection *_bt_gdbus_init_system_gconn(void)
57 if (!g_thread_supported()) {
65 if (system_gconn != NULL)
68 system_gconn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
71 BT_ERR("Unable to connect to dbus: %s", error->message);
72 g_clear_error(&error);
78 GDBusConnection *_bt_gdbus_get_system_gconn(void)
80 GDBusConnection *local_system_gconn = NULL;
83 if (system_gconn == NULL) {
84 system_gconn = _bt_gdbus_init_system_gconn();
85 } else if (g_dbus_connection_is_closed(system_gconn)){
87 local_system_gconn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
89 if (!local_system_gconn) {
90 BT_ERR("Unable to connect to dbus: %s", error->message);
91 g_clear_error(&error);
94 system_gconn = local_system_gconn;
100 void _bt_print_device_address_t(const bluetooth_device_address_t *addr)
102 BT_DBG("%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X\n", addr->addr[0], addr->addr[1], addr->addr[2],
103 addr->addr[3], addr->addr[4], addr->addr[5]);
106 void _bt_set_user_data(int type, void *callback, void *user_data)
108 user_info[type].cb = callback;
109 user_info[type].user_data = user_data;
112 bt_user_info_t *_bt_get_user_data(int type)
114 return &user_info[type];
117 void _bt_common_event_cb(int event, int result, void *param,
118 void *callback, void *user_data)
120 bluetooth_event_param_t bt_event = { 0, };
121 bt_event.event = event;
122 bt_event.result = result;
123 bt_event.param_data = param;
126 ((bluetooth_cb_func_ptr)callback)(bt_event.event, &bt_event,
130 void _bt_input_event_cb(int event, int result, void *param,
131 void *callback, void *user_data)
133 hid_event_param_t bt_event = { 0, };
134 bt_event.event = event;
135 bt_event.result = result;
136 bt_event.param_data = param;
139 ((hid_cb_func_ptr)callback)(bt_event.event, &bt_event,
143 void _bt_headset_event_cb(int event, int result, void *param,
144 void *callback, void *user_data)
146 bt_audio_event_param_t bt_event = { 0, };
147 bt_event.event = event;
148 bt_event.result = result;
149 bt_event.param_data = param;
152 ((bt_audio_func_ptr)callback)(bt_event.event, &bt_event,
156 void _bt_hf_event_cb(int event, int result, void *param,
157 void *callback, void *user_data)
159 bt_hf_event_param_t bt_event = { 0, };
160 bt_event.event = event;
161 bt_event.result = result;
162 bt_event.param_data = param;
165 ((bt_hf_func_ptr)callback)(bt_event.event, &bt_event,
170 void _bt_avrcp_event_cb(int event, int result, void *param,
171 void *callback, void *user_data)
173 media_event_param_t bt_event = { 0, };
174 bt_event.event = event;
175 bt_event.result = result;
176 bt_event.param_data = param;
179 ((media_cb_func_ptr)callback)(bt_event.event, &bt_event,
183 void _bt_divide_device_class(bluetooth_device_class_t *device_class,
186 ret_if(device_class == NULL);
188 device_class->major_class = (unsigned short)(cod & 0x00001F00) >> 8;
189 device_class->minor_class = (unsigned short)((cod & 0x000000FC));
190 device_class->service_class = (unsigned long)((cod & 0x00FF0000));
192 if (cod & 0x002000) {
193 device_class->service_class |=
194 BLUETOOTH_DEVICE_SERVICE_CLASS_LIMITED_DISCOVERABLE_MODE;
198 void _bt_convert_addr_string_to_type(unsigned char *addr,
204 ret_if(address == NULL);
205 ret_if(addr == NULL);
207 for (i = 0; i < BT_ADDRESS_LENGTH_MAX; i++) {
208 addr[i] = strtol(address, &ptr, 16);
209 if (ptr[0] != '\0') {
218 void _bt_convert_addr_type_to_string(char *address,
221 ret_if(address == NULL);
222 ret_if(addr == NULL);
224 g_snprintf(address, BT_ADDRESS_STRING_SIZE,
225 "%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X",
226 addr[0], addr[1], addr[2],
227 addr[3], addr[4], addr[5]);
230 int _bt_copy_utf8_string(char *dest, const char *src, unsigned int length)
237 if (dest == NULL || src == NULL)
238 return BLUETOOTH_ERROR_INVALID_PARAM;
241 while (*p != '\0' && i < length) {
242 next = g_utf8_next_char(p);
245 while (count > 0 && ((i + count) < length)) {
252 return BLUETOOTH_ERROR_NONE;
255 gboolean _bt_utf8_validate(char *name)
259 glong items_written = 0;
261 if (FALSE == g_utf8_validate(name, -1, NULL))
264 u16 = g_utf8_to_utf16(name, -1, NULL, &items_written, NULL);
270 if (items_written != g_utf8_strlen(name, -1))
278 static GDBusProxy *profile_gproxy;
279 static GDBusConnection *gconn;
280 static int latest_id = -1;
281 #define BT_RFCOMM_ID_MAX 245
282 static gboolean id_used[BT_RFCOMM_ID_MAX];
283 GDBusNodeInfo *new_conn_node;
285 static const gchar rfcomm_agent_xml[] =
287 " <interface name='org.bluez.Profile1'>"
288 " <method name='NewConnection'>"
289 " <arg type='o' name='object' direction='in'/>"
290 " <arg type='h' name='fd' direction='in'/>"
291 " <arg type='a{sv}' name='properties' direction='in'/>"
296 /* Remote socket address */
297 struct sockaddr_remote {
299 bluetooth_device_address_t remote_bdaddr;
300 unsigned char channel;
303 static void __get_remote_address(int fd, bluetooth_device_address_t *bdaddr)
305 struct sockaddr_remote address;
306 socklen_t address_len;
308 address_len = sizeof(address);
309 if (getpeername(fd, (struct sockaddr *) &address, &address_len) != 0) {
310 BT_DBG("getpeername failed");
314 memcpy(bdaddr->addr, address.remote_bdaddr.addr,
315 BLUETOOTH_ADDRESS_LENGTH);
318 static void __new_connection_method(GDBusConnection *connection,
320 const gchar *object_path,
321 const gchar *interface_name,
322 const gchar *method_name,
323 GVariant *parameters,
324 GDBusMethodInvocation *invocation,
327 BT_DBG("method %s", method_name);
328 if (g_strcmp0(method_name, "NewConnection") == 0) {
331 GUnixFDList *fd_list;
332 GVariantBuilder *properties;
335 bluetooth_device_address_t remote_addr, remote_addr1;
336 bt_new_connection_cb cb = user_data;
339 g_variant_get(parameters, "(oha{sv})", &obj_path, &index,
342 msg = g_dbus_method_invocation_get_message(invocation);
343 fd_list = g_dbus_message_get_unix_fd_list(msg);
344 if (fd_list == NULL) {
345 GQuark quark = g_quark_from_string("rfcomm-app");
346 GError *err = g_error_new(quark, 0, "No fd in message");
347 g_dbus_method_invocation_return_gerror(invocation, err);
353 fd = g_unix_fd_list_get(fd_list, index, NULL);
355 BT_ERR("Invalid fd return");
356 GQuark quark = g_quark_from_string("rfcomm-app");
357 GError *err = g_error_new(quark, 0, "Invalid FD return");
358 g_dbus_method_invocation_return_gerror(invocation, err);
363 __get_remote_address(fd, &remote_addr);
364 _bt_swap_addr(remote_addr1.addr, remote_addr.addr);
365 _bt_convert_addr_type_to_string(addr, remote_addr1.addr);
366 BT_INFO("fd: %d, address %s", fd, addr);
368 g_dbus_method_invocation_return_value(invocation, NULL);
371 cb(object_path, fd, &remote_addr1);
376 static const GDBusInterfaceVTable method_table = {
377 __new_connection_method,
382 void _bt_swap_addr(unsigned char *dst, const unsigned char *src)
386 for (i = 0; i < 6; i++)
390 int __rfcomm_assign_id(void)
394 BT_DBG("latest_id: %d", latest_id);
396 index = latest_id + 1;
398 if (index >= BT_RFCOMM_ID_MAX)
401 BT_DBG("index: %d", index);
403 while (id_used[index] == TRUE) {
404 if (index == latest_id) {
405 /* No available ID */
406 BT_ERR("All request ID is used");
412 if (index >= BT_RFCOMM_ID_MAX)
417 id_used[index] = TRUE;
419 BT_DBG("Assigned Id: %d", latest_id);
424 void __rfcomm_delete_id(int id)
426 ret_if(id >= BT_RFCOMM_ID_MAX);
431 /* Next server will use this ID */
435 static GDBusConnection *__get_gdbus_connection()
440 gconn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &err);
444 BT_ERR("Unable to connect to dbus: %s", err->message);
453 static GDBusProxy *__bt_gdbus_get_profile_proxy(void)
455 GDBusConnection *gconn;
459 return profile_gproxy;
461 gconn = __get_gdbus_connection();
465 profile_gproxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE,
468 "org.bluez.ProfileManager1",
471 BT_ERR("Unable to create proxy: %s", err->message);
475 return profile_gproxy;
478 void _bt_unregister_gdbus(int object_id)
480 GDBusConnection *gconn;
482 gconn = __get_gdbus_connection();
486 g_dbus_connection_unregister_object(gconn, object_id);
489 int _bt_register_new_conn(const char *path, bt_new_connection_cb cb)
491 GDBusConnection *gconn;
493 GError *error = NULL;
495 gconn = __get_gdbus_connection();
499 if (new_conn_node == NULL)
500 new_conn_node = _bt_get_gdbus_node(rfcomm_agent_xml);
502 if (new_conn_node == NULL)
505 id = g_dbus_connection_register_object(gconn, path,
506 new_conn_node->interfaces[0],
510 BT_ERR("Failed to register: %s", error->message);
515 BT_DBG("NEW CONNECTION ID %d", id);
521 int _bt_register_profile(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
523 GVariantBuilder *option_builder;
527 int result = BLUETOOTH_ERROR_NONE;
529 proxy = __bt_gdbus_get_profile_proxy();
531 BT_ERR("Getting profile proxy failed");
532 return BLUETOOTH_ERROR_INTERNAL;
535 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
536 if (info->authentication)
537 g_variant_builder_add(option_builder, "{sv}",
538 "RequireAuthentication",
539 g_variant_new_boolean(TRUE));
540 if (info->authorization)
541 g_variant_builder_add(option_builder, "{sv}",
542 "RequireAuthorization",
543 g_variant_new_boolean(TRUE));
545 g_variant_builder_add(option_builder, "{sv}",
547 g_variant_new_string(info->role));
549 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
550 * RFCOMM channels based on the availability when two services want
551 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
552 * two services use the same SPP RFCOMM channel. */
553 if (use_default_rfcomm)
554 g_variant_builder_add(option_builder, "{sv}",
556 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
558 g_variant_builder_add(option_builder, "{sv}",
560 g_variant_new_string(info->service));
563 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile",
564 g_variant_new("(osa{sv})", info->obj_path,
567 G_DBUS_CALL_FLAGS_NONE, -1,
570 BT_ERR("RegisterProfile failed: %s", err->message);
572 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
573 result = BLUETOOTH_ERROR_ACCESS_DENIED;
575 result = BLUETOOTH_ERROR_INTERNAL;
580 g_variant_builder_unref(option_builder);
583 g_variant_unref(ret);
588 int _bt_register_profile_platform(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
590 GVariantBuilder *option_builder;
594 int result = BLUETOOTH_ERROR_NONE;
596 proxy = __bt_gdbus_get_profile_proxy();
598 BT_ERR("Getting profile proxy failed");
599 return BLUETOOTH_ERROR_INTERNAL;
602 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
603 if (info->authentication)
604 g_variant_builder_add(option_builder, "{sv}",
605 "RequireAuthentication",
606 g_variant_new_boolean(TRUE));
607 if (info->authorization)
608 g_variant_builder_add(option_builder, "{sv}",
609 "RequireAuthorization",
610 g_variant_new_boolean(TRUE));
612 g_variant_builder_add(option_builder, "{sv}",
614 g_variant_new_string(info->role));
616 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
617 * RFCOMM channels based on the availability when two services want
618 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
619 * two services use the same SPP RFCOMM channel. */
620 if (use_default_rfcomm)
621 g_variant_builder_add(option_builder, "{sv}",
623 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
625 g_variant_builder_add(option_builder, "{sv}",
627 g_variant_new_string(info->service));
630 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile1",
631 g_variant_new("(osa{sv})", info->obj_path,
634 G_DBUS_CALL_FLAGS_NONE, -1,
637 BT_ERR("RegisterProfile failed: %s", err->message);
639 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
640 result = BLUETOOTH_ERROR_ACCESS_DENIED;
642 result = BLUETOOTH_ERROR_INTERNAL;
647 g_variant_builder_unref(option_builder);
650 g_variant_unref(ret);
656 void _bt_unregister_profile(char *path)
662 proxy = __bt_gdbus_get_profile_proxy();
664 BT_ERR("Getting profile proxy failed");
668 ret = g_dbus_proxy_call_sync(proxy, "UnregisterProfile",
669 g_variant_new("(o)", path),
670 G_DBUS_CALL_FLAGS_NONE, -1,
673 BT_ERR("UnregisterProfile failed : %s", err->message);
678 g_variant_unref(ret);
683 GDBusNodeInfo * _bt_get_gdbus_node(const gchar *xml_data)
686 char *name = g_strdup_printf("org.bt.frwk%d", getpid());
688 bus_id = g_bus_own_name(G_BUS_TYPE_SYSTEM,
690 G_BUS_NAME_OWNER_FLAGS_NONE,
696 BT_DBG("Got bus id %d", bus_id);
700 return g_dbus_node_info_new_for_xml(xml_data, NULL);
703 int _bt_connect_profile(char *address, char *uuid, void *cb,
708 DBusGConnection *conn;
709 DBusGProxy *adapter_proxy;
710 GError *error = NULL;
712 conn = _bt_get_system_gconn();
713 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
715 object_path = _bt_get_device_object_path(address);
716 if (object_path == NULL) {
717 BT_ERR("No searched device");
719 adapter_proxy = _bt_get_adapter_proxy(conn);
720 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
722 dbus_g_proxy_call(adapter_proxy, "CreateDevice", &error,
723 G_TYPE_STRING, address,
724 G_TYPE_INVALID, G_TYPE_INVALID);
727 BT_ERR("CreateDevice Fail: %s", error->message);
731 object_path = _bt_get_device_object_path(address);
733 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
736 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
737 object_path, BT_DEVICE_INTERFACE);
739 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
741 if (!dbus_g_proxy_begin_call(proxy, "ConnectProfile",
742 (DBusGProxyCallNotify)cb,
746 BT_ERR("Connect Dbus Call Error");
747 g_object_unref(proxy);
748 return BLUETOOTH_ERROR_INTERNAL;
750 return BLUETOOTH_ERROR_NONE;
753 int _bt_discover_services(char *address, char *uuid, void *cb,
758 DBusGConnection *conn;
759 DBusGProxy *adapter_proxy;
760 GError *error = NULL;
764 conn = _bt_get_system_gconn();
765 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
767 object_path = _bt_get_device_object_path(address);
768 if (object_path == NULL) {
769 BT_ERR("No searched device");
771 adapter_proxy = _bt_get_adapter_proxy(conn);
772 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
774 dbus_g_proxy_call(adapter_proxy, "CreateDevice", &error,
775 G_TYPE_STRING, address,
776 G_TYPE_INVALID, G_TYPE_INVALID);
779 BT_ERR("CreateDevice Fail: %s", error->message);
783 object_path = _bt_get_device_object_path(address);
785 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
787 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
788 object_path, BT_DEVICE_INTERFACE);
790 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
792 if (!dbus_g_proxy_begin_call_with_timeout(proxy, "DiscoverServices",
793 (DBusGProxyCallNotify)cb,
794 func_data, NULL, timeout,
797 BT_ERR("Error: While calling DiscoverServices");
798 g_object_unref(proxy);
799 return BLUETOOTH_ERROR_INTERNAL;
802 return BLUETOOTH_ERROR_NONE;
805 int _bt_cancel_discovers(char *address)
809 DBusGConnection *conn;
811 conn = _bt_get_system_gconn();
812 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
814 object_path = _bt_get_device_object_path(address);
815 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
817 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
818 object_path, BT_DEVICE_INTERFACE);
820 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
821 if (!dbus_g_proxy_call(proxy,
824 G_TYPE_INVALID, G_TYPE_INVALID)) {
825 BT_ERR("Error: while CancelDiscovery");
826 g_object_unref(proxy);
827 return BLUETOOTH_ERROR_INTERNAL;
829 g_object_unref(proxy);
831 return BLUETOOTH_ERROR_NONE;
834 int _bt_discover_service_uuids(char *address, char *remote_uuid)
838 DBusGConnection *conn;
839 GHashTable *hash = NULL;
843 int result = BLUETOOTH_ERROR_INTERNAL;
844 retv_if(remote_uuid == NULL, BLUETOOTH_ERROR_INTERNAL);
846 conn = _bt_get_system_gconn();
847 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
849 object_path = _bt_get_device_object_path(address);
850 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
852 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
853 object_path, BT_PROPERTIES_INTERFACE);
856 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
858 if (!dbus_g_proxy_call(proxy, "GetAll", NULL,
859 G_TYPE_STRING, BT_DEVICE_INTERFACE,
861 dbus_g_type_get_map("GHashTable", G_TYPE_STRING,
862 G_TYPE_VALUE), &hash, G_TYPE_INVALID)) {
863 BT_ERR("Dbus error while GetAll");
864 g_object_unref(proxy);
865 return BLUETOOTH_ERROR_INTERNAL;
867 g_object_unref(proxy);
868 BT_DBG("Remote uuids %s", remote_uuid);
869 value = g_hash_table_lookup(hash, "UUIDs");
871 BT_ERR("No uuids found");
875 uuid_value = g_value_get_boxed(value);
876 if(uuid_value == NULL) {
877 BT_ERR("Error: while obtaining uuids");
881 for (i = 0; uuid_value[i] != NULL; i++) {
882 BT_DBG("Remote uuids %s", uuid_value[i]);
883 if (strcasecmp(uuid_value[i], remote_uuid) == 0) {
884 result = BLUETOOTH_ERROR_NONE;
888 BT_INFO("Specified uuid not found on remote device");
891 g_hash_table_destroy(hash);
896 int _bt_disconnect_profile(char *address, char *uuid, void *cb,
901 DBusGConnection *conn;
903 conn = _bt_get_system_gconn();
904 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
906 object_path = _bt_get_device_object_path(address);
907 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
910 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
911 object_path, BT_DEVICE_INTERFACE);
913 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
915 if (!dbus_g_proxy_begin_call(proxy, "DisconnectProfile",
916 (DBusGProxyCallNotify)cb,
920 BT_ERR("Connect Dbus Call Error");
921 g_object_unref(proxy);
922 return BLUETOOTH_ERROR_INTERNAL;
924 return BLUETOOTH_ERROR_NONE;
927 int _bt_get_adapter_path(GDBusConnection *conn, char *path)
930 GDBusProxy *manager_proxy = NULL;
931 GVariant *result = NULL;
932 char *adapter_path = NULL;
934 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
936 manager_proxy = g_dbus_proxy_new_sync(conn,
937 G_DBUS_PROXY_FLAGS_NONE, NULL,
940 BT_MANAGER_INTERFACE,
943 if (!manager_proxy) {
944 BT_ERR("Unable to create proxy: %s", err->message);
948 result = g_dbus_proxy_call_sync(manager_proxy, "DefaultAdapter", NULL,
949 G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
952 BT_ERR("Fail to get DefaultAdapter (Error: %s)", err->message);
954 BT_ERR("Fail to get DefaultAdapter");
959 if (g_strcmp0(g_variant_get_type_string(result), "(o)")) {
960 BT_ERR("Incorrect result\n");
964 g_variant_get(result, "(&o)", &adapter_path);
966 if (adapter_path == NULL ||
967 strlen(adapter_path) >= BT_ADAPTER_OBJECT_PATH_MAX) {
968 BT_ERR("Adapter path is inproper\n");
973 g_strlcpy(path, adapter_path, BT_ADAPTER_OBJECT_PATH_MAX);
975 g_variant_unref(result);
976 g_object_unref(manager_proxy);
978 return BLUETOOTH_ERROR_NONE;
984 g_variant_unref(result);
987 g_object_unref(manager_proxy);
989 return BLUETOOTH_ERROR_INTERNAL;
993 void _bt_convert_device_path_to_address(const char *device_path,
994 char *device_address)
996 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
999 ret_if(device_path == NULL);
1000 ret_if(device_address == NULL);
1002 dev_addr = strstr(device_path, "dev_");
1003 if (dev_addr != NULL) {
1006 g_strlcpy(address, dev_addr, sizeof(address));
1008 while ((pos = strchr(address, '_')) != NULL) {
1012 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
1016 static char *__bt_extract_adapter_path(DBusMessageIter *msg_iter)
1018 char *object_path = NULL;
1019 DBusMessageIter value_iter;
1021 /* Parse the signature: oa{sa{sv}}} */
1022 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
1023 DBUS_TYPE_OBJECT_PATH, NULL);
1025 dbus_message_iter_get_basic(msg_iter, &object_path);
1026 retv_if(object_path == NULL, NULL);
1028 /* object array (oa) */
1029 retv_if(dbus_message_iter_next(msg_iter) == FALSE, NULL);
1030 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
1031 DBUS_TYPE_ARRAY, NULL);
1033 dbus_message_iter_recurse(msg_iter, &value_iter);
1035 /* string array (sa) */
1036 while (dbus_message_iter_get_arg_type(&value_iter) ==
1037 DBUS_TYPE_DICT_ENTRY) {
1038 char *interface_name = NULL;
1039 DBusMessageIter interface_iter;
1041 dbus_message_iter_recurse(&value_iter, &interface_iter);
1043 retv_if(dbus_message_iter_get_arg_type(&interface_iter) !=
1044 DBUS_TYPE_STRING, NULL);
1046 dbus_message_iter_get_basic(&interface_iter, &interface_name);
1048 if (g_strcmp0(interface_name, "org.bluez.Adapter1") == 0) {
1049 /* Tizen don't allow the multi-adapter */
1050 BT_DBG("Found an adapter: %s", object_path);
1051 return g_strdup(object_path);
1054 dbus_message_iter_next(&value_iter);
1060 static char *__bt_extract_device_path(DBusMessageIter *msg_iter, char *address)
1062 char *object_path = NULL;
1063 char device_address[BT_ADDRESS_STRING_SIZE] = { 0 };
1065 /* Parse the signature: oa{sa{sv}}} */
1066 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
1067 DBUS_TYPE_OBJECT_PATH, NULL);
1069 dbus_message_iter_get_basic(msg_iter, &object_path);
1070 retv_if(object_path == NULL, NULL);
1072 _bt_convert_device_path_to_address(object_path, device_address);
1074 if (g_strcmp0(address, device_address) == 0) {
1075 return g_strdup(object_path);
1081 char *_bt_get_device_object_path(char *address)
1085 DBusMessageIter reply_iter;
1086 DBusMessageIter value_iter;
1088 DBusConnection *conn;
1089 char *object_path = NULL;
1091 conn = _bt_get_system_conn();
1092 retv_if(conn == NULL, NULL);
1094 msg = dbus_message_new_method_call(BT_BLUEZ_NAME, BT_MANAGER_PATH,
1095 BT_MANAGER_INTERFACE,
1096 "GetManagedObjects");
1098 retv_if(msg == NULL, NULL);
1100 /* Synchronous call */
1101 dbus_error_init(&err);
1102 reply = dbus_connection_send_with_reply_and_block(
1105 dbus_message_unref(msg);
1108 BT_ERR("Can't get managed objects");
1110 if (dbus_error_is_set(&err)) {
1111 BT_ERR("%s", err.message);
1112 dbus_error_free(&err);
1117 if (dbus_message_iter_init(reply, &reply_iter) == FALSE) {
1118 BT_ERR("Fail to iterate the reply");
1119 dbus_message_unref(reply);
1123 dbus_message_iter_recurse(&reply_iter, &value_iter);
1125 /* signature of GetManagedObjects: a{oa{sa{sv}}} */
1126 while (dbus_message_iter_get_arg_type(&value_iter) ==
1127 DBUS_TYPE_DICT_ENTRY) {
1128 DBusMessageIter msg_iter;
1130 dbus_message_iter_recurse(&value_iter, &msg_iter);
1132 object_path = __bt_extract_device_path(&msg_iter, address);
1133 if (object_path != NULL) {
1134 BT_DBG("Found the device path %s", object_path);
1138 dbus_message_iter_next(&value_iter);
1141 dbus_message_unref(reply);
1147 DBusGProxy *_bt_get_adapter_proxy(DBusGConnection *conn)
1149 DBusGProxy *adapter_proxy = NULL;
1151 retv_if(conn == NULL, NULL);
1153 adapter_proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
1154 BT_BLUEZ_HCI_PATH, BT_PROPERTIES_INTERFACE);
1156 return adapter_proxy;
1159 void _bt_device_path_to_address(const char *device_path, char *device_address)
1161 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
1162 char *dev_addr = NULL;
1164 if (!device_path || !device_address)
1167 dev_addr = strstr(device_path, "dev_");
1168 if (dev_addr != NULL) {
1171 g_strlcpy(address, dev_addr, sizeof(address));
1173 while ((pos = strchr(address, '_')) != NULL) {
1177 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
1181 DBusGConnection *__bt_init_system_gconn(void)
1185 if (system_conn == NULL)
1186 system_conn = dbus_g_bus_get(DBUS_BUS_SYSTEM, NULL);
1191 DBusGConnection *_bt_get_system_gconn(void)
1193 return (system_conn) ? system_conn : __bt_init_system_gconn();
1196 GDBusConnection *_bt_init_system_gdbus_conn(void)
1199 GError *error = NULL;
1200 if (system_gdbus_conn == NULL) {
1202 g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
1204 BT_ERR("GDBus connection Error : %s \n",
1206 g_clear_error(&error);
1210 return system_gdbus_conn;
1213 DBusConnection *_bt_get_system_conn(void)
1215 DBusGConnection *g_conn;
1217 if (system_conn == NULL) {
1218 g_conn = __bt_init_system_gconn();
1220 g_conn = system_conn;
1223 retv_if(g_conn == NULL, NULL);
1225 return dbus_g_connection_get_connection(g_conn);
1228 int _bt_register_osp_server_in_agent(int type, char *uuid, char *path, int fd)
1231 char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
1232 char path_str[BLUETOOTH_PATH_STRING] = { 0, };
1236 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1238 g_array_append_vals(in_param1, &type, sizeof(int));
1239 g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
1240 g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
1241 g_strlcpy(path_str, path, sizeof(path_str));
1242 g_array_append_vals(in_param3, &path_str, BLUETOOTH_PATH_STRING);
1243 g_array_append_vals(in_param4, &fd, sizeof(int));
1245 ret = _bt_send_request(BT_AGENT_SERVICE, BT_SET_AUTHORIZATION,
1246 in_param1, in_param2, in_param3, in_param4, &out_param);
1248 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1253 int _bt_unregister_osp_server_in_agent(int type, char *uuid)
1256 char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
1257 char path_str[BLUETOOTH_PATH_STRING] = { 0, };
1261 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1263 g_array_append_vals(in_param1, &type, sizeof(int));
1264 g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
1265 g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
1267 ret = _bt_send_request(BT_AGENT_SERVICE, BT_UNSET_AUTHORIZATION,
1268 in_param1, in_param2, in_param3, in_param4, &out_param);
1270 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1275 int _bt_check_privilege(int service_type, int service_function)
1280 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1282 result = _bt_send_request(service_type, service_function,
1283 in_param1, in_param2, in_param3, in_param4, &out_param);
1285 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1290 GVariant *_bt_get_managed_objects(void)
1292 GDBusConnection *g_conn;
1293 GDBusProxy *manager_proxy = NULL;
1294 GVariant *result = NULL;
1295 GError *error = NULL;
1299 g_conn = _bt_gdbus_get_system_gconn();
1300 retv_if(g_conn == NULL, NULL);
1302 manager_proxy = g_dbus_proxy_new_sync(g_conn,
1303 G_DBUS_PROXY_FLAGS_NONE, NULL,
1306 BT_MANAGER_INTERFACE,
1310 BT_ERR("Unable to create proxy: %s", error->message);
1311 g_clear_error(&error);
1315 result = g_dbus_proxy_call_sync (manager_proxy,
1316 "GetManagedObjects", NULL,
1317 G_DBUS_CALL_FLAGS_NONE, -1,
1321 BT_ERR("Fail to get ManagedObjects (Error: %s)", error->message);
1322 g_clear_error(&error);
1325 g_object_unref(manager_proxy);
1331 BT_EXPORT_API int bluetooth_is_supported(void)
1333 int is_supported = 0;
1338 fd = open(RFKILL_NODE, O_RDONLY);
1340 BT_ERR("Fail to open RFKILL node");
1341 return BLUETOOTH_ERROR_INTERNAL;
1344 if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0) {
1345 BT_ERR("Fail to set RFKILL node to non-blocking");
1347 return BLUETOOTH_ERROR_INTERNAL;
1351 len = read(fd, &event, sizeof(event));
1353 BT_ERR("Fail to read events");
1357 if (len != RFKILL_EVENT_SIZE) {
1358 BT_ERR("The size is wrong\n");
1362 if (event.type == RFKILL_TYPE_BLUETOOTH) {
1370 BT_DBG("supported: %d", is_supported);
1372 return is_supported;
1375 BT_EXPORT_API int bluetooth_register_callback(bluetooth_cb_func_ptr callback_ptr, void *user_data)
1379 _bt_gdbus_init_system_gconn();
1380 __bt_init_system_gconn();
1382 ret = _bt_init_event_handler();
1383 if (ret != BLUETOOTH_ERROR_NONE &&
1384 ret != BLUETOOTH_ERROR_ALREADY_INITIALIZED) {
1385 BT_ERR("Fail to init the event handler");
1390 _bt_set_user_data(BT_COMMON, (void *)callback_ptr, user_data);
1392 /* Register All events */
1393 ret = _bt_register_event(BT_ADAPTER_EVENT, (void *)callback_ptr, user_data);
1394 if (ret != BLUETOOTH_ERROR_NONE)
1396 ret = _bt_register_event(BT_DEVICE_EVENT, (void *)callback_ptr, user_data);
1397 if (ret != BLUETOOTH_ERROR_NONE)
1399 ret = _bt_register_event(BT_NETWORK_EVENT, (void *)callback_ptr, user_data);
1400 if (ret != BLUETOOTH_ERROR_NONE)
1402 ret = _bt_register_event(BT_RFCOMM_CLIENT_EVENT, (void *)callback_ptr, user_data);
1403 if (ret != BLUETOOTH_ERROR_NONE)
1405 ret = _bt_register_event(BT_RFCOMM_SERVER_EVENT, (void *)callback_ptr, user_data);
1406 if (ret != BLUETOOTH_ERROR_NONE)
1409 _bt_register_name_owner_changed();
1411 return BLUETOOTH_ERROR_NONE;
1413 BT_ERR("Fail to do _bt_register_event()");
1414 bluetooth_unregister_callback();
1418 BT_EXPORT_API int bluetooth_unregister_callback(void)
1422 ret = _bt_deinit_event_handler();
1423 if (ret != BLUETOOTH_ERROR_NONE) {
1424 BT_ERR("Fail to deinit the event handler");
1427 _bt_unregister_name_owner_changed();
1429 _bt_set_user_data(BT_COMMON, NULL, NULL);
1432 dbus_g_connection_unref(system_conn);
1436 g_object_unref(system_gconn);
1437 system_gconn = NULL;
1439 return BLUETOOTH_ERROR_NONE;