1#!/usr/bin/env python3 2# 3# Copyright (C) 2016 The Android Open Source Project 4# 5# Licensed under the Apache License, Version 2.0 (the "License"); you may not 6# use this file except in compliance with the License. You may obtain a copy of 7# the License at 8# 9# http://www.apache.org/licenses/LICENSE-2.0 10# 11# Unless required by applicable law or agreed to in writing, software 12# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT 13# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the 14# License for the specific language governing permissions and limitations under 15# the License. 16""" 17Test the HFP profile for calling and connection management. 18""" 19 20import time 21 22from acts.test_decorators import test_tracker_info 23from acts_contrib.test_utils.bt.BluetoothBaseTest import BluetoothBaseTest 24from acts_contrib.test_utils.bt.BluetoothCarHfpBaseTest import BluetoothCarHfpBaseTest 25from acts_contrib.test_utils.bt import BtEnum 26from acts_contrib.test_utils.bt import bt_test_utils 27from acts_contrib.test_utils.car import car_bt_utils 28from acts_contrib.test_utils.car import car_telecom_utils 29from acts_contrib.test_utils.tel import tel_defines 30from acts_contrib.test_utils.tel.tel_test_utils import hangup_call 31from acts_contrib.test_utils.tel.tel_test_utils import initiate_call 32from acts_contrib.test_utils.tel.tel_test_utils import wait_and_answer_call 33 34BLUETOOTH_PKG_NAME = "com.android.bluetooth" 35CALL_TYPE_OUTGOING = "CALL_TYPE_OUTGOING" 36CALL_TYPE_INCOMING = "CALL_TYPE_INCOMING" 37default_timeout = 20 38 39 40class BtCarHfpConnectionTest(BluetoothCarHfpBaseTest): 41 def setup_class(self): 42 if not super(BtCarHfpConnectionTest, self).setup_class(): 43 return False 44 45 # Disable all 46 car_bt_utils.set_car_profile_priorities_off(self.hf, self.ag) 47 48 # Enable A2DP 49 bt_test_utils.set_profile_priority( 50 self.hf, self.ag, [BtEnum.BluetoothProfile.HEADSET_CLIENT], 51 BtEnum.BluetoothPriorityLevel.PRIORITY_ON) 52 53 return True 54 55 def setup_test(self): 56 if not super(BtCarHfpConnectionTest, self).setup_test(): 57 return False 58 self.hf.droid.bluetoothDisconnectConnected( 59 self.ag.droid.bluetoothGetLocalAddress()) 60 61 @test_tracker_info(uuid='a6669f9b-fb49-4bd8-aa9c-9d6369e34442') 62 @BluetoothBaseTest.bt_test_wrap 63 def test_call_transfer_disconnect_connect(self): 64 """ 65 Tests that after we connect when an active call is in progress, 66 we show the call. 67 68 Precondition: 69 1. AG & HF are disconnected but paired. 70 71 Steps: 72 1. Make a call from AG role (since disconnected) 73 2. Accept from RE role and transition the call to Active 74 3. Connect AG & HF 75 4. HF should transition into Active call state. 76 77 Returns: 78 Pass if True 79 Fail if False 80 81 Priority: 1 82 """ 83 # make a call on AG 84 if not initiate_call(self.log, self.ag, self.re_phone_number): 85 self.ag.log.error("Failed to initiate call from ag.") 86 return False 87 if not wait_and_answer_call(self.log, self.re): 88 self.re.log.error("Failed to accept call on re.") 89 return False 90 91 # Wait for AG, RE to go into an Active state. 92 if not car_telecom_utils.wait_for_active(self.log, self.ag): 93 self.ag.log.error("AG not in Active state.") 94 return False 95 if not car_telecom_utils.wait_for_active(self.log, self.re): 96 self.re.log.error("RE not in Active state.") 97 return False 98 99 # Now connect the devices. 100 if not bt_test_utils.connect_pri_to_sec( 101 self.hf, self.ag, 102 set([BtEnum.BluetoothProfile.HEADSET_CLIENT.value])): 103 self.log.error("Could not connect HF and AG {} {}".format( 104 self.hf.serial, self.ag.serial)) 105 return False 106 107 # Check that HF is in active state 108 if not car_telecom_utils.wait_for_active(self.log, self.hf): 109 self.hf.log.error("HF not in Active state.") 110 return False 111 112 # Hangup the call and check all devices are clean 113 self.hf.droid.telecomEndCall() 114 ret = True 115 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.hf) 116 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.ag) 117 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.re) 118 119 return ret 120 121 @test_tracker_info(uuid='97727b64-a590-4d84-a257-1facd8aafd16') 122 @BluetoothBaseTest.bt_test_wrap 123 def test_call_transfer_off_on(self): 124 """ 125 Tests that after we turn adapter on when an active call is in 126 progress, we show the call. 127 128 Precondition: 129 1. AG & HF are disconnected but paired. 130 2. HF's adapter is OFF 131 132 Steps: 133 1. Make a call from AG role (since disconnected) 134 2. Accept from RE role and transition the call to Active 135 3. Turn HF's adapter ON 136 4. HF should transition into Active call state. 137 138 Returns: 139 Pass if True 140 Fail if False 141 142 Priority: 1 143 """ 144 # Connect HF & AG 145 if not bt_test_utils.connect_pri_to_sec( 146 self.hf, self.ag, 147 set([BtEnum.BluetoothProfile.HEADSET_CLIENT.value])): 148 self.log.error("Could not connect HF and AG {} {}".format( 149 self.hf.serial, self.ag.serial)) 150 return False 151 152 # make a call on AG 153 if not initiate_call(self.log, self.ag, self.re_phone_number): 154 self.ag.log.error("Failed to initiate call from ag.") 155 return False 156 157 # Wait for all HF 158 if not car_telecom_utils.wait_for_dialing(self.log, self.hf): 159 self.hf.log.error("HF not in ringing state.") 160 return False 161 162 # Accept the call on RE 163 if not wait_and_answer_call(self.log, self.re): 164 self.re.log.error("Failed to accept call on re.") 165 return False 166 # Wait for all HF, AG, RE to go into an Active state. 167 if not car_telecom_utils.wait_for_active(self.log, self.hf): 168 self.hf.log.error("HF not in Active state.") 169 return False 170 if not car_telecom_utils.wait_for_active(self.log, self.ag): 171 self.ag.log.error("AG not in Active state.") 172 return False 173 if not car_telecom_utils.wait_for_active(self.log, self.re): 174 self.re.log.error("RE not in Active state.") 175 return False 176 177 # Turn the adapter OFF on HF 178 if not bt_test_utils.disable_bluetooth(self.hf.droid): 179 self.hf.log.error("Failed to turn BT off on HF.") 180 return False 181 182 # Turn adapter ON on HF 183 if not bt_test_utils.enable_bluetooth(self.hf.droid, self.hf.ed): 184 self.hf.log.error("Failed to turn BT ON after call on HF.") 185 return False 186 187 # Check that HF is in active state 188 if not car_telecom_utils.wait_for_active(self.log, self.hf): 189 self.hf.log.error("HF not in Active state.") 190 return False 191 192 # Hangup the call and check all devices are clean 193 self.hf.droid.telecomEndCall() 194 ret = True 195 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.hf) 196 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.ag) 197 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.re) 198 199 return ret 200 201 @test_tracker_info(uuid='95f76e2c-1cdd-4a7c-8e26-863b4c4242be') 202 @BluetoothBaseTest.bt_test_wrap 203 def test_call_transfer_connect_disconnect_connect(self): 204 """ 205 Test that when we go from connect -> disconnect -> connect on an active 206 call then the call is restored on HF. 207 208 Precondition: 209 1. AG & HF are paired 210 211 Steps: 212 0. Connect AG & HF 213 1. Make a call from HF role 214 2. Accept from RE role and transition the call to Active 215 3. Disconnect AG & HF 216 4. Verify that we don't have any calls on HF 217 5. Connect AG & HF 218 6. Verify that HF gets the call back. 219 220 Returns: 221 Pass if True 222 Fail if False 223 224 Priority: 1 225 """ 226 # Now connect the devices. 227 if not bt_test_utils.connect_pri_to_sec( 228 self.hf, self.ag, 229 set([BtEnum.BluetoothProfile.HEADSET_CLIENT.value])): 230 self.log.error("Could not connect HF and AG {} {}".format( 231 self.hf.serial, self.ag.serial)) 232 return False 233 234 # make a call on HF 235 if not car_telecom_utils.dial_number(self.log, self.hf, 236 self.re_phone_number): 237 self.hf.log.error("HF not in dialing state.") 238 return False 239 240 # Wait for HF, AG to be dialing and RE to be ringing 241 ret = True 242 ret &= car_telecom_utils.wait_for_dialing(self.log, self.hf) 243 #uncomment once sl4a code has been merged. 244 ret &= car_telecom_utils.wait_for_dialing(self.log, self.ag) 245 ret &= car_telecom_utils.wait_for_ringing(self.log, self.re) 246 247 if not ret: 248 self.log.error("Outgoing call did not get established") 249 return False 250 251 # Accept call on RE. 252 if not wait_and_answer_call(self.log, self.re): 253 self.re.log.error("Failed to accept call on re.") 254 return False 255 256 ret &= car_telecom_utils.wait_for_active(self.log, self.hf) 257 ret &= car_telecom_utils.wait_for_active(self.log, self.ag) 258 ret &= car_telecom_utils.wait_for_active(self.log, self.re) 259 260 if not ret: 261 self.log.error("Outgoing call did not transition to active") 262 return False 263 264 # Disconnect HF & AG 265 self.hf.droid.bluetoothDisconnectConnected( 266 self.ag.droid.bluetoothGetLocalAddress()) 267 268 # We use the proxy of the Call going away as HF disconnected 269 if not car_telecom_utils.wait_for_not_in_call(self.log, self.hf): 270 self.hf.log.error("HF still in call after disconnection.") 271 return False 272 273 # Now connect the devices. 274 if not bt_test_utils.connect_pri_to_sec( 275 self.hf, self.ag, 276 set([BtEnum.BluetoothProfile.HEADSET_CLIENT.value])): 277 self.log.error("Could not connect HF and AG {} {}".format( 278 self.hf.serial, self.ag.serial)) 279 # Additional profile connection check for b/ 280 if not bt_test_utils.is_hfp_client_device_connected( 281 self.hf, self.ag.droid.bluetoothGetLocalAddress()): 282 self.hf.log.info( 283 "HFP Client connected even though connection state changed " 284 + " event not found") 285 return False 286 287 # Check that HF is in active state 288 if not car_telecom_utils.wait_for_active(self.log, self.hf): 289 self.hf.log.error("HF not in Active state.") 290 return False 291 292 # Hangup the call and check all devices are clean 293 self.hf.droid.telecomEndCall() 294 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.hf) 295 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.ag) 296 ret &= car_telecom_utils.wait_for_not_in_call(self.log, self.re) 297 298 return ret 299