4 * Copyright (c) 2019 Samsung Electronics Co., Ltd. All rights reserved.
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
10 * http://www.apache.org/licenses/LICENSE-2.0
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
20 #include <sys/errno.h>
25 #include "oal-hardware.h"
26 #include "oal-manager.h"
27 #include "oal-audio-src.h"
28 #include "oal-device-mgr.h"
29 #include "oal-hf-client.h"
30 #include "bt-service-common.h"
32 #include <bt-service-audio-common.h>
33 #include <bt-service-hf-client.h>
34 #include <bt-service-event.h>
36 static void __bt_reply_hf_client_disconnection_pending_request(bt_address_t *address)
38 char addr[BT_ADDRESS_STRING_SIZE] = { 0 };
39 bluetooth_device_address_t device_address;
41 invocation_info_t *req_info;
42 int result = BLUETOOTH_ERROR_NONE;
44 ret_if(NULL == address);
46 memcpy(device_address.addr, address->addr, BLUETOOTH_ADDRESS_LENGTH);
47 _bt_convert_addr_type_to_string(addr, address->addr);
49 BT_DBG("HF Client Disconnection remote device [%s]", addr);
51 /* Find Async request information*/
52 req_info = _bt_get_request_info_data(BT_HF_DISCONNECT, addr);
53 if (NULL == req_info) {
54 BT_INFO("HF DisConnect request not found or possibly already replied..");
56 /* if HF Connect call fails on Bluez DBUS, Async callback from bluez will be received
57 __bt_hf_client_connect_cb) which will internally trigger HF DISCONNECTED event */
58 req_info = _bt_get_request_info_data(BT_HF_CONNECT, addr);
59 if (NULL == req_info) {
60 BT_DBG("HF Connect request not found..");
63 BT_DBG("HF Connect request found..reply this request");
66 BT_DBG("HF DisConnect request found..reply this request");
69 /* In any of the above cases, reply DBUS context */
70 out_param = g_array_new(FALSE, FALSE, sizeof(gchar));
71 g_array_append_vals(out_param, &device_address, sizeof(bluetooth_device_address_t));
72 _bt_service_method_return(req_info->context, out_param, result);
73 g_array_free(out_param, TRUE);
74 _bt_free_info_from_invocation_list(req_info);
77 static void __bt_reply_hf_client_connection_pending_request(bt_address_t *address)
79 char addr[BT_ADDRESS_STRING_SIZE] = { 0 };
80 bluetooth_device_address_t device_address;
82 invocation_info_t *req_info;
83 int result = BLUETOOTH_ERROR_NONE;
85 ret_if(NULL == address);
87 memcpy(device_address.addr, address->addr, BLUETOOTH_ADDRESS_LENGTH);
88 _bt_convert_addr_type_to_string(addr, address->addr);
90 BT_DBG("HF Client Connection remote device [%s]", addr);
92 /* Find Async request information*/
93 req_info = _bt_get_request_info_data(BT_HF_CONNECT, addr);
94 if (NULL == req_info) {
95 BT_DBG("HF Connect request not found or possibly already replied..");
99 BT_DBG("HF Connect request found..");
102 /* In any of the above cases, reply DBUS context */
103 out_param = g_array_new(FALSE, FALSE, sizeof(gchar));
104 g_array_append_vals(out_param, &device_address, sizeof(bluetooth_device_address_t));
105 _bt_service_method_return(req_info->context, out_param, result);
106 g_array_free(out_param, TRUE);
107 _bt_free_info_from_invocation_list(req_info);
112 /* This event handler process events for HF Client role */
113 void _bt_hf_client_event_handler(int oal_event, gpointer event_data)
115 bt_address_t *bt_addr = event_data;
116 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
119 case OAL_EVENT_HF_CLIENT_DISCONNECTED:
120 /* Reply to async request for HF Client connect or disconnect request, if any */
121 BT_INFO("HF client Profile disconnected, reply pending DBUS request and check waiting device");
122 __bt_reply_hf_client_disconnection_pending_request(bt_addr);
123 BT_PERMANENT_LOG("Disconnected HF");
125 _bt_convert_addr_type_to_string(address, bt_addr->addr);
126 _bt_remove_headset_from_list(BT_AUDIO_AG, address);
128 case OAL_EVENT_HF_CLIENT_CONNECTED:
129 BT_INFO("HF Client Profile connected, Event & DBUS context will be handled at finalizing SCO connect..");
130 __bt_reply_hf_client_connection_pending_request(bt_addr);
131 BT_PERMANENT_LOG("Connected HF");
133 _bt_convert_addr_type_to_string(address, bt_addr->addr);
134 _bt_add_headset_to_list(BT_AUDIO_AG, BT_STATE_CONNECTED, address);
136 case OAL_EVENT_HF_CLIENT_CONNECTING:
137 BT_INFO("HF Client Profile connection successful, wait for Audio connect..");
139 case OAL_EVENT_HF_CLIENT_DISCONNECTING:
140 BT_INFO("HF Client Connecting or Disconnecting..No need to send event to app");
147 int _bt_connect_remote_ag(bluetooth_device_address_t *device_address)
149 oal_status_t status = OAL_STATUS_SUCCESS;
150 int result = BLUETOOTH_ERROR_NONE;
151 gboolean is_connected = FALSE;
153 is_connected = device_get_svc_conn_state((bt_address_t*)device_address, HFP_SERVICE_ID);
154 if (is_connected == TRUE) {
155 BT_ERR("HF Client is already connected");
156 return BLUETOOTH_ERROR_ALREADY_CONNECT;
159 status = hf_client_connect((bt_address_t *)device_address);
160 if (status != OAL_STATUS_SUCCESS) {
161 BT_ERR("HF Client Connection could not be established, err: [%d]", status);
162 result = _bt_convert_oal_status_to_bt_error(status);
167 int _bt_disconnect_remote_ag(bluetooth_device_address_t *device_address)
169 oal_status_t status = OAL_STATUS_SUCCESS;
170 int result = BLUETOOTH_ERROR_NONE;
172 status = hf_client_disconnect((bt_address_t *)device_address);
173 if (status != OAL_STATUS_SUCCESS) {
174 BT_ERR("HF Client DisConnection err: [%d]", status);
175 result = _bt_convert_oal_status_to_bt_error(status);