2 * Copyright (c) 2011 Samsung Electronics Co., Ltd All Rights Reserved
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
21 #include <sys/types.h>
23 #include <sys/socket.h>
26 #include <gio/gunixfdlist.h>
29 #include "bluetooth-api.h"
30 #include "bluetooth-audio-api.h"
31 #include "bluetooth-hid-api.h"
32 #include "bluetooth-media-control.h"
33 #include "bt-internal-types.h"
34 #include "bluetooth-ipsp-api.h"
37 #include "bluetooth-gatt-server-api.h"
40 #include "bt-common.h"
41 #include "bt-request-sender.h"
42 #include "bt-event-handler.h"
44 #ifdef TIZEN_DPM_ENABLE
49 static bt_user_info_t user_info[BT_MAX_USER_INFO];
50 static GDBusConnection *system_gdbus_conn = NULL;
55 static GDBusConnection *system_gconn = NULL;
57 static gboolean bt_enabled = FALSE;
59 #define DBUS_TIMEOUT 20 * 1000 /* 20 Seconds */
61 GDBusConnection *g_bus_get_private_conn(void)
65 GDBusConnection *private_gconn = NULL;
67 address = g_dbus_address_get_for_bus_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
68 if (address == NULL) {
70 BT_ERR("Failed to get bus address: %s", error->message);
71 g_clear_error(&error);
76 private_gconn = g_dbus_connection_new_for_address_sync(address,
77 G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_CLIENT |
78 G_DBUS_CONNECTION_FLAGS_MESSAGE_BUS_CONNECTION,
79 NULL, /* GDBusAuthObserver */
84 BT_ERR("Unable to connect to dbus: %s", error->message);
85 g_clear_error(&error);
93 GDBusConnection *_bt_gdbus_init_system_gconn(void)
95 dbus_threads_init_default();
97 if (system_gconn != NULL)
100 system_gconn = g_bus_get_private_conn();
105 GDBusConnection *_bt_gdbus_get_system_gconn(void)
107 if (system_gconn == NULL)
108 system_gconn = _bt_gdbus_init_system_gconn();
109 else if (g_dbus_connection_is_closed(system_gconn))
110 system_gconn = g_bus_get_private_conn();
115 void _bt_print_device_address_t(const bluetooth_device_address_t *addr)
117 BT_DBG("%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X\n", addr->addr[0], addr->addr[1], addr->addr[2],
118 addr->addr[3], addr->addr[4], addr->addr[5]);
121 void _bt_set_user_data(int type, void *callback, void *user_data)
123 user_info[type].cb = callback;
124 user_info[type].user_data = user_data;
127 bt_user_info_t *_bt_get_user_data(int type)
129 return &user_info[type];
132 void _bt_common_event_cb(int event, int result, void *param,
133 void *callback, void *user_data)
135 bluetooth_event_param_t bt_event = { 0, };
136 bt_event.event = event;
137 bt_event.result = result;
138 bt_event.param_data = param;
141 ((bluetooth_cb_func_ptr)callback)(bt_event.event, &bt_event,
145 void _bt_input_event_cb(int event, int result, void *param,
146 void *callback, void *user_data)
148 hid_event_param_t bt_event = { 0, };
149 bt_event.event = event;
150 bt_event.result = result;
151 bt_event.param_data = param;
154 ((hid_cb_func_ptr)callback)(bt_event.event, &bt_event,
158 void _bt_headset_event_cb(int event, int result, void *param,
159 void *callback, void *user_data)
161 bt_audio_event_param_t bt_event = { 0, };
162 bt_event.event = event;
163 bt_event.result = result;
164 bt_event.param_data = param;
167 ((bt_audio_func_ptr)callback)(bt_event.event, &bt_event,
171 void _bt_a2dp_source_event_cb(int event, int result, void *param,
172 void *callback, void *user_data)
174 bt_audio_event_param_t bt_event = { 0, };
175 bt_event.event = event;
176 bt_event.result = result;
177 bt_event.param_data = param;
179 ((bt_audio_func_ptr)callback)(bt_event.event, &bt_event,
183 void _bt_hf_event_cb(int event, int result, void *param,
184 void *callback, void *user_data)
186 bt_hf_event_param_t bt_event = { 0, };
187 bt_event.event = event;
188 bt_event.result = result;
189 bt_event.param_data = param;
192 ((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,
210 void _bt_gatt_server_event_cb(int event, int result, void *param,
211 void *callback, void *user_data)
213 BT_INFO("__bt_gatt_server_event_cb");
214 gatt_server_event_param_t bt_event = { 0, 0, NULL, NULL };
215 bt_event.event = event;
216 bt_event.result = result;
217 bt_event.param_data = param;
219 BT_INFO("GATT Server event callback is registered");
220 ((gatt_server_cb_func_ptr)callback)(bt_event.event, &bt_event,
223 BT_ERR("GATT Server event callback is not registered!!!");
228 void _bt_divide_device_class(bluetooth_device_class_t *device_class,
231 ret_if(device_class == NULL);
233 device_class->major_class = (unsigned short)(cod & 0x00001F00) >> 8;
234 device_class->minor_class = (unsigned short)((cod & 0x000000FC));
235 device_class->service_class = (unsigned long)((cod & 0x00FF0000));
237 if (cod & 0x002000) {
238 device_class->service_class |=
239 BLUETOOTH_DEVICE_SERVICE_CLASS_LIMITED_DISCOVERABLE_MODE;
243 void _bt_convert_addr_string_to_type(unsigned char *addr,
249 ret_if(address == NULL);
250 ret_if(addr == NULL);
252 for (i = 0; i < BT_ADDRESS_LENGTH_MAX; i++) {
253 addr[i] = strtol(address, &ptr, 16);
254 if (ptr[0] != '\0') {
263 void _bt_convert_addr_string_to_secure_string(char *addr,
268 ret_if(address == NULL);
269 ret_if(addr == NULL);
271 len = strlen(address);
272 ret_if(len != BT_ADDRESS_STRING_SIZE - 1);
274 strncpy(addr, address, len);
283 void _bt_convert_addr_type_to_string(char *address,
286 ret_if(address == NULL);
287 ret_if(addr == NULL);
289 g_snprintf(address, BT_ADDRESS_STRING_SIZE,
290 "%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X",
291 addr[0], addr[1], addr[2],
292 addr[3], addr[4], addr[5]);
295 int _bt_copy_utf8_string(char *dest, const char *src, unsigned int length)
302 if (dest == NULL || src == NULL)
303 return BLUETOOTH_ERROR_INVALID_PARAM;
306 while (*p != '\0' && i < length) {
307 next = g_utf8_next_char(p);
310 while (count > 0 && ((i + count) < length)) {
317 return BLUETOOTH_ERROR_NONE;
320 gboolean _bt_utf8_validate(char *name)
324 glong items_written = 0;
326 if (FALSE == g_utf8_validate(name, -1, NULL))
329 u16 = g_utf8_to_utf16(name, -1, NULL, &items_written, NULL);
335 if (items_written != g_utf8_strlen(name, -1))
343 static GDBusProxy *profile_gproxy;
344 static GDBusConnection *gconn_g;
345 static int latest_id = -1;
346 #define BT_RFCOMM_ID_MAX 245
347 static gboolean id_used[BT_RFCOMM_ID_MAX];
348 GDBusNodeInfo *new_conn_node;
350 static const gchar rfcomm_agent_xml[] =
352 " <interface name='org.bluez.Profile1'>"
353 " <method name='NewConnection'>"
354 " <arg type='o' name='object' direction='in'/>"
355 " <arg type='h' name='fd' direction='in'/>"
356 " <arg type='a{sv}' name='properties' direction='in'/>"
358 " <method name='RequestDisconnection'>"
359 " <arg type='o' name='device' direction='in'/>"
364 static void __new_connection_method(GDBusConnection *connection,
366 const gchar *object_path,
367 const gchar *interface_name,
368 const gchar *method_name,
369 GVariant *parameters,
370 GDBusMethodInvocation *invocation,
373 BT_DBG("method %s", method_name);
374 if (g_strcmp0(method_name, "NewConnection") == 0) {
377 GUnixFDList *fd_list;
378 GVariantBuilder *properties;
381 bluetooth_device_address_t remote_addr1;
382 bt_new_connection_cb cb = user_data;
383 char secure_address[BT_ADDRESS_STRING_SIZE] = { 0 };
386 g_variant_get(parameters, "(oha{sv})", &obj_path, &index,
389 msg = g_dbus_method_invocation_get_message(invocation);
390 fd_list = g_dbus_message_get_unix_fd_list(msg);
391 if (fd_list == NULL) {
392 GQuark quark = g_quark_from_string("rfcomm-app");
393 GError *err = g_error_new(quark, 0, "No fd in message");
394 g_dbus_method_invocation_return_gerror(invocation, err);
400 fd = g_unix_fd_list_get(fd_list, index, NULL);
402 BT_ERR("Invalid fd return");
403 GQuark quark = g_quark_from_string("rfcomm-app");
404 GError *err = g_error_new(quark, 0, "Invalid FD return");
405 g_dbus_method_invocation_return_gerror(invocation, err);
410 _bt_convert_device_path_to_address(obj_path, addr);
411 _bt_convert_addr_string_to_type(remote_addr1.addr, (const char *)addr);
413 _bt_convert_addr_string_to_secure_string(secure_address, addr);
414 BT_INFO("fd: %d, address %s", fd, secure_address);
416 g_dbus_method_invocation_return_value(invocation, NULL);
419 cb(object_path, fd, &remote_addr1);
420 } else if (g_strcmp0(method_name, "RequestDisconnection") == 0) {
421 g_dbus_method_invocation_return_value(invocation, NULL);
426 static const GDBusInterfaceVTable method_table = {
427 __new_connection_method,
433 void _bt_swap_addr(unsigned char *dst, const unsigned char *src)
437 for (i = 0; i < 6; i++)
441 int __rfcomm_assign_id(void)
445 BT_DBG("latest_id: %d", latest_id);
447 index = latest_id + 1;
449 if (index >= BT_RFCOMM_ID_MAX)
452 BT_DBG("index: %d", index);
454 while (id_used[index] == TRUE) {
455 if (index == latest_id) {
456 /* No available ID */
457 BT_ERR("All request ID is used");
463 if (index >= BT_RFCOMM_ID_MAX)
468 id_used[index] = TRUE;
470 BT_DBG("Assigned Id: %d", latest_id);
475 void __rfcomm_delete_id(int id)
477 ret_if(id >= BT_RFCOMM_ID_MAX);
482 /* Next server will use this ID */
486 static GDBusConnection *__get_gdbus_connection()
489 gconn_g = g_bus_get_private_conn();
494 static GDBusProxy *__bt_gdbus_get_profile_proxy(void)
496 GDBusConnection *gconn;
500 return profile_gproxy;
502 gconn = __get_gdbus_connection();
506 profile_gproxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE,
509 "org.bluez.ProfileManager1",
512 BT_ERR("Unable to create proxy: %s", err->message);
517 return profile_gproxy;
520 static GDBusProxy *__bt_gdbus_get_device_proxy(char *object_path)
522 GDBusConnection *gconn;
524 GDBusProxy *device_gproxy;
526 gconn = __get_gdbus_connection();
530 device_gproxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE,
537 BT_ERR("Unable to create proxy: %s", err->message);
542 return device_gproxy;
545 void _bt_unregister_gdbus(int object_id)
547 GDBusConnection *gconn;
549 gconn = __get_gdbus_connection();
553 g_dbus_connection_unregister_object(gconn, object_id);
556 int _bt_register_new_conn(const char *path, bt_new_connection_cb cb)
558 GDBusConnection *gconn;
560 GError *error = NULL;
562 gconn = __get_gdbus_connection();
566 if (new_conn_node == NULL)
567 new_conn_node = _bt_get_gdbus_node(rfcomm_agent_xml);
569 if (new_conn_node == NULL)
572 id = g_dbus_connection_register_object(gconn, path,
573 new_conn_node->interfaces[0],
577 BT_ERR("Failed to register: %s", error->message);
582 BT_DBG("NEW CONNECTION ID %d", id);
587 static GDBusProxy * __bt_gdbus_get_adapter_proxy()
590 GDBusProxy *manager_proxy = NULL;
591 GDBusProxy *adapter_proxy = NULL;
592 GDBusConnection *conn;
593 GVariant *result = NULL;
594 char *adapter_path = NULL;
596 conn = __get_gdbus_connection();
597 retv_if(conn == NULL, NULL);
599 manager_proxy = g_dbus_proxy_new_sync(conn,
600 G_DBUS_PROXY_FLAGS_NONE, NULL,
603 BT_MANAGER_INTERFACE,
606 if (!manager_proxy) {
607 BT_ERR("Unable to create proxy: %s", err->message);
611 result = g_dbus_proxy_call_sync(manager_proxy, "DefaultAdapter", NULL,
612 G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
615 BT_ERR("Fail to get DefaultAdapter (Error: %s)", err->message);
617 BT_ERR("Fail to get DefaultAdapter");
622 if (g_strcmp0(g_variant_get_type_string(result), "(o)")) {
623 BT_ERR("Incorrect result\n");
627 g_variant_get(result, "(&o)", &adapter_path);
629 if (adapter_path == NULL ||
630 strlen(adapter_path) >= BT_ADAPTER_OBJECT_PATH_MAX) {
631 BT_ERR("Adapter path is inproper\n");
635 BT_INFO("Adapter Path %s", adapter_path);
637 adapter_proxy = g_dbus_proxy_new_sync(conn,
638 G_DBUS_PROXY_FLAGS_NONE, NULL,
641 BT_ADAPTER_INTERFACE,
644 BT_ERR("DBus Error message: [%s]", err->message);
650 g_object_unref(manager_proxy);
652 g_variant_unref(result);
653 return adapter_proxy;
656 int _bt_register_new_conn_ex(const char *path, const char *bus_name, bt_new_connection_cb cb)
658 GDBusConnection *gconn;
660 GError *error = NULL;
662 gconn = __get_gdbus_connection();
666 if (new_conn_node == NULL)
667 new_conn_node = _bt_get_gdbus_node_ex(rfcomm_agent_xml, bus_name);
669 if (new_conn_node == NULL)
672 id = g_dbus_connection_register_object(gconn, path,
673 new_conn_node->interfaces[0],
677 BT_ERR("Failed to register: %s", error->message);
682 BT_DBG("NEW CONNECTION ID %d", id);
687 int _bt_register_profile(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
689 GVariantBuilder *option_builder;
693 int result = BLUETOOTH_ERROR_NONE;
695 proxy = __bt_gdbus_get_profile_proxy();
697 BT_ERR("Getting profile proxy failed");
698 return BLUETOOTH_ERROR_INTERNAL;
701 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
702 if (info->authentication)
703 g_variant_builder_add(option_builder, "{sv}",
704 "RequireAuthentication",
705 g_variant_new_boolean(TRUE));
706 if (info->authorization)
707 g_variant_builder_add(option_builder, "{sv}",
708 "RequireAuthorization",
709 g_variant_new_boolean(TRUE));
711 g_variant_builder_add(option_builder, "{sv}",
713 g_variant_new_string(info->role));
715 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
716 * RFCOMM channels based on the availability when two services want
717 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
718 * two services use the same SPP RFCOMM channel. */
719 if (use_default_rfcomm)
720 g_variant_builder_add(option_builder, "{sv}",
722 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
724 g_variant_builder_add(option_builder, "{sv}",
726 g_variant_new_string(info->service));
728 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile",
729 g_variant_new("(osa{sv})", info->obj_path,
732 G_DBUS_CALL_FLAGS_NONE, -1,
735 g_dbus_error_strip_remote_error(err);
736 BT_ERR("RegisterProfile failed: %s", err->message);
738 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
739 result = BLUETOOTH_ERROR_ACCESS_DENIED;
741 result = BLUETOOTH_ERROR_INTERNAL;
746 g_variant_builder_unref(option_builder);
749 g_variant_unref(ret);
754 int _bt_register_profile_ex(bt_register_profile_info_t *info, gboolean use_default_rfcomm, const char *name, const char *path)
756 GVariantBuilder *option_builder;
760 int result = BLUETOOTH_ERROR_NONE;
762 proxy = __bt_gdbus_get_profile_proxy();
764 BT_ERR("Getting profile proxy failed");
765 return BLUETOOTH_ERROR_INTERNAL;
768 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
769 if (info->authentication)
770 g_variant_builder_add(option_builder, "{sv}",
771 "RequireAuthentication",
772 g_variant_new_boolean(TRUE));
773 if (info->authorization)
774 g_variant_builder_add(option_builder, "{sv}",
775 "RequireAuthorization",
776 g_variant_new_boolean(TRUE));
778 g_variant_builder_add(option_builder, "{sv}",
780 g_variant_new_string(info->role));
782 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
783 * RFCOMM channels based on the availability when two services want
784 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
785 * two services use the same SPP RFCOMM channel. */
786 if (use_default_rfcomm)
787 g_variant_builder_add(option_builder, "{sv}",
789 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
791 g_variant_builder_add(option_builder, "{sv}",
793 g_variant_new_string(info->service));
795 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile2",
796 g_variant_new("(osssa{sv})", info->obj_path,
801 G_DBUS_CALL_FLAGS_NONE, -1,
804 g_dbus_error_strip_remote_error(err);
805 BT_ERR("RegisterProfile failed: %s", err->message);
807 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
808 result = BLUETOOTH_ERROR_ACCESS_DENIED;
810 result = BLUETOOTH_ERROR_INTERNAL;
815 g_variant_builder_unref(option_builder);
818 g_variant_unref(ret);
823 int _bt_register_profile_platform(bt_register_profile_info_t *info, gboolean use_default_rfcomm)
825 GVariantBuilder *option_builder;
829 int result = BLUETOOTH_ERROR_NONE;
831 proxy = __bt_gdbus_get_profile_proxy();
833 BT_ERR("Getting profile proxy failed");
834 return BLUETOOTH_ERROR_INTERNAL;
837 option_builder = g_variant_builder_new(G_VARIANT_TYPE("a{sv}"));
838 if (info->authentication)
839 g_variant_builder_add(option_builder, "{sv}",
840 "RequireAuthentication",
841 g_variant_new_boolean(TRUE));
842 if (info->authorization)
843 g_variant_builder_add(option_builder, "{sv}",
844 "RequireAuthorization",
845 g_variant_new_boolean(TRUE));
847 g_variant_builder_add(option_builder, "{sv}",
849 g_variant_new_string(info->role));
851 /* Setting RFCOMM channel to default value 0; would allow bluez to assign
852 * RFCOMM channels based on the availability when two services want
853 * to use the RFCOMM along with SPP. Hence bluez makes sure that no
854 * two services use the same SPP RFCOMM channel. */
855 if (use_default_rfcomm)
856 g_variant_builder_add(option_builder, "{sv}",
858 g_variant_new_uint16(RFCOMM_DEFAULT_PROFILE_CHANNEL));
860 g_variant_builder_add(option_builder, "{sv}",
862 g_variant_new_string(info->service));
864 ret = g_dbus_proxy_call_sync(proxy, "RegisterProfile1",
865 g_variant_new("(osa{sv})", info->obj_path,
868 G_DBUS_CALL_FLAGS_NONE, -1,
872 g_dbus_error_strip_remote_error(err);
873 BT_ERR("RegisterProfile failed: %s", err->message);
875 if (g_strrstr(err->message, BT_ACCESS_DENIED_MSG))
876 result = BLUETOOTH_ERROR_ACCESS_DENIED;
878 result = BLUETOOTH_ERROR_INTERNAL;
883 g_variant_builder_unref(option_builder);
886 g_variant_unref(ret);
892 void _bt_unregister_profile(char *path)
898 proxy = __bt_gdbus_get_profile_proxy();
900 BT_ERR("Getting profile proxy failed");
904 ret = g_dbus_proxy_call_sync(proxy, "UnregisterProfile",
905 g_variant_new("(o)", path),
906 G_DBUS_CALL_FLAGS_NONE, -1,
909 BT_ERR("UnregisterProfile failed : %s", err->message);
914 g_variant_unref(ret);
919 GDBusNodeInfo * _bt_get_gdbus_node(const gchar *xml_data)
922 char *name = g_strdup_printf("org.bt.frwk%d", getpid());
924 bus_id = g_bus_own_name(G_BUS_TYPE_SYSTEM,
926 G_BUS_NAME_OWNER_FLAGS_NONE,
932 BT_DBG("Got bus id %d", bus_id);
936 return g_dbus_node_info_new_for_xml(xml_data, NULL);
939 GDBusNodeInfo * _bt_get_gdbus_node_ex(const gchar *xml_data, const char *bus_name)
942 char *name = g_strdup(bus_name);
943 bus_id = g_bus_own_name(G_BUS_TYPE_SYSTEM,
945 G_BUS_NAME_OWNER_FLAGS_NONE,
951 BT_DBG("Got bus id %d", bus_id);
955 return g_dbus_node_info_new_for_xml(xml_data, NULL);
958 int _bt_connect_profile(char *address, const char *uuid, void *cb,
962 GDBusProxy *adapter_proxy;
966 object_path = _bt_get_device_object_path(address);
968 if (object_path == NULL) {
969 GVariant *ret = NULL;
970 BT_ERR("No searched device");
971 adapter_proxy = __bt_gdbus_get_adapter_proxy();
973 if (adapter_proxy == NULL) {
974 BT_ERR("adapter proxy is NULL");
975 return BLUETOOTH_ERROR_INTERNAL;
978 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
979 g_variant_new("(s)", address),
980 G_DBUS_CALL_FLAGS_NONE,
985 BT_ERR("CreateDevice Failed: %s", err->message);
989 g_variant_unref(ret);
990 g_object_unref(adapter_proxy);
991 object_path = _bt_get_device_object_path(address);
993 if (object_path == NULL)
994 return BLUETOOTH_ERROR_INTERNAL;
997 proxy = __bt_gdbus_get_device_proxy(object_path);
1000 if (proxy == NULL) {
1001 BT_ERR("Error while getting proxy");
1002 return BLUETOOTH_ERROR_INTERNAL;
1005 g_dbus_proxy_call(proxy, "ConnectProfile",
1006 g_variant_new("(s)", uuid),
1007 G_DBUS_CALL_FLAGS_NONE,
1009 (GAsyncReadyCallback)cb,
1012 return BLUETOOTH_ERROR_NONE;
1015 int _bt_discover_services(char *address, char *uuid, void *cb,
1020 GDBusProxy *adapter_proxy;
1022 object_path = _bt_get_device_object_path(address);
1023 if (object_path == NULL) {
1024 GVariant *ret = NULL;
1025 BT_ERR("No searched device");
1026 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1027 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1028 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1029 g_variant_new("(s)", address),
1030 G_DBUS_CALL_FLAGS_NONE,
1034 BT_ERR("CreateDevice Failed: %s", err->message);
1035 g_clear_error(&err);
1038 g_variant_unref(ret);
1039 g_object_unref(adapter_proxy);
1040 object_path = _bt_get_device_object_path(address);
1041 if (object_path == NULL)
1042 return BLUETOOTH_ERROR_INTERNAL;
1044 proxy = __bt_gdbus_get_device_proxy(object_path);
1045 g_free(object_path);
1046 if (proxy == NULL) {
1047 BT_ERR("Error while getting proxy");
1048 return BLUETOOTH_ERROR_INTERNAL;
1050 g_dbus_proxy_call(proxy, "DiscoverServices",
1051 g_variant_new("(s)", uuid),
1052 G_DBUS_CALL_FLAGS_NONE,
1054 (GAsyncReadyCallback)cb,
1057 return BLUETOOTH_ERROR_NONE;
1060 int _bt_cancel_discovers(char *address)
1064 GDBusProxy *adapter_proxy;
1065 GVariant *ret = NULL;
1067 object_path = _bt_get_device_object_path(address);
1068 if (object_path == NULL) {
1069 BT_ERR("No searched device");
1070 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1071 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1072 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1073 g_variant_new("(s)", address),
1074 G_DBUS_CALL_FLAGS_NONE,
1078 BT_ERR("CreateDevice Failed: %s", err->message);
1079 g_clear_error(&err);
1082 g_variant_unref(ret);
1083 g_object_unref(adapter_proxy);
1084 object_path = _bt_get_device_object_path(address);
1085 if (object_path == NULL)
1086 return BLUETOOTH_ERROR_INTERNAL;
1088 proxy = __bt_gdbus_get_device_proxy(object_path);
1089 g_free(object_path);
1090 ret = g_dbus_proxy_call_sync(proxy, "CancelDiscovery",
1092 G_DBUS_CALL_FLAGS_NONE,
1096 BT_ERR("DBus Error message: [%s]", err->message);
1097 g_clear_error(&err);
1098 return BLUETOOTH_ERROR_INTERNAL;
1101 g_variant_unref(ret);
1103 g_object_unref(proxy);
1104 return BLUETOOTH_ERROR_NONE;
1107 int _bt_discover_service_uuids(char *address, char *remote_uuid)
1111 GDBusConnection *gconn;
1113 char **uuid_value = NULL;
1116 GVariant *value = NULL;
1117 GVariant *ret = NULL;
1118 int result = BLUETOOTH_ERROR_INTERNAL;
1120 retv_if(remote_uuid == NULL, BLUETOOTH_ERROR_INTERNAL);
1121 gconn = __get_gdbus_connection();
1122 retv_if(gconn == NULL, BLUETOOTH_ERROR_INTERNAL);
1123 object_path = _bt_get_device_object_path(address);
1124 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
1126 proxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE, NULL,
1127 BT_BLUEZ_NAME, object_path, BT_PROPERTIES_INTERFACE, NULL,
1129 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1131 BT_ERR("DBus Error: [%s]", err->message);
1132 g_clear_error(&err);
1134 ret = g_dbus_proxy_call_sync(proxy, "GetAll",
1135 g_variant_new("(s)", BT_DEVICE_INTERFACE),
1136 G_DBUS_CALL_FLAGS_NONE,
1140 result = BLUETOOTH_ERROR_INTERNAL;
1141 BT_ERR("DBus Error : %s", err->message);
1142 g_clear_error(&err);
1146 BT_ERR("g_dbus_proxy_call_sync function return NULL");
1147 result = BLUETOOTH_ERROR_INTERNAL;
1151 g_variant_get(ret, "(@a{sv})", &value);
1152 g_variant_unref(ret);
1154 GVariant *temp_value = g_variant_lookup_value(value, "UUIDs",
1155 G_VARIANT_TYPE_STRING_ARRAY);
1158 size = g_variant_get_size(temp_value);
1160 uuid_value = (char **)g_variant_get_strv(temp_value, &size);
1161 BT_DBG("Size items %d", size);
1164 for (i = 0; uuid_value[i] != NULL; i++) {
1165 BT_DBG("Remote uuids %s", uuid_value[i]);
1166 if (strcasecmp(uuid_value[i], remote_uuid) == 0) {
1167 result = BLUETOOTH_ERROR_NONE;
1168 g_variant_unref(temp_value);
1174 g_variant_unref(temp_value);
1179 g_object_unref(proxy);
1181 g_variant_unref(value);
1189 int _bt_get_cod_by_address(char *address, bluetooth_device_class_t *dev_class)
1193 GDBusConnection *gconn;
1195 GVariant *value = NULL;
1196 GVariant *result = NULL;
1197 unsigned int class = 0x00;
1198 int ret = BLUETOOTH_ERROR_INTERNAL;
1200 gconn = __get_gdbus_connection();
1201 retv_if(gconn == NULL, BLUETOOTH_ERROR_INTERNAL);
1202 object_path = _bt_get_device_object_path(address);
1204 retv_if(object_path == NULL, BLUETOOTH_ERROR_INTERNAL);
1206 proxy = g_dbus_proxy_new_sync(gconn, G_DBUS_PROXY_FLAGS_NONE, NULL,
1207 BT_BLUEZ_NAME, object_path, BT_PROPERTIES_INTERFACE, NULL,
1209 retv_if(proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1211 BT_ERR("DBus Error: [%s]", err->message);
1212 g_clear_error(&err);
1215 result = g_dbus_proxy_call_sync(proxy, "GetAll",
1216 g_variant_new("(s)", BT_DEVICE_INTERFACE),
1217 G_DBUS_CALL_FLAGS_NONE,
1221 ret = BLUETOOTH_ERROR_INTERNAL;
1222 BT_ERR("DBus Error : %s", err->message);
1223 g_clear_error(&err);
1226 if (result == NULL) {
1227 BT_ERR("g_dbus_proxy_call_sync function return NULL");
1228 ret = BLUETOOTH_ERROR_INTERNAL;
1231 g_variant_get(result, "(@a{sv})", &value);
1232 g_variant_unref(result);
1234 GVariant *temp_value = g_variant_lookup_value(value, "Class",
1235 G_VARIANT_TYPE_UINT32);
1236 class = g_variant_get_uint32(temp_value);
1237 _bt_divide_device_class(dev_class, class);
1239 g_variant_unref(temp_value);
1244 g_object_unref(proxy);
1246 g_variant_unref(value);
1252 int _bt_disconnect_profile(char *address, const char *uuid, void *cb,
1258 GDBusProxy *adapter_proxy;
1259 object_path = _bt_get_device_object_path(address);
1260 if (object_path == NULL) {
1261 GVariant *ret = NULL;
1262 BT_ERR("No searched device");
1263 adapter_proxy = __bt_gdbus_get_adapter_proxy();
1264 retv_if(adapter_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
1265 ret = g_dbus_proxy_call_sync(adapter_proxy, "CreateDevice",
1266 g_variant_new("(s)", address),
1267 G_DBUS_CALL_FLAGS_NONE,
1271 BT_ERR("CreateDevice Failed: %s", err->message);
1275 g_variant_unref(ret);
1276 g_object_unref(adapter_proxy);
1277 object_path = _bt_get_device_object_path(address);
1278 if (object_path == NULL)
1279 return BLUETOOTH_ERROR_INTERNAL;
1281 proxy = __bt_gdbus_get_device_proxy(object_path);
1282 g_free(object_path);
1283 if (proxy == NULL) {
1284 BT_ERR("Error while getting proxy");
1285 return BLUETOOTH_ERROR_INTERNAL;
1287 g_dbus_proxy_call(proxy, "DisconnectProfile",
1288 g_variant_new("(s)", uuid),
1289 G_DBUS_CALL_FLAGS_NONE,
1291 (GAsyncReadyCallback)cb,
1294 return BLUETOOTH_ERROR_NONE;
1297 int _bt_get_adapter_path(GDBusConnection *conn, char *path)
1300 GDBusProxy *manager_proxy = NULL;
1301 GVariant *result = NULL;
1302 char *adapter_path = NULL;
1304 retv_if(conn == NULL, BLUETOOTH_ERROR_INTERNAL);
1306 manager_proxy = g_dbus_proxy_new_sync(conn,
1307 G_DBUS_PROXY_FLAGS_NONE, NULL,
1310 BT_MANAGER_INTERFACE,
1313 if (!manager_proxy) {
1314 BT_ERR("Unable to create proxy: %s", err->message);
1318 result = g_dbus_proxy_call_sync(manager_proxy, "DefaultAdapter", NULL,
1319 G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
1322 if (!g_strrstr(err->message, "ServiceUnknown"))
1323 BT_ERR("Fail to get DefaultAdapter (Error: %s)", err->message);
1325 BT_ERR("Fail to get DefaultAdapter");
1331 if (g_strcmp0(g_variant_get_type_string(result), "(o)")) {
1332 BT_ERR("Incorrect result\n");
1336 g_variant_get(result, "(&o)", &adapter_path);
1338 if (adapter_path == NULL ||
1339 strlen(adapter_path) >= BT_ADAPTER_OBJECT_PATH_MAX) {
1340 BT_ERR("Adapter path is inproper\n");
1345 g_strlcpy(path, adapter_path, BT_ADAPTER_OBJECT_PATH_MAX);
1347 g_variant_unref(result);
1348 g_object_unref(manager_proxy);
1350 return BLUETOOTH_ERROR_NONE;
1353 g_clear_error(&err);
1356 g_variant_unref(result);
1359 g_object_unref(manager_proxy);
1361 return BLUETOOTH_ERROR_INTERNAL;
1365 void _bt_convert_device_path_to_address(const char *device_path,
1366 char *device_address)
1368 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
1371 ret_if(device_path == NULL);
1372 ret_if(device_address == NULL);
1374 dev_addr = strstr(device_path, "dev_");
1375 if (dev_addr != NULL) {
1378 g_strlcpy(address, dev_addr, sizeof(address));
1380 while ((pos = strchr(address, '_')) != NULL)
1383 g_strlcpy(device_address, address, BT_ADDRESS_STRING_SIZE);
1387 static char *__bt_extract_device_path(GVariantIter *iter, char *address)
1389 char *object_path = NULL;
1390 char device_address[BT_ADDRESS_STRING_SIZE] = { 0 };
1391 /* Parse the signature: oa{sa{sv}}} */
1392 while (g_variant_iter_loop(iter, "{&oa{sa{sv}}}", &object_path,
1394 retv_if(object_path == NULL, NULL);
1395 _bt_convert_device_path_to_address(object_path, device_address);
1397 if (g_strcmp0(address, device_address) == 0)
1398 return g_strdup(object_path);
1403 char *_bt_get_device_object_path(char *address)
1406 GDBusProxy *proxy = NULL;
1407 GVariant *result = NULL;
1408 GVariantIter *iter = NULL;
1409 GDBusConnection *conn = NULL;
1410 char *object_path = NULL;
1412 conn = _bt_gdbus_get_system_gconn();
1413 retv_if(conn == NULL, NULL);
1415 proxy = g_dbus_proxy_new_sync(conn,
1416 G_DBUS_PROXY_FLAGS_NONE, NULL,
1419 BT_MANAGER_INTERFACE,
1423 BT_ERR("Unable to create proxy: %s", err->message);
1427 result = g_dbus_proxy_call_sync(proxy, "GetManagedObjects", NULL,
1428 G_DBUS_CALL_FLAGS_NONE, -1, NULL, &err);
1431 BT_ERR("Fail to get GetManagedObjects (Error: %s)", err->message);
1433 BT_ERR("Fail to get GetManagedObjects");
1438 g_variant_get(result, "(a{oa{sa{sv}}})", &iter);
1439 object_path = __bt_extract_device_path(iter, address);
1441 g_variant_unref(result);
1442 g_object_unref(proxy);
1443 g_variant_iter_free(iter);
1447 g_clear_error(&err);
1450 g_object_unref(proxy);
1455 GDBusConnection *_bt_init_system_gdbus_conn(void)
1457 GError *error = NULL;
1458 if (system_gdbus_conn == NULL) {
1460 g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
1462 BT_ERR("GDBus connection Error : %s \n",
1464 g_clear_error(&error);
1468 return system_gdbus_conn;
1472 int _bt_register_osp_server_in_agent(int type, char *uuid, char *path, int fd)
1475 char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
1476 char path_str[BLUETOOTH_PATH_STRING] = { 0, };
1480 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1482 g_array_append_vals(in_param1, &type, sizeof(int));
1483 g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
1484 g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
1485 g_strlcpy(path_str, path, sizeof(path_str));
1486 g_array_append_vals(in_param3, &path_str, BLUETOOTH_PATH_STRING);
1487 g_array_append_vals(in_param4, &fd, sizeof(int));
1489 ret = _bt_send_request(BT_AGENT_SERVICE, BT_SET_AUTHORIZATION,
1490 in_param1, in_param2, in_param3, in_param4, &out_param);
1492 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1497 int _bt_unregister_osp_server_in_agent(int type, char *uuid)
1500 char uuid_str[BLUETOOTH_UUID_STRING_MAX] = { 0, };
1504 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1506 g_array_append_vals(in_param1, &type, sizeof(int));
1507 g_strlcpy(uuid_str, uuid, sizeof(uuid_str));
1508 g_array_append_vals(in_param2, &uuid_str, BLUETOOTH_UUID_STRING_MAX);
1510 ret = _bt_send_request(BT_AGENT_SERVICE, BT_UNSET_AUTHORIZATION,
1511 in_param1, in_param2, in_param3, in_param4, &out_param);
1513 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1518 int _bt_check_privilege(int service_type, int service_function)
1522 BT_CHECK_ENABLED(return);
1525 BT_ALLOC_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1527 result = _bt_send_request(service_type, service_function,
1528 in_param1, in_param2, in_param3, in_param4, &out_param);
1530 BT_FREE_PARAMS(in_param1, in_param2, in_param3, in_param4, out_param);
1535 GVariant *_bt_get_managed_objects(void)
1537 GDBusConnection *g_conn;
1538 GDBusProxy *manager_proxy = NULL;
1539 GVariant *result = NULL;
1540 GError *error = NULL;
1544 g_conn = _bt_gdbus_get_system_gconn();
1545 retv_if(g_conn == NULL, NULL);
1547 manager_proxy = g_dbus_proxy_new_sync(g_conn,
1548 G_DBUS_PROXY_FLAGS_NONE, NULL,
1551 BT_MANAGER_INTERFACE,
1555 BT_ERR("Unable to create proxy: %s", error->message);
1556 g_clear_error(&error);
1560 result = g_dbus_proxy_call_sync(manager_proxy,
1561 "GetManagedObjects", NULL,
1562 G_DBUS_CALL_FLAGS_NONE, -1,
1566 BT_ERR("Fail to get ManagedObjects (Error: %s)", error->message);
1567 g_clear_error(&error);
1570 g_object_unref(manager_proxy);
1576 gboolean _bt_check_enabled_internal(void)
1578 if (bt_enabled == TRUE)
1581 if (bluetooth_check_adapter() == BLUETOOTH_ADAPTER_ENABLED)
1587 void _bt_set_adapter_internal_status(gboolean enabled)
1589 bt_enabled = enabled;
1592 int _bt_get_uuid_specification_name(const char *uuid, char **name)
1599 const char *specification_name;
1600 } bt_uuid_name[] = {
1602 {"1800", "Generic Access"},
1603 {"1801", "Generic Attribute"},
1604 {"1802", "Immediate Alert"},
1605 {"1803", "Link Loss"},
1606 {"1804", "Tx Power"},
1607 {"1805", "Current Time Service"},
1608 {"1806", "Reference Time Update Service"},
1609 {"1807", "Next DST Change Service"},
1610 {"1808", "Glucose"},
1611 {"1809", "Health Thermometer"},
1612 {"180A", "Device Information"},
1613 {"180D", "Heart Rate"},
1614 {"180F", "Battery Service"},
1615 {"1810", "Blood Pressure"},
1616 {"1811", "Alert Notification Service"},
1617 {"1812", "Human Interface Device"},
1619 /* GATT Declarations */
1620 {"2800", "Primary Service Declaration"},
1621 {"2801", "Secondary Service Declaration"},
1622 {"2802", "Include Declaration"},
1623 {"2803", "Characteristic Declaration"},
1625 /* GATT Descriptors */
1626 {"2900", "Characteristic Extended Properties"},
1627 {"2901", "Characteristic User Description"},
1628 {"2902", "Client Characteristic Configuration"},
1629 {"2903", "Server Characteristic Configuration"},
1630 {"2904", "Characteristic Format"},
1631 {"2905", "Characteristic Aggregate Formate"},
1632 {"2906", "Valid Range"},
1633 {"2907", "External Report Reference"},
1634 {"2908", "Report Reference"},
1636 /* GATT Characteristics */
1637 {"2A00", "Device Name"},
1638 {"2A01", "Appearance"},
1639 {"2A02", "Peripheral Privacy Flag"},
1640 {"2A03", "Reconnection Address"},
1641 {"2A04", "Peripheral Preferred Connection Parameters"},
1642 {"2A05", "Service Changed"},
1643 {"2A06", "Alert Level"},
1644 {"2A07", "Tx Power Level"},
1645 {"2A08", "Date Time"},
1646 {"2A09", "Day of Week"},
1647 {"2A0A", "Day Date Time"},
1648 {"2A19", "Battery Level"},
1649 {"2A1E", "Intermediate Temperature"},
1650 {"2A23", "System ID"},
1651 {"2A24", "Model Number String"},
1652 {"2A25", "Serial Number String"},
1653 {"2A26", "Firmware Revision String"},
1654 {"2A27", "Hardware Revision String"},
1655 {"2A28", "Software Revision String"},
1656 {"2A29", "Manufacturer Name String"},
1657 {"2A2A", "IEEE 11073-20601 Regulatory Certification Data List"},
1658 {"2A2B", "Current Time"},
1659 {"2A37", "Heart Rate Measurement"},
1660 {"2A38", "Body Sensor Location"},
1661 {"2A3F", "Alert Status"},
1662 {"2A46", "New Alert"},
1663 {"2A4A", "HID Information"},
1664 {"2A4C", "HID Control Point"},
1668 {"7905F431-B5CE-4E99-A40F-4B1E122D00D0", "Apple Notification Center Service"},
1669 {"9FBF120D-6301-42D9-8C58-25E699A21DBD", "Notifications Source"},
1670 {"69D1D8F3-45E1-49A8-9821-9BBDFDAAD9D9", "Control Point"},
1671 {"22EAC6E9-24D6-4BB5-BE44-B36ACE7C7BFB", "Data Source"},
1672 {"89D3502B-0F36-433A-8EF4-C502AD55F8DC", "Apple Media Service"},
1673 {"9B3C81D8-57B1-4A8A-B8DF-0E56F7CA51C2", "Remote Command"},
1674 {"2F7CABCE-808D-411F-9A0C-BB92BA96C102", "Entity Update"},
1675 {"C6B2F38C-23AB-46D8-A6AB-A3A870BBD5D7", "Entity Attribute"},
1676 {"9A3F68E0-86CE-11E5-A309-0002A5D5C51B", "Samsung Gear Manager Service"},
1677 {"c2f2cc0f-c085-4dd4-be5a-aca3074bbc72", "Control Point"},
1678 {"cece518b-28d7-4171-92d5-76a1e249a3b9", "Notifications Source"},
1683 return BLUETOOTH_ERROR_INVALID_PARAM;
1684 if (strlen(uuid) == 36) {
1685 if (!g_ascii_strncasecmp(uuid + 9, "0000-1000-8000-00805F9B34FB", 27))
1691 } else if (strlen(uuid) >= 8)
1694 for (i = 0; bt_uuid_name[i].uuid; i++) {
1695 if (!g_ascii_strncasecmp(uuid + offset, bt_uuid_name[i].uuid, uuid_len)) {
1696 *name = g_strdup(bt_uuid_name[i].specification_name);
1697 return BLUETOOTH_ERROR_NONE;
1701 *name = g_strdup("Unknown");
1702 return BLUETOOTH_ERROR_NONE;
1705 BT_EXPORT_API int bluetooth_is_supported(void)
1707 int is_supported = 0;
1712 fd = open(RFKILL_NODE, O_RDONLY);
1714 BT_ERR("Fail to open RFKILL node");
1715 return BLUETOOTH_ERROR_INTERNAL;
1718 if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0) {
1719 BT_ERR("Fail to set RFKILL node to non-blocking");
1721 return BLUETOOTH_ERROR_INTERNAL;
1725 len = read(fd, &event, sizeof(event));
1727 BT_ERR("Fail to read events");
1731 if (len != RFKILL_EVENT_SIZE) {
1732 BT_ERR("The size is wrong\n");
1736 if (event.type == RFKILL_TYPE_BLUETOOTH) {
1744 BT_DBG("supported: %d", is_supported);
1746 return is_supported;
1749 BT_EXPORT_API int bluetooth_register_callback(bluetooth_cb_func_ptr callback_ptr, void *user_data)
1753 _bt_gdbus_init_system_gconn();
1755 ret = _bt_init_event_handler();
1756 if (ret != BLUETOOTH_ERROR_NONE &&
1757 ret != BLUETOOTH_ERROR_ALREADY_INITIALIZED) {
1758 BT_ERR("Fail to init the event handler");
1763 _bt_set_user_data(BT_COMMON, (void *)callback_ptr, user_data);
1765 /* Register All events */
1766 if (ret != BLUETOOTH_ERROR_ALREADY_INITIALIZED) {
1767 ret = _bt_register_event(BT_ADAPTER_EVENT, (void *)callback_ptr, user_data);
1768 if (ret != BLUETOOTH_ERROR_NONE)
1770 ret = _bt_register_event(BT_DEVICE_EVENT, (void *)callback_ptr, user_data);
1771 if (ret != BLUETOOTH_ERROR_NONE)
1773 ret = _bt_register_event(BT_NETWORK_EVENT, (void *)callback_ptr, user_data);
1774 if (ret != BLUETOOTH_ERROR_NONE)
1776 ret = _bt_register_event(BT_RFCOMM_CLIENT_EVENT, (void *)callback_ptr, user_data);
1777 if (ret != BLUETOOTH_ERROR_NONE)
1779 ret = _bt_register_event(BT_RFCOMM_SERVER_EVENT, (void *)callback_ptr, user_data);
1780 if (ret != BLUETOOTH_ERROR_NONE)
1783 #ifdef GATT_NO_RELAY
1784 ret = _bt_register_event(BT_GATT_BLUEZ_EVENT, (void *)callback_ptr, user_data);
1785 if (ret != BLUETOOTH_ERROR_NONE)
1788 ret = _bt_register_event(BT_HDP_EVENT, (void *)callback_ptr, user_data);
1789 if (ret != BLUETOOTH_ERROR_NONE)
1793 _bt_register_name_owner_changed();
1795 return BLUETOOTH_ERROR_NONE;
1797 BT_ERR("Fail to do _bt_register_event()");
1798 bluetooth_unregister_callback();
1802 BT_EXPORT_API int bluetooth_unregister_callback(void)
1807 ret = _bt_deinit_event_handler();
1808 if (ret != BLUETOOTH_ERROR_NONE)
1809 BT_ERR("Fail to deinit the event handler");
1811 _bt_unregister_name_owner_changed();
1813 _bt_set_user_data(BT_COMMON, NULL, NULL);
1816 g_object_unref(system_gconn);
1817 system_gconn = NULL;
1819 return BLUETOOTH_ERROR_NONE;