blob: 6a5fb65300b5a4c0851f906105fcab00a7907ebf [file] [log] [blame]
# Copyright 2020 The Chromium OS Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""A Bluetooth adapter MTBF test of some Bluetooth use cases.
To add a new use case we just need to inherit from the existing test class
and then call the desired test methods in the batch method below. This allows
the test case to be used as both part of a MTBF batch and a normal batch.
"""
import threading
import time
from autotest_lib.server.cros.bluetooth.bluetooth_adapter_audio_tests import \
BluetoothAdapterAudioTests
from autotest_lib.server.cros.bluetooth.bluetooth_adapter_better_together \
import BluetoothAdapterBetterTogether
from autotest_lib.server.cros.bluetooth.bluetooth_adapter_hidreports_tests \
import BluetoothAdapterHIDReportTests
from autotest_lib.server.cros.bluetooth.bluetooth_adapter_quick_tests import \
BluetoothAdapterQuickTests
from autotest_lib.client.cros.bluetooth.bluetooth_audio_test_data import A2DP
# How long for a short suspend in seconds
SHORT_SUSPEND_SEC = 10
# The timeout in seconds for resume and sleep actions
ACTION_TIMEOUT_SEC = 10
# Iterations to run the short mouse report test, this equals about 10 mins
MOUSE_TEST_ITERATION_SHORT = 15
# Iterations to run the long mouse report test, this equals about 30 mins
MOUSE_TEST_ITERATION_LONG = 50
# Iterations to run the keyboard report test, this equals about 10 mins
KEYBOARD_TEST_ITERATION = 60
# Iterations to run the A2DP report test, this equals about 30 mins
A2DP_TEST_ITERATION = 225
class bluetooth_AdapterMTBF(BluetoothAdapterBetterTogether,
BluetoothAdapterHIDReportTests,
BluetoothAdapterAudioTests):
"""A Batch of Bluetooth adapter tests for MTBF. This test is written
as a batch of tests in order to reduce test time, since auto-test ramp up
time is costly. The batch is using BluetoothAdapterQuickTests wrapper
methods to start and end a test and a batch of tests.
This class can be called to run the entire test batch or to run a
specific test only
"""
MTBF_TIMEOUT_MINS = 300
batch_wrapper = BluetoothAdapterQuickTests.quick_test_batch_decorator
mtbf_wrapper = BluetoothAdapterQuickTests.quick_test_mtbf_decorator
test_wrapper = BluetoothAdapterQuickTests.quick_test_test_decorator
@test_wrapper('MTBF Typical Use Cases',
devices={'BLE_MOUSE': 1}, shared_devices_count=1)
def typical_use_cases_test(self):
"""Do some initialization work then start the typical MTBF test loop"""
mouse = self.devices['BLE_MOUSE'][0]
self.run_typical_use_cases(mouse)
@mtbf_wrapper(timeout_mins=MTBF_TIMEOUT_MINS, test_name='typical_use_cases')
def run_typical_use_cases(self, mouse):
"""Run typical MTBF test scenarios:
1. Pair the mouse
2. Run the better together test
3. Suspend/Resume
4. Run concurrent mouse report and A2DP tests for 30 minutes
5. Suspend and wake up the DUT by mosue
6. Pair the keyboard
7. Run concurrent mouse and keyboard report tests for 10 minutes
"""
self.test_device_pairing(mouse)
# Reuse the shared device as a phone
shared_peer = self.shared_peers[0]
phone = self.reset_device(shared_peer, 'BLE_PHONE')
# Run the better together test on the phone
self.test_better_together(phone)
# Restore the discovery filter since better together test changed it
self.test_set_discovery_filter({'Transport':'auto'})
# Reuse the shared device as an bluetooth audio
audio = self.reset_device(shared_peer, 'BLUETOOTH_AUDIO')
self.test_suspend_resume(device=mouse)
# Run A2DP and mouse tests concurrently
audio_thread = threading.Thread(target=self.test_audio, args=(audio,))
mouse_thread = threading.Thread(
target=self.test_mouse, args=(mouse, MOUSE_TEST_ITERATION_LONG))
audio_thread.start()
mouse_thread.start()
audio_thread.join()
mouse_thread.join()
# Mouse wakeup test
self.test_suspend_and_mouse_wakeup(mouse)
# Remove the pairing from the audio device
audio.RemoveDevice(self.bluetooth_facade.address)
# Reuse the shared device as a CL keyboard
keyboard = self.reset_device(shared_peer, 'KEYBOARD')
# Pair the keyboard
self.test_device_pairing(keyboard)
# Run the mouse and keyboard report tests concurrently
mouse_thread = threading.Thread(
target=self.test_mouse, args=(mouse, MOUSE_TEST_ITERATION_SHORT))
keyboard_thread = threading.Thread(
target=self.test_keyboard, args=(keyboard,))
mouse_thread.start()
keyboard_thread.start()
mouse_thread.join()
keyboard_thread.join()
# Remove the pairing from both peer devices
self.test_remove_pairing(mouse.address)
self.test_remove_pairing(keyboard.address)
def test_better_together(self, phone):
"""Test better together"""
phone.RemoveDevice(self.bluetooth_facade.address)
self.test_smart_unlock(address=phone.address)
def test_mouse(self, mouse, iteration):
"""Run mouse report test for certain iterations"""
for i in range(iteration):
self.run_mouse_tests(device=mouse)
def test_keyboard(self, keyboard):
"""Run keyboard report test for certain iterations"""
for i in range(KEYBOARD_TEST_ITERATION):
self.run_keyboard_tests(device=keyboard)
def test_audio(self, device):
"""Test A2DP
This test plays A2DP audio on the DUT and record on the peer device,
then verify the legitimacy of the frames recorded.
"""
device.RemoveDevice(self.bluetooth_facade.address)
self.initialize_bluetooth_audio(device, A2DP)
self.test_device_set_discoverable(device, True)
self.test_discover_device(device.address)
self.test_stop_discovery()
self.test_pairing(device.address, device.pin, trusted=True)
device.SetTrustedByRemoteAddress(self.bluetooth_facade.address)
self.test_connection_by_adapter(device.address)
for i in range(A2DP_TEST_ITERATION):
self.test_a2dp_sinewaves(device)
self.test_disconnection_by_adapter(device.address)
self.cleanup_bluetooth_audio(device, A2DP)
self.test_remove_device_object(device.address)
def test_device_pairing(self, device):
"""Test device pairing"""
device.RemoveDevice(self.bluetooth_facade.address)
self.test_device_set_discoverable(device, True)
self.test_discover_device(device.address)
self.test_stop_discovery()
time.sleep(self.TEST_SLEEP_SECS)
self.test_pairing(device.address, device.pin, trusted=True)
time.sleep(self.TEST_SLEEP_SECS)
self.test_connection_by_adapter(device.address)
def test_suspend_resume(self, device):
"""Test the device can connect after suspending and resuming"""
boot_id = self.host.get_boot_id()
suspend = self.suspend_async(
suspend_time=SHORT_SUSPEND_SEC, allow_early_resume=True)
self.test_suspend_and_wait_for_sleep(
suspend, sleep_timeout=ACTION_TIMEOUT_SEC)
self.test_wait_for_resume(
boot_id, suspend, resume_timeout=ACTION_TIMEOUT_SEC)
# LE can't reconnect without advertising/discoverable
self.test_device_set_discoverable(device, True)
self.test_connection_by_device(device)
def test_suspend_and_mouse_wakeup(self, mouse):
"""Test the device can be waken up by the mouse"""
boot_id = self.host.get_boot_id()
suspend = self.suspend_async(
suspend_time=SHORT_SUSPEND_SEC, allow_early_resume=True)
self.test_adapter_wake_enabled()
self.test_suspend_and_wait_for_sleep(
suspend, sleep_timeout=ACTION_TIMEOUT_SEC)
# Trigger peer wakeup
peer_wake = self.device_connect_async('BLE_MOUSE', mouse,
self.bluetooth_facade.address)
peer_wake.start()
# Expect a quick resume. If a timeout occurs, test fails.
self.test_wait_for_resume(
boot_id, suspend, resume_timeout=ACTION_TIMEOUT_SEC,
fail_on_timeout=True)
# Finish peer wake process
peer_wake.join()
# Make sure we're actually connected
self.test_device_is_connected(mouse.address)
@test_wrapper('MTBF Better Together Stress', devices={'BLE_PHONE': 1})
def better_together_stress_test(self):
"""Run better together stress test"""
phone = self.devices['BLE_PHONE'][0]
phone.RemoveDevice(self.bluetooth_facade.address)
self.run_better_together_stress(address=phone.address)
@mtbf_wrapper(
timeout_mins=MTBF_TIMEOUT_MINS, test_name='better_together_stress')
def run_better_together_stress(self, address):
"""Run better together stress test"""
self.test_smart_unlock(address)
@batch_wrapper('Adapter MTBF')
def mtbf_batch_run(self, num_iterations=1, test_name=None):
"""Run the Bluetooth MTBF test batch or a specific
given test. The wrapper of this method is implemented in
batch_decorator. Using the decorator a test batch method can
implement the only its core tests invocations and let the
decorator handle the wrapper, which is taking care for whether to
run a specific test or the batch as a whole, and running the batch
in iterations
@param num_iterations: how many iterations to run
@param test_name: specific test to run otherwise None to run the
whole batch
"""
# TODO: finalize the test cases that need to be run as MTBF
self.typical_use_cases_test()
def run_once(self, host, num_iterations=1, test_name=None, args_dict=None):
"""Run the batch of Bluetooth MTBF tests
@param host: the DUT, usually a chromebook
@param num_iterations: the number of rounds to execute the test
@test_name: the test to run, or None for all tests
"""
# Initialize and run the test batch or the requested specific test
self.set_fail_fast(args_dict, True)
self.quick_test_init(host, use_btpeer=True)
self.mtbf_batch_run(num_iterations, test_name)
self.quick_test_cleanup()