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"
40 #include "bluetooth-ipsp-api.h"
42 #include "bt-common.h"
43 #include "bt-request-sender.h"
44 #include "bt-event-handler.h"
46 static bt_user_info_t user_info[BT_MAX_USER_INFO];
47 static DBusConnection *system_conn = NULL;
48 static GDBusConnection *system_gdbus_conn = NULL;
53 static GDBusConnection *system_gconn = NULL;
55 #define DBUS_TIMEOUT 20 * 1000 /* 20 Seconds */
57 GDBusConnection *g_bus_get_private_conn(void)
61 GDBusConnection *private_gconn = NULL;
63 address = g_dbus_address_get_for_bus_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
64 if (address == NULL) {
66 BT_ERR ("Failed to get bus address: %s", error->message);
67 g_clear_error(&error);
72 private_gconn = g_dbus_connection_new_for_address_sync (address,
73 G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_CLIENT |
74 G_DBUS_CONNECTION_FLAGS_MESSAGE_BUS_CONNECTION,
75 NULL, /* GDBusAuthObserver */
80 BT_ERR("Unable to connect to dbus: %s", error->message);
81 g_clear_error(&error);
89 GDBusConnection *_bt_gdbus_init_system_gconn(void)
91 dbus_threads_init_default();
95 if (system_gconn != NULL)
98 system_gconn = g_bus_get_private_conn();
103 GDBusConnection *_bt_gdbus_get_system_gconn(void)
105 if (system_gconn == NULL) {
106 system_gconn = _bt_gdbus_init_system_gconn();
107 } else if (g_dbus_connection_is_closed(system_gconn)){
108 system_gconn = g_bus_get_private_conn();
114 void _bt_print_device_address_t(const bluetooth_device_address_t *addr)
116 BT_DBG("%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X\n", addr->addr[0], addr->addr[1], addr->addr[2],
117 addr->addr[3], addr->addr[4], addr->addr[5]);
120 void _bt_set_user_data(int type, void *callback, void *user_data)
122 user_info[type].cb = callback;
123 user_info[type].user_data = user_data;
126 bt_user_info_t *_bt_get_user_data(int type)
128 return &user_info[type];
131 void _bt_common_event_cb(int event, int result, void *param,
132 void *callback, void *user_data)
134 bluetooth_event_param_t bt_event = { 0, };
135 bt_event.event = event;
136 bt_event.result = result;
137 bt_event.param_data = param;
140 ((bluetooth_cb_func_ptr)callback)(bt_event.event, &bt_event,
144 void _bt_input_event_cb(int event, int result, void *param,
145 void *callback, void *user_data)
147 hid_event_param_t bt_event = { 0, };
148 bt_event.event = event;
149 bt_event.result = result;
150 bt_event.param_data = param;
153 ((hid_cb_func_ptr)callback)(bt_event.event, &bt_event,
157 void _bt_headset_event_cb(int event, int result, void *param,
158 void *callback, void *user_data)
160 bt_audio_event_param_t bt_event = { 0, };
161 bt_event.event = event;
162 bt_event.result = result;
163 bt_event.param_data = param;
166 ((bt_audio_func_ptr)callback)(bt_event.event, &bt_event,
170 void _bt_a2dp_source_event_cb(int event, int result, void *param,
171 void *callback, void *user_data)
173 bt_audio_event_param_t bt_event = { 0, };
174 bt_event.event = event;
175 bt_event.result = result;
176 bt_event.param_data = param;
178 ((bt_audio_func_ptr)callback)(bt_event.event, &bt_event,
182 void _bt_hf_event_cb(int event, int result, void *param,
183 void *callback, void *user_data)
185 bt_hf_event_param_t bt_event = { 0, };
186 bt_event.event = event;
187 bt_event.result = result;
188 bt_event.param_data = param;
191 ((bt_hf_func_ptr)callback)(bt_event.event, &bt_event,
196 void _bt_avrcp_event_cb(int event, int result, void *param,
197 void *callback, void *user_data)
199 media_event_param_t bt_event = { 0, };
200 bt_event.event = event;
201 bt_event.result = result;
202 bt_event.param_data = param;
205 ((media_cb_func_ptr)callback)(bt_event.event, &bt_event,
209 void _bt_divide_device_class(bluetooth_device_class_t *device_class,
212 ret_if(device_class == NULL);
214 device_class->major_class = (unsigned short)(cod & 0x00001F00) >> 8;
215 device_class->minor_class = (unsigned short)((cod & 0x000000FC));
216 device_class->service_class = (unsigned long)((cod & 0x00FF0000));
218 if (cod & 0x002000) {
219 device_class->service_class |=
220 BLUETOOTH_DEVICE_SERVICE_CLASS_LIMITED_DISCOVERABLE_MODE;
224 void _bt_convert_addr_string_to_type(unsigned char *addr,
230 ret_if(address == NULL);
231 ret_if(addr == NULL);
233 for (i = 0; i < BT_ADDRESS_LENGTH_MAX; i++) {
234 addr[i] = strtol(address, &ptr, 16);
235 if (ptr[0] != '\0') {
244 void _bt_convert_addr_type_to_string(char *address,
247 ret_if(address == NULL);
248 ret_if(addr == NULL);
250 g_snprintf(address, BT_ADDRESS_STRING_SIZE,
251 "%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X",
252 addr[0], addr[1], addr[2],
253 addr[3], addr[4], addr[5]);
256 int _bt_copy_utf8_string(char *dest, const char *src, unsigned int length)
263 if (dest == NULL || src == NULL)
264 return BLUETOOTH_ERROR_INVALID_PARAM;
267 while (*p != '\0' && i < length) {
268 next = g_utf8_next_char(p);
271 while (count > 0 && ((i + count) < length)) {
278 return BLUETOOTH_ERROR_NONE;
281 gboolean _bt_utf8_validate(char *name)
285 glong items_written = 0;
287 if (FALSE == g_utf8_validate(name, -1, NULL))
290 u16 = g_utf8_to_utf16(name, -1, NULL, &items_written, NULL);
296 if (items_written != g_utf8_strlen(name, -1))
304 static GDBusProxy *profile_gproxy;
305 static GDBusConnection *gconn;
306 static int latest_id = -1;
307 #define BT_RFCOMM_ID_MAX 245
308 static gboolean id_used[BT_RFCOMM_ID_MAX];
309 GDBusNodeInfo *new_conn_node;
311 static const gchar rfcomm_agent_xml[] =
313 " <interface name='org.bluez.Profile1'>"
314 " <method name='NewConnection'>"
315 " <arg type='o' name='object' direction='in'/>"
316 " <arg type='h' name='fd' direction='in'/>"
317 " <arg type='a{sv}' name='properties' direction='in'/>"
319 " <method name='RequestDisconnection'>"
320 " <arg type='o' name='device' direction='in'/>"
325 static void __new_connection_method(GDBusConnection *connection,
327 const gchar *object_path,
328 const gchar *interface_name,
329 const gchar *method_name,
330 GVariant *parameters,
331 GDBusMethodInvocation *invocation,
334 BT_DBG("method %s", method_name);
335 if (g_strcmp0(method_name, "NewConnection") == 0) {
338 GUnixFDList *fd_list;
339 GVariantBuilder *properties;
342 bluetooth_device_address_t remote_addr1;
343 bt_new_connection_cb cb = user_data;
346 g_variant_get(parameters, "(oha{sv})", &obj_path, &index,
349 msg = g_dbus_method_invocation_get_message(invocation);
350 fd_list = g_dbus_message_get_unix_fd_list(msg);
351 if (fd_list == NULL) {
352 GQuark quark = g_quark_from_string("rfcomm-app");
353 GError *err = g_error_new(quark, 0, "No fd in message");
354 g_dbus_method_invocation_return_gerror(invocation, err);
360 fd = g_unix_fd_list_get(fd_list, index, NULL);
362 BT_ERR("Invalid fd return");
363 GQuark quark = g_quark_from_string("rfcomm-app");
364 GError *err = g_error_new(quark, 0, "Invalid FD return");
365 g_dbus_method_invocation_return_gerror(invocation, err);
369 BT_INFO("Object Path %s", obj_path);
371 _bt_device_path_to_address(obj_path, addr);
372 _bt_convert_addr_string_to_type(remote_addr1.addr, (const char *)addr);
373 BT_INFO("fd: %d, address %s", fd, addr);
375 g_dbus_method_invocation_return_value(invocation, NULL);
378 cb(object_path, fd, &remote_addr1);
379 } else if (g_strcmp0(method_name, "RequestDisconnection") == 0) {
380 g_dbus_method_invocation_return_value(invocation, NULL);
385 static const GDBusInterfaceVTable method_table = {
386 __new_connection_method,
391 void _bt_swap_addr(unsigned char *dst, const unsigned char *src)
395 for (i = 0; i < 6; i++)
399 int __rfcomm_assign_id(void)
403 BT_DBG("latest_id: %d", latest_id);
405 index = latest_id + 1;
407 if (index >= BT_RFCOMM_ID_MAX)
410 BT_DBG("index: %d", index);
412 while (id_used[index] == TRUE) {
413 if (index == latest_id) {
414 /* No available ID */
415 BT_ERR("All request ID is used");
421 if (index >= BT_RFCOMM_ID_MAX)
426 id_used[index] = TRUE;
428 BT_DBG("Assigned Id: %d", latest_id);
433 void __rfcomm_delete_id(int id)
435 ret_if(id >= BT_RFCOMM_ID_MAX);
440 /* Next server will use this ID */
444 static GDBusConnection *__get_gdbus_connection()
447 gconn = g_bus_get_private_conn();
452 static GDBusProxy *__bt_gdbus_get_profile_proxy(void)
454 GDBusConnection *gconn;
458 return profile_gproxy;
460 gconn = __get_gdbus_connection();
464 profile_gproxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE,
467 "org.bluez.ProfileManager1",
470 BT_ERR("Unable to create proxy: %s", err->message);
475 return profile_gproxy;
478 static GDBusProxy *__bt_gdbus_get_device_proxy(char *object_path)
480 GDBusConnection *gconn;
482 GDBusProxy *device_gproxy;
484 gconn = __get_gdbus_connection();
488 device_gproxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE,
495 BT_ERR("Unable to create proxy: %s", err->message);
500 return device_gproxy;
503 void _bt_unregister_gdbus(int object_id)
505 GDBusConnection *gconn;
507 gconn = __get_gdbus_connection();
511 g_dbus_connection_unregister_object(gconn, object_id);
514 int _bt_register_new_conn(const char *path, bt_new_connection_cb cb)
516 GDBusConnection *gconn;
518 GError *error = NULL;
520 gconn = __get_gdbus_connection();
524 if (new_conn_node == NULL)
525 new_conn_node = _bt_get_gdbus_node(rfcomm_agent_xml);
527 if (new_conn_node == NULL)
530 id = g_dbus_connection_register_object(gconn, path,
531 new_conn_node->interfaces[0],
535 BT_ERR("Failed to register: %s", error->message);
540 BT_DBG("NEW CONNECTION ID %d", id);
545 static GDBusProxy * __bt_gdbus_get_adapter_proxy()
548 GDBusProxy *manager_proxy = NULL;
549 GDBusProxy *adapter_proxy = NULL;
550 GDBusConnection *conn;
551 GVariant *result = NULL;
552 char *adapter_path = NULL;
554 conn = __get_gdbus_connection();
555 retv_if(conn == NULL, NULL);
557 manager_proxy = g_dbus_proxy_new_sync(conn,
558 G_DBUS_PROXY_FLAGS_NONE, NULL,
561 BT_MANAGER_INTERFACE,
564 if (!manager_proxy) {
565 BT_ERR("Unable to create proxy: %s", err->message);
569 result = g_dbus_proxy_call_sync(manager_proxy, "DefaultAdapter", NULL,
570 G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
573 BT_ERR("Fail to get DefaultAdapter (Error: %s)", err->message);
575 BT_ERR("Fail to get DefaultAdapter");
580 if (g_strcmp0(g_variant_get_type_string(result), "(o)")) {
581 BT_ERR("Incorrect result\n");
585 g_variant_get(result, "(&o)", &adapter_path);
587 if (adapter_path == NULL ||
588 strlen(adapter_path) >= BT_ADAPTER_OBJECT_PATH_MAX) {
589 BT_ERR("Adapter path is inproper\n");
593 BT_INFO("Adapter Path %s", adapter_path);
595 adapter_proxy = g_dbus_proxy_new_sync(conn,
596 G_DBUS_PROXY_FLAGS_NONE, NULL,
599 BT_ADAPTER_INTERFACE,
602 BT_ERR("DBus Error message: [%s]", err->message);
608 g_object_unref(manager_proxy);
610 g_variant_unref(result);
611 return adapter_proxy;
614 int _bt_register_new_conn_ex(const char *path, const char *bus_name,bt_new_connection_cb cb)
616 GDBusConnection *gconn;
618 GError *error = NULL;
620 gconn = __get_gdbus_connection();
624 if (new_conn_node == NULL)
625 new_conn_node = _bt_get_gdbus_node_ex(rfcomm_agent_xml, bus_name);
627 if (new_conn_node == NULL)
630 id = g_dbus_connection_register_object(gconn, path,
631 new_conn_node->interfaces[0],
635 BT_ERR("Failed to register: %s", error->message);
640 BT_DBG("NEW CONNECTION ID %d", id);
645 int _bt_register_profile(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
647 GVariantBuilder *option_builder;
651 int result = BLUETOOTH_ERROR_NONE;
653 proxy = __bt_gdbus_get_profile_proxy();
655 BT_ERR("Getting profile proxy failed");
656 return BLUETOOTH_ERROR_INTERNAL;
659 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
660 if (info->authentication)
661 g_variant_builder_add(option_builder, "{sv}",
662 "RequireAuthentication",
663 g_variant_new_boolean(TRUE));
664 if (info->authorization)
665 g_variant_builder_add(option_builder, "{sv}",
666 "RequireAuthorization",
667 g_variant_new_boolean(TRUE));
669 g_variant_builder_add(option_builder, "{sv}",
671 g_variant_new_string(info->role));
673 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
674 * RFCOMM channels based on the availability when two services want
675 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
676 * two services use the same SPP RFCOMM channel. */
677 if (use_default_rfcomm)
678 g_variant_builder_add(option_builder, "{sv}",
680 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
682 g_variant_builder_add(option_builder, "{sv}",
684 g_variant_new_string(info->service));
686 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile",
687 g_variant_new("(osa{sv})", info->obj_path,
690 G_DBUS_CALL_FLAGS_NONE, -1,
693 g_dbus_error_strip_remote_error(err);
694 BT_ERR("RegisterProfile failed: %s", err->message);
696 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
697 result = BLUETOOTH_ERROR_ACCESS_DENIED;
699 result = BLUETOOTH_ERROR_INTERNAL;
704 g_variant_builder_unref(option_builder);
707 g_variant_unref(ret);
712 int _bt_register_profile_ex(bt_register_profile_info_t *info, gboolean use_default_rfcomm, const char *name, const char *path)
714 GVariantBuilder *option_builder;
718 int result = BLUETOOTH_ERROR_NONE;
720 proxy = __bt_gdbus_get_profile_proxy();
722 BT_ERR("Getting profile proxy failed");
723 return BLUETOOTH_ERROR_INTERNAL;
726 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
727 if (info->authentication)
728 g_variant_builder_add(option_builder, "{sv}",
729 "RequireAuthentication",
730 g_variant_new_boolean(TRUE));
731 if (info->authorization)
732 g_variant_builder_add(option_builder, "{sv}",
733 "RequireAuthorization",
734 g_variant_new_boolean(TRUE));
736 g_variant_builder_add(option_builder, "{sv}",
738 g_variant_new_string(info->role));
740 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
741 * RFCOMM channels based on the availability when two services want
742 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
743 * two services use the same SPP RFCOMM channel. */
744 if (use_default_rfcomm)
745 g_variant_builder_add(option_builder, "{sv}",
747 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
749 g_variant_builder_add(option_builder, "{sv}",
751 g_variant_new_string(info->service));
753 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile2",
754 g_variant_new("(osssa{sv})", info->obj_path,
759 G_DBUS_CALL_FLAGS_NONE, -1,
762 g_dbus_error_strip_remote_error(err);
763 BT_ERR("RegisterProfile failed: %s", err->message);
765 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
766 result = BLUETOOTH_ERROR_ACCESS_DENIED;
768 result = BLUETOOTH_ERROR_INTERNAL;
773 g_variant_builder_unref(option_builder);
776 g_variant_unref(ret);
781 int _bt_register_profile_platform(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
783 GVariantBuilder *option_builder;
787 int result = BLUETOOTH_ERROR_NONE;
789 proxy = __bt_gdbus_get_profile_proxy();
791 BT_ERR("Getting profile proxy failed");
792 return BLUETOOTH_ERROR_INTERNAL;
795 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
796 if (info->authentication)
797 g_variant_builder_add(option_builder, "{sv}",
798 "RequireAuthentication",
799 g_variant_new_boolean(TRUE));
800 if (info->authorization)
801 g_variant_builder_add(option_builder, "{sv}",
802 "RequireAuthorization",
803 g_variant_new_boolean(TRUE));
805 g_variant_builder_add(option_builder, "{sv}",
807 g_variant_new_string(info->role));
809 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
810 * RFCOMM channels based on the availability when two services want
811 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
812 * two services use the same SPP RFCOMM channel. */
813 if (use_default_rfcomm)
814 g_variant_builder_add(option_builder, "{sv}",
816 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
818 g_variant_builder_add(option_builder, "{sv}",
820 g_variant_new_string(info->service));
822 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile1",
823 g_variant_new("(osa{sv})", info->obj_path,
826 G_DBUS_CALL_FLAGS_NONE, -1,
830 g_dbus_error_strip_remote_error(err);
831 BT_ERR("RegisterProfile failed: %s", err->message);
833 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
834 result = BLUETOOTH_ERROR_ACCESS_DENIED;
836 result = BLUETOOTH_ERROR_INTERNAL;
841 g_variant_builder_unref(option_builder);
844 g_variant_unref(ret);
850 void _bt_unregister_profile(char *path)
856 proxy = __bt_gdbus_get_profile_proxy();
858 BT_ERR("Getting profile proxy failed");
862 ret = g_dbus_proxy_call_sync(proxy, "UnregisterProfile",
863 g_variant_new("(o)", path),
864 G_DBUS_CALL_FLAGS_NONE, -1,
867 BT_ERR("UnregisterProfile failed : %s", err->message);
872 g_variant_unref(ret);
877 GDBusNodeInfo * _bt_get_gdbus_node(const gchar *xml_data)
880 char *name = g_strdup_printf("org.bt.frwk%d", getpid());
882 bus_id = g_bus_own_name(G_BUS_TYPE_SYSTEM,
884 G_BUS_NAME_OWNER_FLAGS_NONE,
890 BT_DBG("Got bus id %d", bus_id);
894 return g_dbus_node_info_new_for_xml(xml_data, NULL);
897 GDBusNodeInfo * _bt_get_gdbus_node_ex(const gchar *xml_data, const char *bus_name)
900 char *name = g_strdup(bus_name);
901 bus_id = g_bus_own_name(G_BUS_TYPE_SYSTEM,
903 G_BUS_NAME_OWNER_FLAGS_NONE,
909 BT_DBG("Got bus id %d", bus_id);
913 return g_dbus_node_info_new_for_xml(xml_data, NULL);
916 int _bt_connect_profile(char *address, char *uuid, void *cb,
920 GDBusProxy *adapter_proxy;
924 object_path = _bt_get_device_object_path(address);
926 if (object_path == NULL) {
927 GVariant *ret = NULL;
928 BT_ERR("No searched device");
929 adapter_proxy = __bt_gdbus_get_adapter_proxy();
931 if (adapter_proxy == NULL) {
932 BT_ERR("adapter proxy is NULL");
933 return BLUETOOTH_ERROR_INTERNAL;
936 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
937 g_variant_new("(s)", address),
938 G_DBUS_CALL_FLAGS_NONE,
943 BT_ERR("CreateDevice Failed: %s", err->message);
947 g_variant_unref(ret);
948 g_object_unref(adapter_proxy);
949 object_path = _bt_get_device_object_path(address);
951 if (object_path == NULL)
952 return BLUETOOTH_ERROR_INTERNAL;
955 proxy = __bt_gdbus_get_device_proxy(object_path);
959 BT_ERR("Error while getting proxy");
960 return BLUETOOTH_ERROR_INTERNAL;
963 g_dbus_proxy_call(proxy, "ConnectProfile",
964 g_variant_new("(s)", uuid),
965 G_DBUS_CALL_FLAGS_NONE,
967 (GAsyncReadyCallback)cb,
970 return BLUETOOTH_ERROR_NONE;
973 int _bt_discover_services(char *address, char *uuid, void *cb,
978 GDBusProxy *adapter_proxy;
980 object_path = _bt_get_device_object_path(address);
981 if (object_path == NULL) {
982 GVariant *ret = NULL;
983 BT_ERR("No searched device");
984 adapter_proxy = __bt_gdbus_get_adapter_proxy();
985 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
986 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
987 g_variant_new("(s)", address),
988 G_DBUS_CALL_FLAGS_NONE,
992 BT_ERR("CreateDevice Failed: %s", err->message);
996 g_variant_unref(ret);
998 g_object_unref(adapter_proxy);
1000 object_path = _bt_get_device_object_path(address);
1001 if (object_path == NULL)
1002 return BLUETOOTH_ERROR_INTERNAL;
1004 proxy = __bt_gdbus_get_device_proxy(object_path);
1005 g_free(object_path);
1006 if (proxy == NULL) {
1007 BT_ERR("Error while getting proxy");
1008 return BLUETOOTH_ERROR_INTERNAL;
1010 g_dbus_proxy_call(proxy, "DiscoverServices",
1011 g_variant_new("(s)", uuid),
1012 G_DBUS_CALL_FLAGS_NONE,
1014 (GAsyncReadyCallback)cb,
1017 return BLUETOOTH_ERROR_NONE;
1020 int _bt_cancel_discovers(char *address)
1024 GDBusProxy *adapter_proxy;
1026 object_path = _bt_get_device_object_path(address);
1027 if (object_path == NULL) {
1028 GVariant *ret = NULL;
1029 BT_ERR("No searched device");
1030 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1031 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1032 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1033 g_variant_new("(s)", address),
1034 G_DBUS_CALL_FLAGS_NONE,
1038 BT_ERR("CreateDevice Failed: %s", err->message);
1039 g_clear_error(&err);
1042 g_variant_unref(ret);
1044 g_object_unref(adapter_proxy);
1046 object_path = _bt_get_device_object_path(address);
1047 if (object_path == NULL)
1048 return BLUETOOTH_ERROR_INTERNAL;
1050 proxy = __bt_gdbus_get_device_proxy(object_path);
1051 g_free(object_path);
1052 g_dbus_proxy_call_sync(proxy, "CancelDiscovery",
1054 G_DBUS_CALL_FLAGS_NONE,
1058 BT_ERR("DBus Error message: [%s]", err->message);
1059 g_clear_error(&err);
1060 return BLUETOOTH_ERROR_INTERNAL;
1063 g_object_unref(proxy);
1064 return BLUETOOTH_ERROR_NONE;
1067 int _bt_discover_service_uuids(char *address, char *remote_uuid)
1071 GDBusConnection *gconn;
1073 char **uuid_value = NULL;
1076 GVariant *value = NULL;
1077 GVariant *ret = NULL;
1078 int result = BLUETOOTH_ERROR_INTERNAL;
1080 retv_if(remote_uuid == NULL, BLUETOOTH_ERROR_INTERNAL);
1081 gconn = __get_gdbus_connection();
1082 retv_if(gconn == NULL, BLUETOOTH_ERROR_INTERNAL);
1083 object_path = _bt_get_device_object_path(address);
1084 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
1086 proxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE, NULL,
1087 BT_BLUEZ_NAME, object_path, BT_PROPERTIES_INTERFACE, NULL,
1089 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1091 BT_ERR("DBus Error: [%s]", err->message);
1092 g_clear_error(&err);
1094 ret = g_dbus_proxy_call_sync(proxy, "GetAll",
1095 g_variant_new("(s)", BT_DEVICE_INTERFACE),
1096 G_DBUS_CALL_FLAGS_NONE,
1100 result = BLUETOOTH_ERROR_INTERNAL;
1101 BT_ERR("DBus Error : %s", err->message);
1102 g_clear_error(&err);
1106 BT_ERR("g_dbus_proxy_call_sync function return NULL");
1107 result = BLUETOOTH_ERROR_INTERNAL;
1111 g_variant_get(ret, "(@a{sv})", &value);
1112 g_variant_unref(ret);
1114 GVariant *temp_value = g_variant_lookup_value(value, "UUIDs",
1115 G_VARIANT_TYPE_STRING_ARRAY);
1118 size = g_variant_get_size(temp_value);
1120 uuid_value = (char **)g_variant_get_strv(temp_value, &size);
1121 BT_DBG("Size items %d", size);
1124 for (i = 0; uuid_value[i] != NULL; i++) {
1125 BT_DBG("Remote uuids %s", uuid_value[i]);
1126 if (strcasecmp(uuid_value[i], remote_uuid) == 0) {
1127 result = BLUETOOTH_ERROR_NONE;
1128 g_variant_unref(temp_value);
1134 g_variant_unref(temp_value);
1139 g_object_unref(proxy);
1141 g_variant_unref(value);
1149 int _bt_get_cod_by_address(char *address, bluetooth_device_class_t *dev_class)
1153 GDBusConnection *gconn;
1155 GVariant *value = NULL;
1156 GVariant *result = NULL;
1157 unsigned int class = 0x00;
1158 int ret = BLUETOOTH_ERROR_INTERNAL;
1160 gconn = __get_gdbus_connection();
1161 retv_if(gconn == NULL, BLUETOOTH_ERROR_INTERNAL);
1162 object_path = _bt_get_device_object_path(address);
1164 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
1166 proxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE, NULL,
1167 BT_BLUEZ_NAME, object_path, BT_PROPERTIES_INTERFACE, NULL,
1169 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1171 BT_ERR("DBus Error: [%s]", err->message);
1172 g_clear_error(&err);
1175 result = g_dbus_proxy_call_sync(proxy, "GetAll",
1176 g_variant_new("(s)", BT_DEVICE_INTERFACE),
1177 G_DBUS_CALL_FLAGS_NONE,
1181 ret = BLUETOOTH_ERROR_INTERNAL;
1182 BT_ERR("DBus Error : %s", err->message);
1183 g_clear_error(&err);
1186 if (result == NULL) {
1187 BT_ERR("g_dbus_proxy_call_sync function return NULL");
1188 ret = BLUETOOTH_ERROR_INTERNAL;
1191 g_variant_get(result, "(@a{sv})", &value);
1192 g_variant_unref(result);
1194 GVariant *temp_value = g_variant_lookup_value(value, "Class",
1195 G_VARIANT_TYPE_UINT32);
1196 class = g_variant_get_uint32(temp_value);
1197 _bt_divide_device_class(dev_class, class);
1199 g_variant_unref(temp_value);
1204 g_object_unref(proxy);
1206 g_variant_unref(value);
1212 int _bt_disconnect_profile(char *address, char *uuid, void *cb,
1218 GDBusProxy *adapter_proxy;
1219 object_path = _bt_get_device_object_path(address);
1220 if (object_path == NULL) {
1221 GVariant *ret = NULL;
1222 BT_ERR("No searched device");
1223 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1224 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1225 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1226 g_variant_new("(s)", address),
1227 G_DBUS_CALL_FLAGS_NONE,
1231 BT_ERR("CreateDevice Failed: %s", err->message);
1235 g_variant_unref(ret);
1236 g_object_unref(adapter_proxy);
1237 object_path = _bt_get_device_object_path(address);
1238 if (object_path == NULL)
1239 return BLUETOOTH_ERROR_INTERNAL;
1241 proxy = __bt_gdbus_get_device_proxy(object_path);
1242 g_free(object_path);
1243 if (proxy == NULL) {
1244 BT_ERR("Error while getting proxy");
1245 return BLUETOOTH_ERROR_INTERNAL;
1247 g_dbus_proxy_call(proxy, "DisconnectProfile",
1248 g_variant_new("(s)", uuid),
1249 G_DBUS_CALL_FLAGS_NONE,
1251 (GAsyncReadyCallback)cb,
1254 return BLUETOOTH_ERROR_NONE;
1257 int _bt_get_adapter_path(GDBusConnection *conn, char *path)
1260 GDBusProxy *manager_proxy = NULL;
1261 GVariant *result = NULL;
1262 char *adapter_path = NULL;
1264 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
1266 manager_proxy = g_dbus_proxy_new_sync(conn,
1267 G_DBUS_PROXY_FLAGS_NONE, NULL,
1270 BT_MANAGER_INTERFACE,
1273 if (!manager_proxy) {
1274 BT_ERR("Unable to create proxy: %s", err->message);
1278 result = g_dbus_proxy_call_sync(manager_proxy, "DefaultAdapter", NULL,
1279 G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
1282 BT_ERR("Fail to get DefaultAdapter (Error: %s)", err->message);
1284 BT_ERR("Fail to get DefaultAdapter");
1289 if (g_strcmp0(g_variant_get_type_string(result), "(o)")) {
1290 BT_ERR("Incorrect result\n");
1294 g_variant_get(result, "(&o)", &adapter_path);
1296 if (adapter_path == NULL ||
1297 strlen(adapter_path) >= BT_ADAPTER_OBJECT_PATH_MAX) {
1298 BT_ERR("Adapter path is inproper\n");
1303 g_strlcpy(path, adapter_path, BT_ADAPTER_OBJECT_PATH_MAX);
1305 g_variant_unref(result);
1306 g_object_unref(manager_proxy);
1308 return BLUETOOTH_ERROR_NONE;
1311 g_clear_error(&err);
1314 g_variant_unref(result);
1317 g_object_unref(manager_proxy);
1319 return BLUETOOTH_ERROR_INTERNAL;
1323 void _bt_convert_device_path_to_address(const char *device_path,
1324 char *device_address)
1326 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
1329 ret_if(device_path == NULL);
1330 ret_if(device_address == NULL);
1332 dev_addr = strstr(device_path, "dev_");
1333 if (dev_addr != NULL) {
1336 g_strlcpy(address, dev_addr, sizeof(address));
1338 while ((pos = strchr(address, '_')) != NULL) {
1342 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
1346 static char *__bt_extract_device_path(GVariantIter *iter, char *address)
1348 char *object_path = NULL;
1349 char device_address[BT_ADDRESS_STRING_SIZE] = { 0 };
1350 /* Parse the signature: oa{sa{sv}}} */
1351 while (g_variant_iter_loop(iter, "{&oa{sa{sv}}}", &object_path,
1353 retv_if(object_path == NULL, NULL);
1354 _bt_convert_device_path_to_address(object_path, device_address);
1356 if (g_strcmp0(address, device_address) == 0) {
1357 return g_strdup(object_path);
1363 char *_bt_get_device_object_path(char *address)
1366 GDBusProxy *proxy = NULL;
1367 GVariant *result = NULL;
1368 GVariantIter *iter = NULL;
1369 GDBusConnection *conn = NULL;
1370 char *object_path = NULL;
1372 conn = _bt_gdbus_get_system_gconn();
1373 retv_if(conn == NULL, NULL);
1375 proxy = g_dbus_proxy_new_sync(conn,
1376 G_DBUS_PROXY_FLAGS_NONE, NULL,
1379 BT_MANAGER_INTERFACE,
1383 BT_ERR("Unable to create proxy: %s", err->message);
1387 result = g_dbus_proxy_call_sync(proxy, "GetManagedObjects", NULL,
1388 G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
1391 BT_ERR("Fail to get GetManagedObjects (Error: %s)", err->message);
1393 BT_ERR("Fail to get GetManagedObjects");
1398 g_variant_get(result, "(a{oa{sa{sv}}})", &iter);
1399 object_path = __bt_extract_device_path(iter, address);
1401 g_variant_unref(result);
1402 g_object_unref(proxy);
1403 g_variant_iter_free(iter);
1407 g_clear_error(&err);
1410 g_object_unref(proxy);
1415 void _bt_device_path_to_address(const char *device_path, char *device_address)
1417 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
1418 char *dev_addr = NULL;
1420 if (!device_path || !device_address)
1423 dev_addr = strstr(device_path, "dev_");
1424 if (dev_addr != NULL) {
1427 g_strlcpy(address, dev_addr, sizeof(address));
1429 while ((pos = strchr(address, '_')) != NULL) {
1433 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
1437 /* TODO : replace the dbus-glib APIs to gdbus APIs */
1438 DBusConnection *__bt_init_system_conn(void)
1440 if (system_conn == NULL)
1441 system_conn = dbus_bus_get_private(DBUS_BUS_SYSTEM, NULL);
1444 dbus_connection_setup_with_g_main(system_conn, NULL);
1445 dbus_connection_set_exit_on_disconnect(system_conn, FALSE);
1451 DBusConnection *_bt_get_system_conn(void)
1453 DBusConnection *conn = NULL;
1455 if (system_conn == NULL) {
1456 conn = __bt_init_system_conn();
1465 int _bt_register_osp_server_in_agent(int type, char *uuid, char *path, int fd)
1468 char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
1469 char path_str[BLUETOOTH_PATH_STRING] = { 0, };
1473 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1475 g_array_append_vals(in_param1, &type, sizeof(int));
1476 g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
1477 g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
1478 g_strlcpy(path_str, path, sizeof(path_str));
1479 g_array_append_vals(in_param3, &path_str, BLUETOOTH_PATH_STRING);
1480 g_array_append_vals(in_param4, &fd, sizeof(int));
1482 ret = _bt_send_request(BT_AGENT_SERVICE, BT_SET_AUTHORIZATION,
1483 in_param1, in_param2, in_param3, in_param4, &out_param);
1485 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1490 int _bt_unregister_osp_server_in_agent(int type, char *uuid)
1493 char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
1497 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1499 g_array_append_vals(in_param1, &type, sizeof(int));
1500 g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
1501 g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
1503 ret = _bt_send_request(BT_AGENT_SERVICE, BT_UNSET_AUTHORIZATION,
1504 in_param1, in_param2, in_param3, in_param4, &out_param);
1506 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1511 int _bt_check_privilege(int service_type, int service_function)
1516 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1518 result = _bt_send_request(service_type, service_function,
1519 in_param1, in_param2, in_param3, in_param4, &out_param);
1521 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1526 GVariant *_bt_get_managed_objects(void)
1528 GDBusConnection *g_conn;
1529 GDBusProxy *manager_proxy = NULL;
1530 GVariant *result = NULL;
1531 GError *error = NULL;
1535 g_conn = _bt_gdbus_get_system_gconn();
1536 retv_if(g_conn == NULL, NULL);
1538 manager_proxy = g_dbus_proxy_new_sync(g_conn,
1539 G_DBUS_PROXY_FLAGS_NONE, NULL,
1542 BT_MANAGER_INTERFACE,
1546 BT_ERR("Unable to create proxy: %s", error->message);
1547 g_clear_error(&error);
1551 result = g_dbus_proxy_call_sync (manager_proxy,
1552 "GetManagedObjects", NULL,
1553 G_DBUS_CALL_FLAGS_NONE, -1,
1557 BT_ERR("Fail to get ManagedObjects (Error: %s)", error->message);
1558 g_clear_error(&error);
1561 g_object_unref(manager_proxy);
1567 BT_EXPORT_API int bluetooth_is_supported(void)
1569 int is_supported = 0;
1574 fd = open(RFKILL_NODE, O_RDONLY);
1576 BT_ERR("Fail to open RFKILL node");
1577 return BLUETOOTH_ERROR_INTERNAL;
1580 if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0) {
1581 BT_ERR("Fail to set RFKILL node to non-blocking");
1583 return BLUETOOTH_ERROR_INTERNAL;
1587 len = read(fd, &event, sizeof(event));
1589 BT_ERR("Fail to read events");
1593 if (len != RFKILL_EVENT_SIZE) {
1594 BT_ERR("The size is wrong\n");
1598 if (event.type == RFKILL_TYPE_BLUETOOTH) {
1606 BT_DBG("supported: %d", is_supported);
1608 return is_supported;
1611 BT_EXPORT_API int bluetooth_register_callback(bluetooth_cb_func_ptr callback_ptr, void *user_data)
1615 _bt_gdbus_init_system_gconn();
1616 __bt_init_system_conn();
1618 ret = _bt_init_event_handler();
1619 if (ret != BLUETOOTH_ERROR_NONE &&
1620 ret != BLUETOOTH_ERROR_ALREADY_INITIALIZED) {
1621 BT_ERR("Fail to init the event handler");
1626 _bt_set_user_data(BT_COMMON, (void *)callback_ptr, user_data);
1628 /* Register All events */
1629 ret = _bt_register_event(BT_ADAPTER_EVENT, (void *)callback_ptr, user_data);
1630 if (ret != BLUETOOTH_ERROR_NONE)
1632 ret = _bt_register_event(BT_DEVICE_EVENT, (void *)callback_ptr, user_data);
1633 if (ret != BLUETOOTH_ERROR_NONE)
1635 ret = _bt_register_event(BT_NETWORK_EVENT, (void *)callback_ptr, user_data);
1636 if (ret != BLUETOOTH_ERROR_NONE)
1638 ret = _bt_register_event(BT_RFCOMM_CLIENT_EVENT, (void *)callback_ptr, user_data);
1639 if (ret != BLUETOOTH_ERROR_NONE)
1641 ret = _bt_register_event(BT_RFCOMM_SERVER_EVENT, (void *)callback_ptr, user_data);
1642 if (ret != BLUETOOTH_ERROR_NONE)
1644 #ifdef GATT_NO_RELAY
1645 ret = _bt_register_event(BT_GATT_BLUEZ_EVENT, (void *)callback_ptr, user_data);
1646 if (ret != BLUETOOTH_ERROR_NONE)
1650 _bt_register_name_owner_changed();
1652 return BLUETOOTH_ERROR_NONE;
1654 BT_ERR("Fail to do _bt_register_event()");
1655 bluetooth_unregister_callback();
1659 BT_EXPORT_API int bluetooth_unregister_callback(void)
1664 ret = _bt_deinit_event_handler();
1665 if (ret != BLUETOOTH_ERROR_NONE) {
1666 BT_ERR("Fail to deinit the event handler");
1669 _bt_unregister_name_owner_changed();
1671 _bt_set_user_data(BT_COMMON, NULL, NULL);
1674 dbus_connection_flush(system_conn);
1675 dbus_connection_close(system_conn);
1676 dbus_connection_unref(system_conn);
1680 g_object_unref(system_gconn);
1681 system_gconn = NULL;
1683 _bt_gdbus_deinit_proxys();
1684 return BLUETOOTH_ERROR_NONE;