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;
52 static size_t cookie_size;
54 static GDBusConnection *system_gconn = NULL;
56 GDBusConnection *_bt_gdbus_init_system_gconn(void)
60 if (!g_thread_supported()) {
68 if (system_gconn != NULL)
71 system_gconn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
74 BT_ERR("Unable to connect to dbus: %s", error->message);
75 g_clear_error(&error);
81 GDBusConnection *_bt_gdbus_get_system_gconn(void)
83 GDBusConnection *local_system_gconn = NULL;
86 if (system_gconn == NULL) {
87 system_gconn = _bt_gdbus_init_system_gconn();
88 } else if (g_dbus_connection_is_closed(system_gconn)){
90 local_system_gconn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
92 if (!local_system_gconn) {
93 BT_ERR("Unable to connect to dbus: %s", error->message);
94 g_clear_error(&error);
97 system_gconn = local_system_gconn;
103 void _bt_print_device_address_t(const bluetooth_device_address_t *addr)
105 BT_DBG("%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X\n", addr->addr[0], addr->addr[1], addr->addr[2],
106 addr->addr[3], addr->addr[4], addr->addr[5]);
109 void _bt_set_user_data(int type, void *callback, void *user_data)
111 user_info[type].cb = callback;
112 user_info[type].user_data = user_data;
115 bt_user_info_t *_bt_get_user_data(int type)
117 return &user_info[type];
120 void _bt_common_event_cb(int event, int result, void *param,
121 void *callback, void *user_data)
123 bluetooth_event_param_t bt_event = { 0, };
124 bt_event.event = event;
125 bt_event.result = result;
126 bt_event.param_data = param;
129 ((bluetooth_cb_func_ptr)callback)(bt_event.event, &bt_event,
133 void _bt_input_event_cb(int event, int result, void *param,
134 void *callback, void *user_data)
136 hid_event_param_t bt_event = { 0, };
137 bt_event.event = event;
138 bt_event.result = result;
139 bt_event.param_data = param;
142 ((hid_cb_func_ptr)callback)(bt_event.event, &bt_event,
146 void _bt_headset_event_cb(int event, int result, void *param,
147 void *callback, void *user_data)
149 bt_audio_event_param_t bt_event = { 0, };
150 bt_event.event = event;
151 bt_event.result = result;
152 bt_event.param_data = param;
155 ((bt_audio_func_ptr)callback)(bt_event.event, &bt_event,
159 void _bt_hf_event_cb(int event, int result, void *param,
160 void *callback, void *user_data)
162 bt_hf_event_param_t bt_event = { 0, };
163 bt_event.event = event;
164 bt_event.result = result;
165 bt_event.param_data = param;
168 ((bt_hf_func_ptr)callback)(bt_event.event, &bt_event,
173 void _bt_avrcp_event_cb(int event, int result, void *param,
174 void *callback, void *user_data)
176 media_event_param_t bt_event = { 0, };
177 bt_event.event = event;
178 bt_event.result = result;
179 bt_event.param_data = param;
182 ((media_cb_func_ptr)callback)(bt_event.event, &bt_event,
186 void _bt_divide_device_class(bluetooth_device_class_t *device_class,
189 ret_if(device_class == NULL);
191 device_class->major_class = (unsigned short)(cod & 0x00001F00) >> 8;
192 device_class->minor_class = (unsigned short)((cod & 0x000000FC));
193 device_class->service_class = (unsigned long)((cod & 0x00FF0000));
195 if (cod & 0x002000) {
196 device_class->service_class |=
197 BLUETOOTH_DEVICE_SERVICE_CLASS_LIMITED_DISCOVERABLE_MODE;
201 void _bt_convert_addr_string_to_type(unsigned char *addr,
207 ret_if(address == NULL);
208 ret_if(addr == NULL);
210 for (i = 0; i < BT_ADDRESS_LENGTH_MAX; i++) {
211 addr[i] = strtol(address, &ptr, 16);
212 if (ptr[0] != '\0') {
221 void _bt_convert_addr_type_to_string(char *address,
224 ret_if(address == NULL);
225 ret_if(addr == NULL);
227 g_snprintf(address, BT_ADDRESS_STRING_SIZE,
228 "%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X",
229 addr[0], addr[1], addr[2],
230 addr[3], addr[4], addr[5]);
233 int _bt_copy_utf8_string(char *dest, const char *src, unsigned int length)
240 if (dest == NULL || src == NULL)
241 return BLUETOOTH_ERROR_INVALID_PARAM;
244 while (*p != '\0' && i < length) {
245 next = g_utf8_next_char(p);
248 while (count > 0 && ((i + count) < length)) {
255 return BLUETOOTH_ERROR_NONE;
258 gboolean _bt_utf8_validate(char *name)
262 glong items_written = 0;
264 if (FALSE == g_utf8_validate(name, -1, NULL))
267 u16 = g_utf8_to_utf16(name, -1, NULL, &items_written, NULL);
273 if (items_written != g_utf8_strlen(name, -1))
281 static GDBusProxy *profile_gproxy;
282 static GDBusConnection *gconn;
283 static int latest_id = -1;
284 #define BT_RFCOMM_ID_MAX 245
285 static gboolean id_used[BT_RFCOMM_ID_MAX];
286 GDBusNodeInfo *new_conn_node;
288 static const gchar rfcomm_agent_xml[] =
290 " <interface name='org.bluez.Profile1'>"
291 " <method name='NewConnection'>"
292 " <arg type='o' name='object' direction='in'/>"
293 " <arg type='h' name='fd' direction='in'/>"
294 " <arg type='a{sv}' name='properties' direction='in'/>"
299 /* Remote socket address */
300 struct sockaddr_remote {
302 bluetooth_device_address_t remote_bdaddr;
303 unsigned char channel;
306 static void __get_remote_address(int fd, bluetooth_device_address_t *bdaddr)
308 struct sockaddr_remote address;
309 socklen_t address_len;
311 address_len = sizeof(address);
312 if (getpeername(fd, (struct sockaddr *) &address, &address_len) != 0) {
313 BT_DBG("getpeername failed");
317 memcpy(bdaddr->addr, address.remote_bdaddr.addr,
318 BLUETOOTH_ADDRESS_LENGTH);
321 static void __new_connection_method(GDBusConnection *connection,
323 const gchar *object_path,
324 const gchar *interface_name,
325 const gchar *method_name,
326 GVariant *parameters,
327 GDBusMethodInvocation *invocation,
330 BT_DBG("method %s", method_name);
331 if (g_strcmp0(method_name, "NewConnection") == 0) {
334 GUnixFDList *fd_list;
335 GVariantBuilder *properties;
338 bluetooth_device_address_t remote_addr, remote_addr1;
339 bt_new_connection_cb cb = user_data;
342 g_variant_get(parameters, "(oha{sv})", &obj_path, &index,
345 msg = g_dbus_method_invocation_get_message(invocation);
346 fd_list = g_dbus_message_get_unix_fd_list(msg);
347 if (fd_list == NULL) {
348 GQuark quark = g_quark_from_string("rfcomm-app");
349 GError *err = g_error_new(quark, 0, "No fd in message");
350 g_dbus_method_invocation_return_gerror(invocation, err);
356 fd = g_unix_fd_list_get(fd_list, index, NULL);
358 BT_ERR("Invalid fd return");
359 GQuark quark = g_quark_from_string("rfcomm-app");
360 GError *err = g_error_new(quark, 0, "Invalid FD return");
361 g_dbus_method_invocation_return_gerror(invocation, err);
366 __get_remote_address(fd, &remote_addr);
367 _bt_swap_addr(remote_addr1.addr, remote_addr.addr);
368 _bt_convert_addr_type_to_string(addr, remote_addr1.addr);
369 BT_INFO("fd: %d, address %s", fd, addr);
371 g_dbus_method_invocation_return_value(invocation, NULL);
374 cb(object_path, fd, &remote_addr1);
379 static const GDBusInterfaceVTable method_table = {
380 __new_connection_method,
385 void _bt_swap_addr(unsigned char *dst, const unsigned char *src)
389 for (i = 0; i < 6; i++)
393 int __rfcomm_assign_id(void)
397 BT_DBG("latest_id: %d", latest_id);
399 index = latest_id + 1;
401 if (index >= BT_RFCOMM_ID_MAX)
404 BT_DBG("index: %d", index);
406 while (id_used[index] == TRUE) {
407 if (index == latest_id) {
408 /* No available ID */
409 BT_ERR("All request ID is used");
415 if (index >= BT_RFCOMM_ID_MAX)
420 id_used[index] = TRUE;
422 BT_DBG("Assigned Id: %d", latest_id);
427 void __rfcomm_delete_id(int id)
429 ret_if(id >= BT_RFCOMM_ID_MAX);
434 /* Next server will use this ID */
438 static GDBusConnection *__get_gdbus_connection()
443 gconn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &err);
447 BT_ERR("Unable to connect to dbus: %s", err->message);
456 static GDBusProxy *__bt_gdbus_get_profile_proxy(void)
458 GDBusConnection *gconn;
462 return profile_gproxy;
464 gconn = __get_gdbus_connection();
468 profile_gproxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE,
471 "org.bluez.ProfileManager1",
474 BT_ERR("Unable to create proxy: %s", err->message);
478 return profile_gproxy;
481 void _bt_unregister_gdbus(int object_id)
483 GDBusConnection *gconn;
485 gconn = __get_gdbus_connection();
489 g_dbus_connection_unregister_object(gconn, object_id);
492 int _bt_register_new_conn(const char *path, bt_new_connection_cb cb)
494 GDBusConnection *gconn;
496 GError *error = NULL;
498 gconn = __get_gdbus_connection();
502 if (new_conn_node == NULL)
503 new_conn_node = _bt_get_gdbus_node(rfcomm_agent_xml);
505 if (new_conn_node == NULL)
508 id = g_dbus_connection_register_object(gconn, path,
509 new_conn_node->interfaces[0],
513 BT_ERR("Failed to register: %s", error->message);
518 BT_DBG("NEW CONNECTION ID %d", id);
524 int _bt_register_profile(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
526 GVariantBuilder *option_builder;
530 int result = BLUETOOTH_ERROR_NONE;
532 proxy = __bt_gdbus_get_profile_proxy();
534 BT_ERR("Getting profile proxy failed");
535 return BLUETOOTH_ERROR_INTERNAL;
538 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
539 if (info->authentication)
540 g_variant_builder_add(option_builder, "{sv}",
541 "RequireAuthentication",
542 g_variant_new_boolean(TRUE));
543 if (info->authorization)
544 g_variant_builder_add(option_builder, "{sv}",
545 "RequireAuthorization",
546 g_variant_new_boolean(TRUE));
548 g_variant_builder_add(option_builder, "{sv}",
550 g_variant_new_string(info->role));
552 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
553 * RFCOMM channels based on the availability when two services want
554 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
555 * two services use the same SPP RFCOMM channel. */
556 if (use_default_rfcomm)
557 g_variant_builder_add(option_builder, "{sv}",
559 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
561 g_variant_builder_add(option_builder, "{sv}",
563 g_variant_new_string(info->service));
566 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile",
567 g_variant_new("(osa{sv})", info->obj_path,
570 G_DBUS_CALL_FLAGS_NONE, -1,
573 BT_ERR("RegisterProfile failed: %s", err->message);
575 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
576 result = BLUETOOTH_ERROR_ACCESS_DENIED;
578 result = BLUETOOTH_ERROR_INTERNAL;
583 g_variant_builder_unref(option_builder);
586 g_variant_unref(ret);
591 int _bt_register_profile_platform(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
593 GVariantBuilder *option_builder;
597 int result = BLUETOOTH_ERROR_NONE;
599 proxy = __bt_gdbus_get_profile_proxy();
601 BT_ERR("Getting profile proxy failed");
602 return BLUETOOTH_ERROR_INTERNAL;
605 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
606 if (info->authentication)
607 g_variant_builder_add(option_builder, "{sv}",
608 "RequireAuthentication",
609 g_variant_new_boolean(TRUE));
610 if (info->authorization)
611 g_variant_builder_add(option_builder, "{sv}",
612 "RequireAuthorization",
613 g_variant_new_boolean(TRUE));
615 g_variant_builder_add(option_builder, "{sv}",
617 g_variant_new_string(info->role));
619 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
620 * RFCOMM channels based on the availability when two services want
621 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
622 * two services use the same SPP RFCOMM channel. */
623 if (use_default_rfcomm)
624 g_variant_builder_add(option_builder, "{sv}",
626 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
628 g_variant_builder_add(option_builder, "{sv}",
630 g_variant_new_string(info->service));
633 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile1",
634 g_variant_new("(osa{sv})", info->obj_path,
637 G_DBUS_CALL_FLAGS_NONE, -1,
640 BT_ERR("RegisterProfile failed: %s", err->message);
642 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
643 result = BLUETOOTH_ERROR_ACCESS_DENIED;
645 result = BLUETOOTH_ERROR_INTERNAL;
650 g_variant_builder_unref(option_builder);
653 g_variant_unref(ret);
659 void _bt_unregister_profile(char *path)
665 proxy = __bt_gdbus_get_profile_proxy();
667 BT_ERR("Getting profile proxy failed");
671 ret = g_dbus_proxy_call_sync(proxy, "UnregisterProfile",
672 g_variant_new("(o)", path),
673 G_DBUS_CALL_FLAGS_NONE, -1,
676 BT_ERR("UnregisterProfile failed : %s", err->message);
681 g_variant_unref(ret);
686 GDBusNodeInfo * _bt_get_gdbus_node(const gchar *xml_data)
689 char *name = g_strdup_printf("org.bt.frwk%d", getpid());
691 bus_id = g_bus_own_name(G_BUS_TYPE_SYSTEM,
693 G_BUS_NAME_OWNER_FLAGS_NONE,
699 BT_DBG("Got bus id %d", bus_id);
703 return g_dbus_node_info_new_for_xml(xml_data, NULL);
706 int _bt_connect_profile(char *address, char *uuid, void *cb,
711 DBusGConnection *conn;
712 DBusGProxy *adapter_proxy;
713 GError *error = NULL;
715 conn = _bt_get_system_gconn();
716 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
718 object_path = _bt_get_device_object_path(address);
719 if (object_path == NULL) {
720 BT_ERR("No searched device");
722 adapter_proxy = _bt_get_adapter_proxy(conn);
723 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
725 dbus_g_proxy_call(adapter_proxy, "CreateDevice", &error,
726 G_TYPE_STRING, address,
727 G_TYPE_INVALID, G_TYPE_INVALID);
730 BT_ERR("CreateDevice Fail: %s", error->message);
734 object_path = _bt_get_device_object_path(address);
736 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
739 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
740 object_path, BT_DEVICE_INTERFACE);
742 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
744 if (!dbus_g_proxy_begin_call(proxy, "ConnectProfile",
745 (DBusGProxyCallNotify)cb,
749 BT_ERR("Connect Dbus Call Error");
750 g_object_unref(proxy);
751 return BLUETOOTH_ERROR_INTERNAL;
753 return BLUETOOTH_ERROR_NONE;
756 int _bt_discover_services(char *address, char *uuid, void *cb,
761 DBusGConnection *conn;
762 DBusGProxy *adapter_proxy;
763 GError *error = NULL;
767 conn = _bt_get_system_gconn();
768 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
770 object_path = _bt_get_device_object_path(address);
771 if (object_path == NULL) {
772 BT_ERR("No searched device");
774 adapter_proxy = _bt_get_adapter_proxy(conn);
775 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
777 dbus_g_proxy_call(adapter_proxy, "CreateDevice", &error,
778 G_TYPE_STRING, address,
779 G_TYPE_INVALID, G_TYPE_INVALID);
782 BT_ERR("CreateDevice Fail: %s", error->message);
786 object_path = _bt_get_device_object_path(address);
788 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
790 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
791 object_path, BT_DEVICE_INTERFACE);
793 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
795 if (!dbus_g_proxy_begin_call_with_timeout(proxy, "DiscoverServices",
796 (DBusGProxyCallNotify)cb,
797 func_data, NULL, timeout,
800 BT_ERR("Error: While calling DiscoverServices");
801 g_object_unref(proxy);
802 return BLUETOOTH_ERROR_INTERNAL;
805 return BLUETOOTH_ERROR_NONE;
808 int _bt_cancel_discovers(char *address)
812 DBusGConnection *conn;
814 conn = _bt_get_system_gconn();
815 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
817 object_path = _bt_get_device_object_path(address);
818 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
820 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
821 object_path, BT_DEVICE_INTERFACE);
823 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
824 if (!dbus_g_proxy_call(proxy,
827 G_TYPE_INVALID, G_TYPE_INVALID)) {
828 BT_ERR("Error: while CancelDiscovery");
829 g_object_unref(proxy);
830 return BLUETOOTH_ERROR_INTERNAL;
832 g_object_unref(proxy);
834 return BLUETOOTH_ERROR_NONE;
837 int _bt_discover_service_uuids(char *address, char *remote_uuid)
841 DBusGConnection *conn;
842 GHashTable *hash = NULL;
846 int result = BLUETOOTH_ERROR_INTERNAL;
847 retv_if(remote_uuid == NULL, BLUETOOTH_ERROR_INTERNAL);
849 conn = _bt_get_system_gconn();
850 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
852 object_path = _bt_get_device_object_path(address);
853 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
855 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
856 object_path, BT_PROPERTIES_INTERFACE);
859 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
861 if (!dbus_g_proxy_call(proxy, "GetAll", NULL,
862 G_TYPE_STRING, BT_DEVICE_INTERFACE,
864 dbus_g_type_get_map("GHashTable", G_TYPE_STRING,
865 G_TYPE_VALUE), &hash, G_TYPE_INVALID)) {
866 BT_ERR("Dbus error while GetAll");
867 g_object_unref(proxy);
868 return BLUETOOTH_ERROR_INTERNAL;
870 g_object_unref(proxy);
871 BT_DBG("Remote uuids %s", remote_uuid);
872 value = g_hash_table_lookup(hash, "UUIDs");
874 BT_ERR("No uuids found");
878 uuid_value = g_value_get_boxed(value);
879 if(uuid_value == NULL) {
880 BT_ERR("Error: while obtaining uuids");
884 for (i = 0; uuid_value[i] != NULL; i++) {
885 BT_DBG("Remote uuids %s", uuid_value[i]);
886 if (strcasecmp(uuid_value[i], remote_uuid) == 0) {
887 result = BLUETOOTH_ERROR_NONE;
891 BT_INFO("Specified uuid not found on remote device");
894 g_hash_table_destroy(hash);
899 int _bt_disconnect_profile(char *address, char *uuid, void *cb,
904 DBusGConnection *conn;
906 conn = _bt_get_system_gconn();
907 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
909 object_path = _bt_get_device_object_path(address);
910 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
913 proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
914 object_path, BT_DEVICE_INTERFACE);
916 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
918 if (!dbus_g_proxy_begin_call(proxy, "DisconnectProfile",
919 (DBusGProxyCallNotify)cb,
923 BT_ERR("Connect Dbus Call Error");
924 g_object_unref(proxy);
925 return BLUETOOTH_ERROR_INTERNAL;
927 return BLUETOOTH_ERROR_NONE;
930 int _bt_get_adapter_path(GDBusConnection *conn, char *path)
933 GDBusProxy *manager_proxy = NULL;
934 GVariant *result = NULL;
935 char *adapter_path = NULL;
937 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
939 manager_proxy = g_dbus_proxy_new_sync(conn,
940 G_DBUS_PROXY_FLAGS_NONE, NULL,
943 BT_MANAGER_INTERFACE,
946 if (!manager_proxy) {
947 BT_ERR("Unable to create proxy: %s", err->message);
951 result = g_dbus_proxy_call_sync(manager_proxy, "DefaultAdapter", NULL,
952 G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
955 BT_ERR("Fail to get DefaultAdapter (Error: %s)", err->message);
957 BT_ERR("Fail to get DefaultAdapter");
962 if (g_strcmp0(g_variant_get_type_string(result), "(o)")) {
963 BT_ERR("Incorrect result\n");
967 g_variant_get(result, "(&o)", &adapter_path);
969 if (adapter_path == NULL ||
970 strlen(adapter_path) >= BT_ADAPTER_OBJECT_PATH_MAX) {
971 BT_ERR("Adapter path is inproper\n");
976 g_strlcpy(path, adapter_path, BT_ADAPTER_OBJECT_PATH_MAX);
978 g_variant_unref(result);
979 g_object_unref(manager_proxy);
981 return BLUETOOTH_ERROR_NONE;
987 g_variant_unref(result);
990 g_object_unref(manager_proxy);
992 return BLUETOOTH_ERROR_INTERNAL;
996 void _bt_convert_device_path_to_address(const char *device_path,
997 char *device_address)
999 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
1002 ret_if(device_path == NULL);
1003 ret_if(device_address == NULL);
1005 dev_addr = strstr(device_path, "dev_");
1006 if (dev_addr != NULL) {
1009 g_strlcpy(address, dev_addr, sizeof(address));
1011 while ((pos = strchr(address, '_')) != NULL) {
1015 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
1019 static char *__bt_extract_adapter_path(DBusMessageIter *msg_iter)
1021 char *object_path = NULL;
1022 DBusMessageIter value_iter;
1024 /* Parse the signature: oa{sa{sv}}} */
1025 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
1026 DBUS_TYPE_OBJECT_PATH, NULL);
1028 dbus_message_iter_get_basic(msg_iter, &object_path);
1029 retv_if(object_path == NULL, NULL);
1031 /* object array (oa) */
1032 retv_if(dbus_message_iter_next(msg_iter) == FALSE, NULL);
1033 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
1034 DBUS_TYPE_ARRAY, NULL);
1036 dbus_message_iter_recurse(msg_iter, &value_iter);
1038 /* string array (sa) */
1039 while (dbus_message_iter_get_arg_type(&value_iter) ==
1040 DBUS_TYPE_DICT_ENTRY) {
1041 char *interface_name = NULL;
1042 DBusMessageIter interface_iter;
1044 dbus_message_iter_recurse(&value_iter, &interface_iter);
1046 retv_if(dbus_message_iter_get_arg_type(&interface_iter) !=
1047 DBUS_TYPE_STRING, NULL);
1049 dbus_message_iter_get_basic(&interface_iter, &interface_name);
1051 if (g_strcmp0(interface_name, "org.bluez.Adapter1") == 0) {
1052 /* Tizen don't allow the multi-adapter */
1053 BT_DBG("Found an adapter: %s", object_path);
1054 return g_strdup(object_path);
1057 dbus_message_iter_next(&value_iter);
1063 static char *__bt_extract_device_path(DBusMessageIter *msg_iter, char *address)
1065 char *object_path = NULL;
1066 char device_address[BT_ADDRESS_STRING_SIZE] = { 0 };
1068 /* Parse the signature: oa{sa{sv}}} */
1069 retv_if(dbus_message_iter_get_arg_type(msg_iter) !=
1070 DBUS_TYPE_OBJECT_PATH, NULL);
1072 dbus_message_iter_get_basic(msg_iter, &object_path);
1073 retv_if(object_path == NULL, NULL);
1075 _bt_convert_device_path_to_address(object_path, device_address);
1077 if (g_strcmp0(address, device_address) == 0) {
1078 return g_strdup(object_path);
1084 char *_bt_get_device_object_path(char *address)
1088 DBusMessageIter reply_iter;
1089 DBusMessageIter value_iter;
1091 DBusConnection *conn;
1092 char *object_path = NULL;
1094 conn = _bt_get_system_conn();
1095 retv_if(conn == NULL, NULL);
1097 msg = dbus_message_new_method_call(BT_BLUEZ_NAME, BT_MANAGER_PATH,
1098 BT_MANAGER_INTERFACE,
1099 "GetManagedObjects");
1101 retv_if(msg == NULL, NULL);
1103 /* Synchronous call */
1104 dbus_error_init(&err);
1105 reply = dbus_connection_send_with_reply_and_block(
1108 dbus_message_unref(msg);
1111 BT_ERR("Can't get managed objects");
1113 if (dbus_error_is_set(&err)) {
1114 BT_ERR("%s", err.message);
1115 dbus_error_free(&err);
1120 if (dbus_message_iter_init(reply, &reply_iter) == FALSE) {
1121 BT_ERR("Fail to iterate the reply");
1122 dbus_message_unref(reply);
1126 dbus_message_iter_recurse(&reply_iter, &value_iter);
1128 /* signature of GetManagedObjects: a{oa{sa{sv}}} */
1129 while (dbus_message_iter_get_arg_type(&value_iter) ==
1130 DBUS_TYPE_DICT_ENTRY) {
1131 DBusMessageIter msg_iter;
1133 dbus_message_iter_recurse(&value_iter, &msg_iter);
1135 object_path = __bt_extract_device_path(&msg_iter, address);
1136 if (object_path != NULL) {
1137 BT_DBG("Found the device path %s", object_path);
1141 dbus_message_iter_next(&value_iter);
1144 dbus_message_unref(reply);
1150 DBusGProxy *_bt_get_adapter_proxy(DBusGConnection *conn)
1152 DBusGProxy *adapter_proxy = NULL;
1154 retv_if(conn == NULL, NULL);
1156 adapter_proxy = dbus_g_proxy_new_for_name(conn, BT_BLUEZ_NAME,
1157 BT_BLUEZ_HCI_PATH, BT_PROPERTIES_INTERFACE);
1159 return adapter_proxy;
1162 void _bt_device_path_to_address(const char *device_path, char *device_address)
1164 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
1165 char *dev_addr = NULL;
1167 if (!device_path || !device_address)
1170 dev_addr = strstr(device_path, "dev_");
1171 if (dev_addr != NULL) {
1174 g_strlcpy(address, dev_addr, sizeof(address));
1176 while ((pos = strchr(address, '_')) != NULL) {
1180 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
1184 DBusGConnection *__bt_init_system_gconn(void)
1188 if (system_conn == NULL)
1189 system_conn = dbus_g_bus_get(DBUS_BUS_SYSTEM, NULL);
1194 DBusGConnection *_bt_get_system_gconn(void)
1196 return (system_conn) ? system_conn : __bt_init_system_gconn();
1199 GDBusConnection *_bt_init_system_gdbus_conn(void)
1202 GError *error = NULL;
1203 if (system_gdbus_conn == NULL) {
1205 g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
1207 BT_ERR("GDBus connection Error : %s \n",
1209 g_clear_error(&error);
1213 return system_gdbus_conn;
1216 DBusConnection *_bt_get_system_conn(void)
1218 DBusGConnection *g_conn;
1220 if (system_conn == NULL) {
1221 g_conn = __bt_init_system_gconn();
1223 g_conn = system_conn;
1226 retv_if(g_conn == NULL, NULL);
1228 return dbus_g_connection_get_connection(g_conn);
1231 int _bt_register_osp_server_in_agent(int type, char *uuid, char *path, int fd)
1234 char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
1235 char path_str[BLUETOOTH_PATH_STRING] = { 0, };
1239 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1241 g_array_append_vals(in_param1, &type, sizeof(int));
1242 g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
1243 g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
1244 g_strlcpy(path_str, path, sizeof(path_str));
1245 g_array_append_vals(in_param3, &path_str, BLUETOOTH_PATH_STRING);
1246 g_array_append_vals(in_param4, &fd, sizeof(int));
1248 ret = _bt_send_request(BT_AGENT_SERVICE, BT_SET_AUTHORIZATION,
1249 in_param1, in_param2, in_param3, in_param4, &out_param);
1251 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1256 int _bt_unregister_osp_server_in_agent(int type, char *uuid)
1259 char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
1260 char path_str[BLUETOOTH_PATH_STRING] = { 0, };
1264 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1266 g_array_append_vals(in_param1, &type, sizeof(int));
1267 g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
1268 g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
1270 ret = _bt_send_request(BT_AGENT_SERVICE, BT_UNSET_AUTHORIZATION,
1271 in_param1, in_param2, in_param3, in_param4, &out_param);
1273 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1278 int _bt_check_privilege(int service_type, int service_function)
1283 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1285 result = _bt_send_request(service_type, service_function,
1286 in_param1, in_param2, in_param3, in_param4, &out_param);
1288 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1293 GVariant *_bt_get_managed_objects(void)
1295 GDBusConnection *g_conn;
1296 GDBusProxy *manager_proxy = NULL;
1297 GVariant *result = NULL;
1298 GError *error = NULL;
1302 g_conn = _bt_gdbus_get_system_gconn();
1303 retv_if(g_conn == NULL, NULL);
1305 manager_proxy = g_dbus_proxy_new_sync(g_conn,
1306 G_DBUS_PROXY_FLAGS_NONE, NULL,
1309 BT_MANAGER_INTERFACE,
1313 BT_ERR("Unable to create proxy: %s", error->message);
1314 g_clear_error(&error);
1318 result = g_dbus_proxy_call_sync (manager_proxy,
1319 "GetManagedObjects", NULL,
1320 G_DBUS_CALL_FLAGS_NONE, -1,
1324 BT_ERR("Fail to get ManagedObjects (Error: %s)", error->message);
1325 g_clear_error(&error);
1328 g_object_unref(manager_proxy);
1334 BT_EXPORT_API int bluetooth_is_supported(void)
1336 int is_supported = 0;
1341 fd = open(RFKILL_NODE, O_RDONLY);
1343 BT_ERR("Fail to open RFKILL node");
1344 return BLUETOOTH_ERROR_INTERNAL;
1347 if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0) {
1348 BT_ERR("Fail to set RFKILL node to non-blocking");
1350 return BLUETOOTH_ERROR_INTERNAL;
1354 len = read(fd, &event, sizeof(event));
1356 BT_ERR("Fail to read events");
1360 if (len != RFKILL_EVENT_SIZE) {
1361 BT_ERR("The size is wrong\n");
1365 if (event.type == RFKILL_TYPE_BLUETOOTH) {
1373 BT_DBG("supported: %d", is_supported);
1375 return is_supported;
1378 BT_EXPORT_API int bluetooth_register_callback(bluetooth_cb_func_ptr callback_ptr, void *user_data)
1382 _bt_gdbus_init_system_gconn();
1383 __bt_init_system_gconn();
1385 ret = _bt_init_event_handler();
1386 if (ret != BLUETOOTH_ERROR_NONE &&
1387 ret != BLUETOOTH_ERROR_ALREADY_INITIALIZED) {
1388 BT_ERR("Fail to init the event handler");
1393 _bt_set_user_data(BT_COMMON, (void *)callback_ptr, user_data);
1395 /* Register All events */
1396 ret = _bt_register_event(BT_ADAPTER_EVENT, (void *)callback_ptr, user_data);
1397 if (ret != BLUETOOTH_ERROR_NONE)
1399 ret = _bt_register_event(BT_DEVICE_EVENT, (void *)callback_ptr, user_data);
1400 if (ret != BLUETOOTH_ERROR_NONE)
1402 ret = _bt_register_event(BT_NETWORK_EVENT, (void *)callback_ptr, user_data);
1403 if (ret != BLUETOOTH_ERROR_NONE)
1405 ret = _bt_register_event(BT_RFCOMM_CLIENT_EVENT, (void *)callback_ptr, user_data);
1406 if (ret != BLUETOOTH_ERROR_NONE)
1408 ret = _bt_register_event(BT_RFCOMM_SERVER_EVENT, (void *)callback_ptr, user_data);
1409 if (ret != BLUETOOTH_ERROR_NONE)
1412 _bt_register_name_owner_changed();
1414 return BLUETOOTH_ERROR_NONE;
1416 BT_ERR("Fail to do _bt_register_event()");
1417 bluetooth_unregister_callback();
1421 BT_EXPORT_API int bluetooth_unregister_callback(void)
1425 ret = _bt_deinit_event_handler();
1426 if (ret != BLUETOOTH_ERROR_NONE) {
1427 BT_ERR("Fail to deinit the event handler");
1430 _bt_unregister_name_owner_changed();
1432 _bt_set_user_data(BT_COMMON, NULL, NULL);
1435 dbus_g_connection_unref(system_conn);
1439 g_object_unref(system_gconn);
1440 system_gconn = NULL;
1442 return BLUETOOTH_ERROR_NONE;