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 script to exercises different ways Ble Advertisements can run in 18concurrency. This test was designed to be run in a shield box. 19""" 20 21import concurrent 22import os 23import time 24 25from queue import Empty 26from acts.test_decorators import test_tracker_info 27from acts_contrib.test_utils.bt.BluetoothBaseTest import BluetoothBaseTest 28from acts_contrib.test_utils.bt.bt_test_utils import BtTestUtilsError 29from acts_contrib.test_utils.bt.bt_constants import ble_advertise_settings_modes 30from acts_contrib.test_utils.bt.bt_constants import ble_scan_settings_callback_types 31from acts_contrib.test_utils.bt.bt_constants import ble_scan_settings_modes 32from acts_contrib.test_utils.bt.bt_constants import adv_succ 33from acts_contrib.test_utils.bt.bt_test_utils import generate_ble_advertise_objects 34from acts_contrib.test_utils.bt.bt_test_utils import generate_ble_scan_objects 35from acts_contrib.test_utils.bt.bt_test_utils import get_advanced_droid_list 36from acts_contrib.test_utils.bt.bt_test_utils import reset_bluetooth 37from acts_contrib.test_utils.bt.bt_test_utils import scan_and_verify_n_advertisements 38from acts_contrib.test_utils.bt.bt_constants import scan_result 39from acts_contrib.test_utils.bt.bt_test_utils import setup_n_advertisements 40from acts_contrib.test_utils.bt.bt_test_utils import take_btsnoop_logs 41from acts_contrib.test_utils.bt.bt_test_utils import teardown_n_advertisements 42 43 44class ConcurrentBleAdvertisingTest(BluetoothBaseTest): 45 default_timeout = 10 46 droid_list = [] 47 max_advertisements = -1 48 49 def setup_class(self): 50 super().setup_class() 51 self.scn_ad = self.android_devices[0] 52 self.adv_ad = self.android_devices[1] 53 self.droid_list = get_advanced_droid_list(self.android_devices) 54 self.max_advertisements = self.droid_list[1]['max_advertisements'] 55 56 def setup_test(self): 57 super().setup_test() 58 return reset_bluetooth(self.android_devices) 59 60 def _verify_n_advertisements(self, num_advertisements): 61 try: 62 advertise_callback_list = setup_n_advertisements( 63 self.adv_ad, num_advertisements) 64 except BtTestUtilsError: 65 return False 66 try: 67 scan_and_verify_n_advertisements(self.scn_ad, num_advertisements) 68 except BtTestUtilsError: 69 return False 70 teardown_n_advertisements(self.adv_ad, 71 len(advertise_callback_list), 72 advertise_callback_list) 73 return True 74 75 @BluetoothBaseTest.bt_test_wrap 76 @test_tracker_info(uuid='abc03874-6d7a-4b5d-9f29-18731a102793') 77 def test_max_advertisements_defaults(self): 78 """Testing max advertisements. 79 80 Test that a single device can have the max advertisements 81 concurrently advertising. 82 83 Steps: 84 1. Setup the scanning android device. 85 2. Setup the advertiser android device. 86 3. Start scanning on the max_advertisements as defined in the script. 87 4. Verify that all advertisements are found. 88 89 Expected Result: 90 All advertisements should start without errors. 91 92 Returns: 93 Pass if True 94 Fail if False 95 96 TAGS: LE, Advertising, Concurrency 97 Priority: 0 98 """ 99 return self._verify_n_advertisements(self.max_advertisements) 100 101 @BluetoothBaseTest.bt_test_wrap 102 @test_tracker_info(uuid='50ee137e-eb71-40ef-b72f-a5fd646190d2') 103 def test_max_advertisements_include_device_name_and_filter_device_name( 104 self): 105 """Testing max advertisement variant. 106 107 Test that a single device can have the max advertisements 108 concurrently advertising. Include the device name as a part of the filter 109 and advertisement data. 110 111 Steps: 112 1. Setup the scanning android device. 113 2. Setup the advertiser android device. 114 3. Include device name in each advertisement. 115 4. Include device name filter in the scanner. 116 5. Start scanning on the max_advertisements as defined in the script. 117 6. Verify that all advertisements are found. 118 119 Expected Result: 120 All advertisements should start without errors. 121 122 Returns: 123 Pass if True 124 Fail if False 125 126 TAGS: LE, Advertising, Concurrency 127 Priority: 2 128 """ 129 self.adv_ad.droid.bleSetAdvertiseDataIncludeDeviceName(True) 130 self.scn_ad.droid.bleSetScanFilterDeviceName( 131 self.adv_ad.droid.bluetoothGetLocalName()) 132 return self._verify_n_advertisements(self.max_advertisements) 133 134 @BluetoothBaseTest.bt_test_wrap 135 @test_tracker_info(uuid='f7e9ba2b-6286-4510-a8a0-f1df831056c0') 136 def test_max_advertisements_exclude_device_name_and_filter_device_name( 137 self): 138 """Test max advertisement variant. 139 140 Test that a single device can have the max advertisements concurrently 141 advertising. Include the device name as a part of the filter but not the 142 advertisement data. 143 144 Steps: 145 1. Setup the scanning android device. 146 2. Setup the advertiser android device. 147 3. Include device name filter in the scanner. 148 4. Start scanning on the max_advertisements as defined in the script. 149 5. Verify that no advertisements are found. 150 151 Expected Result: 152 All advertisements should start without errors. 153 154 Returns: 155 Pass if True 156 Fail if False 157 158 TAGS: LE, Advertising, Concurrency 159 Priority: 2 160 """ 161 self.adv_ad.droid.bleSetAdvertiseDataIncludeDeviceName(False) 162 self.scn_ad.droid.bleSetScanFilterDeviceName( 163 self.adv_ad.droid.bluetoothGetLocalName()) 164 return not self._verify_n_advertisements(self.max_advertisements) 165 166 @BluetoothBaseTest.bt_test_wrap 167 @test_tracker_info(uuid='6ce102d7-61e1-4ca0-bcfb-767437b86c2b') 168 def test_max_advertisements_with_manufacturer_data(self): 169 """Test max advertisement variant. 170 171 Test that a single device can have the max advertisements concurrently 172 advertising. Include the manufacturer data as a part of the filter and 173 advertisement data. 174 175 Steps: 176 1. Setup the scanning android device. 177 2. Setup the advertiser android device. 178 3. Include manufacturer data in each advertisement. 179 4. Include manufacturer data filter in the scanner. 180 5. Start scanning on the max_advertisements as defined in the script. 181 6. Verify that all advertisements are found. 182 183 Expected Result: 184 All advertisements should start without errors. 185 186 Returns: 187 Pass if True 188 Fail if False 189 190 TAGS: LE, Advertising, Concurrency 191 Priority: 2 192 """ 193 self.scn_ad.droid.bleSetScanFilterManufacturerData(1, [1]) 194 self.adv_ad.droid.bleAddAdvertiseDataManufacturerId(1, [1]) 195 return self._verify_n_advertisements(self.max_advertisements) 196 197 @BluetoothBaseTest.bt_test_wrap 198 @test_tracker_info(uuid='2fc7d5e8-1539-42a8-8681-ce0b8bfc0924') 199 def test_max_advertisements_with_manufacturer_data_mask(self): 200 """Test max advertisements variant. 201 202 Test that a single device can have the max advertisements concurrently 203 advertising. Include the manufacturer data mask as a part of the filter 204 and advertisement data. 205 206 Steps: 207 1. Setup the scanning android device. 208 2. Setup the advertiser android device. 209 3. Include manufacturer data in each advertisement. 210 4. Include manufacturer data mask filter in the scanner. 211 5. Start scanning on the max_advertisements as defined in the script. 212 6. Verify that all advertisements are found. 213 214 Expected Result: 215 All advertisements should start without errors. 216 217 Returns: 218 Pass if True 219 Fail if False 220 221 TAGS: LE, Advertising, Concurrency 222 Priority: 2 223 """ 224 self.scn_ad.droid.bleSetScanFilterManufacturerData(1, [1], [1]) 225 self.adv_ad.droid.bleAddAdvertiseDataManufacturerId(1, [1]) 226 return self._verify_n_advertisements(self.max_advertisements) 227 228 @BluetoothBaseTest.bt_test_wrap 229 @test_tracker_info(uuid='9ef615ed-1705-44ae-ab5b-f7e8fb4bb770') 230 def test_max_advertisements_with_service_data(self): 231 """Test max advertisement variant. 232 233 Test that a single device can have the max advertisements concurrently 234 advertising. Include the service data as a part of the filter and 235 advertisement data. 236 237 Steps: 238 1. Setup the scanning android device. 239 2. Setup the advertiser android device. 240 3. Include service data in each advertisement. 241 4. Include service data filter in the scanner. 242 5. Start scanning on the max_advertisements as defined in the script. 243 6. Verify that all advertisements are found. 244 245 Expected Result: 246 All advertisements should start without errors. 247 248 Returns: 249 Pass if True 250 Fail if False 251 252 TAGS: LE, Advertising, Concurrency 253 Priority: 2 254 """ 255 test_result = True 256 filter_list = self.scn_ad.droid.bleGenFilterList() 257 self.scn_ad.droid.bleSetScanFilterServiceData( 258 "0000110A-0000-1000-8000-00805F9B34FB", [11, 17, 80]) 259 self.adv_ad.droid.bleAddAdvertiseDataServiceData( 260 "0000110A-0000-1000-8000-00805F9B34FB", [11, 17, 80]) 261 return self._verify_n_advertisements(self.max_advertisements) 262 263 @BluetoothBaseTest.bt_test_wrap 264 @test_tracker_info(uuid='9ef615ed-1705-44ae-ab5b-f7e8fb4bb770') 265 def test_max_advertisements_with_manufacturer_data_mask_and_include_device_name( 266 self): 267 """Test max advertisement variant. 268 269 Test that a single device can have the max advertisements concurrently 270 advertising. Include the device name and manufacturer data as a part of 271 the filter and advertisement data. 272 273 Steps: 274 1. Setup the scanning android device. 275 2. Setup the advertiser android device. 276 3. Include device name and manufacturer data in each advertisement. 277 4. Include device name and manufacturer data filter in the scanner. 278 5. Start scanning on the max_advertisements as defined in the script. 279 6. Verify that all advertisements are found. 280 281 Expected Result: 282 All advertisements should start without errors. 283 284 Returns: 285 Pass if True 286 Fail if False 287 288 TAGS: LE, Advertising, Concurrency 289 Priority: 2 290 """ 291 self.adv_ad.droid.bleSetAdvertiseDataIncludeDeviceName(True) 292 self.scn_ad.droid.bleSetScanFilterDeviceName( 293 self.adv_ad.droid.bluetoothGetLocalName()) 294 self.scn_ad.droid.bleSetScanFilterManufacturerData(1, [1], [1]) 295 self.adv_ad.droid.bleAddAdvertiseDataManufacturerId(1, [1]) 296 return self._verify_n_advertisements(self.max_advertisements) 297 298 @BluetoothBaseTest.bt_test_wrap 299 @test_tracker_info(uuid='c2ca85fb-6663-431d-aa30-5286a85dbbe0') 300 def test_max_advertisements_with_service_uuids(self): 301 """Test max advertisement variant. 302 303 Test that a single device can have the max advertisements concurrently 304 advertising. Include the service uuid as a part of the filter and 305 advertisement data. 306 307 Steps: 308 1. Setup the scanning android device. 309 2. Setup the advertiser android device. 310 3. Include service uuid in each advertisement. 311 4. Include service uuid filter in the scanner. 312 5. Start scanning on the max_advertisements as defined in the script. 313 6. Verify that all advertisements are found. 314 315 Expected Result: 316 All advertisements should start without errors. 317 318 Returns: 319 Pass if True 320 Fail if False 321 322 TAGS: LE, Advertising, Concurrency 323 Priority: 1 324 """ 325 self.scn_ad.droid.bleSetScanFilterServiceUuid( 326 "00000000-0000-1000-8000-00805f9b34fb") 327 self.adv_ad.droid.bleSetAdvertiseDataSetServiceUuids( 328 ["00000000-0000-1000-8000-00805f9b34fb"]) 329 return self._verify_n_advertisements(self.max_advertisements) 330 331 @BluetoothBaseTest.bt_test_wrap 332 @test_tracker_info(uuid='756e026f-64d7-4a2f-935a-3790c0ac4503') 333 def test_max_advertisements_with_service_uuid_and_service_mask(self): 334 """Test max advertisements variant. 335 336 Test that a single device can have the max advertisements concurrently 337 advertising. Include the service mask as a part of the filter and 338 advertisement data. 339 340 Steps: 341 1. Setup the scanning android device. 342 2. Setup the advertiser android device. 343 3. Include service uuid in each advertisement. 344 4. Include service mask filter in the scanner. 345 5. Start scanning on the max_advertisements as defined in the script. 346 6. Verify that all advertisements are found. 347 348 Expected Result: 349 All advertisements should start without errors. 350 351 Returns: 352 Pass if True 353 Fail if False 354 355 TAGS: LE, Advertising, Concurrency 356 Priority: 2 357 """ 358 self.scn_ad.droid.bleSetScanFilterServiceUuid( 359 "00000000-0000-1000-8000-00805f9b34fb", 360 "00000000-0000-1000-8000-00805f9b34fb") 361 self.adv_ad.droid.bleSetAdvertiseDataSetServiceUuids( 362 ["00000000-0000-1000-8000-00805f9b34fb"]) 363 return self._verify_n_advertisements(self.max_advertisements) 364 365 @BluetoothBaseTest.bt_test_wrap 366 @test_tracker_info(uuid='79c4b6cd-9f07-49a9-829f-69b29ea8d322') 367 def test_max_advertisements_plus_one(self): 368 """Test max advertisements plus one. 369 370 Test that a single device can have the max advertisements concurrently 371 advertising but fail on starting the max advertisements plus one. 372 filter and advertisement data. 373 374 Steps: 375 1. Setup the scanning android device. 376 2. Setup the advertiser android device. 377 3. Start max_advertisements + 1. 378 379 Expected Result: 380 The last advertisement should fail. 381 382 Returns: 383 Pass if True 384 Fail if False 385 386 TAGS: LE, Advertising, Concurrency 387 Priority: 0 388 """ 389 return not self._verify_n_advertisements(self.max_advertisements + 1) 390 391 @BluetoothBaseTest.bt_test_wrap 392 @test_tracker_info(uuid='0bd6e490-a501-4fe1-88e5-9b77970c0b95') 393 def test_start_two_advertisements_on_same_callback(self): 394 """Test invalid advertisement scenario. 395 396 Test that a single device cannot have two advertisements start on the 397 same callback. 398 399 Steps: 400 1. Setup the scanning android device. 401 2. Setup the advertiser android device. 402 3. Call start ble advertising on the same callback. 403 404 Expected Result: 405 The second call of start advertising on the same callback should fail. 406 407 Returns: 408 Pass if True 409 Fail if False 410 411 TAGS: LE, Advertising, Concurrency 412 Priority: 1 413 """ 414 test_result = True 415 advertise_callback, advertise_data, advertise_settings = ( 416 generate_ble_advertise_objects(self.adv_ad.droid)) 417 self.adv_ad.droid.bleStartBleAdvertising( 418 advertise_callback, advertise_data, advertise_settings) 419 try: 420 self.adv_ad.ed.pop_event( 421 adv_succ.format(advertise_callback), self.default_timeout) 422 except Empty as error: 423 self.log.error("Test failed with Empty error: {}".format(error)) 424 return False 425 except concurrent.futures._base.TimeoutError as error: 426 self.log.debug( 427 "Test failed, filtering callback onSuccess never occurred: {}" 428 .format(error)) 429 try: 430 self.adv_ad.droid.bleStartBleAdvertising( 431 advertise_callback, advertise_data, advertise_settings) 432 self.adv_ad.ed.pop_event( 433 adv_succ.format(advertise_callback), self.default_timeout) 434 test_result = False 435 except Empty as error: 436 self.log.debug("Test passed with Empty error: {}".format(error)) 437 except concurrent.futures._base.TimeoutError as error: 438 self.log.debug( 439 "Test passed, filtering callback onSuccess never occurred: {}" 440 .format(error)) 441 442 return test_result 443 444 @BluetoothBaseTest.bt_test_wrap 445 @test_tracker_info(uuid='12632b31-22b9-4121-80b6-1263b9d90909') 446 def test_toggle_advertiser_bt_state(self): 447 """Test forcing stopping advertisements. 448 449 Test that a single device resets its callbacks when the bluetooth state is 450 reset. There should be no advertisements. 451 452 Steps: 453 1. Setup the scanning android device. 454 2. Setup the advertiser android device. 455 3. Call start ble advertising. 456 4. Toggle bluetooth on and off. 457 5. Scan for any advertisements. 458 459 Expected Result: 460 No advertisements should be found after toggling Bluetooth on the 461 advertising device. 462 463 Returns: 464 Pass if True 465 Fail if False 466 467 TAGS: LE, Advertising, Concurrency 468 Priority: 2 469 """ 470 test_result = True 471 self.adv_ad.droid.bleSetAdvertiseDataIncludeDeviceName(True) 472 advertise_callback, advertise_data, advertise_settings = ( 473 generate_ble_advertise_objects(self.adv_ad.droid)) 474 self.adv_ad.droid.bleStartBleAdvertising( 475 advertise_callback, advertise_data, advertise_settings) 476 try: 477 self.adv_ad.ed.pop_event( 478 adv_succ.format(advertise_callback), self.default_timeout) 479 except Empty as error: 480 self.log.error("Test failed with Empty error: {}".format(error)) 481 return False 482 except concurrent.futures._base.TimeoutError as error: 483 self.log.error( 484 "Test failed, filtering callback onSuccess never occurred: {}". 485 format(error)) 486 self.scn_ad.droid.bleSetScanSettingsScanMode( 487 ble_scan_settings_modes['low_latency']) 488 self.scn_ad.droid.bleSetScanFilterDeviceName( 489 self.adv_ad.droid.bluetoothGetLocalName()) 490 filter_list, scan_settings, scan_callback = generate_ble_scan_objects( 491 self.scn_ad.droid) 492 self.scn_ad.droid.bleBuildScanFilter(filter_list) 493 self.scn_ad.droid.bleStartBleScan(filter_list, scan_settings, 494 scan_callback) 495 try: 496 self.scn_ad.ed.pop_event( 497 scan_result.format(scan_callback), self.default_timeout) 498 except Empty as error: 499 self.log.error("Test failed with: {}".format(error)) 500 return False 501 self.scn_ad.droid.bleStopBleScan(scan_callback) 502 test_result = reset_bluetooth([self.android_devices[1]]) 503 self.scn_ad.droid.bleStartBleScan(filter_list, scan_settings, 504 scan_callback) 505 if not test_result: 506 return False 507 try: 508 expected_event = scan_result.format(scan_callback) 509 event = self.scn_ad.ed.pop_event(expected_event, 510 self.default_timeout) 511 self.log.error("Event {} not expected. Found: {}".format( 512 expected_event, event)) 513 return False 514 except Empty as error: 515 self.log.debug("Test passed with: {}".format(error)) 516 self.scn_ad.droid.bleStopBleScan(scan_callback) 517 self.adv_ad.droid.bleStopBleAdvertising(advertise_callback) 518 return True 519 520 @BluetoothBaseTest.bt_test_wrap 521 @test_tracker_info(uuid='785c5c77-d5d4-4d0f-8b7b-3eb1f1646d2c') 522 def test_restart_advertise_callback_after_bt_toggle(self): 523 """Test starting an advertisement on a cleared out callback. 524 525 Test that a single device resets its callbacks when the bluetooth state 526 is reset. 527 528 Steps: 529 1. Setup the scanning android device. 530 2. Setup the advertiser android device. 531 3. Call start ble advertising. 532 4. Toggle bluetooth on and off. 533 5. Call start ble advertising on the same callback. 534 535 Expected Result: 536 Starting an advertisement on a callback id after toggling bluetooth 537 should fail. 538 539 Returns: 540 Pass if True 541 Fail if False 542 543 TAGS: LE, Advertising, Concurrency 544 Priority: 1 545 """ 546 test_result = True 547 advertise_callback, advertise_data, advertise_settings = ( 548 generate_ble_advertise_objects(self.adv_ad.droid)) 549 self.adv_ad.droid.bleStartBleAdvertising( 550 advertise_callback, advertise_data, advertise_settings) 551 try: 552 self.adv_ad.ed.pop_event( 553 adv_succ.format(advertise_callback), self.default_timeout) 554 except Empty as error: 555 self.log.error("Test failed with Empty error: {}".format(error)) 556 test_result = False 557 except concurrent.futures._base.TimeoutError as error: 558 self.log.debug( 559 "Test failed, filtering callback onSuccess never occurred: {}". 560 format(error)) 561 test_result = reset_bluetooth([self.android_devices[1]]) 562 if not test_result: 563 return test_result 564 self.adv_ad.droid.bleStartBleAdvertising( 565 advertise_callback, advertise_data, advertise_settings) 566 try: 567 self.adv_ad.ed.pop_event( 568 adv_succ.format(advertise_callback), self.default_timeout) 569 except Empty as error: 570 self.log.error("Test failed with Empty error: {}".format(error)) 571 test_result = False 572 except concurrent.futures._base.TimeoutError as error: 573 self.log.debug( 574 "Test failed, filtering callback onSuccess never occurred: {}". 575 format(error)) 576 return test_result 577 578 @BluetoothBaseTest.bt_test_wrap 579 @test_tracker_info(uuid='dd5529b7-6774-4580-8b29-d84568c15442') 580 def test_timeout(self): 581 """Test starting advertiser with timeout. 582 583 Test that when a timeout is used, the advertiser is cleaned properly, 584 and next one can be started. 585 586 Steps: 587 1. Setup the advertiser android device with 4 second timeout. 588 2. Call start ble advertising. 589 3. Wait 5 seconds, to make sure advertiser times out. 590 4. Repeat steps 1-4 four times. 591 592 Expected Result: 593 Starting the advertising should succeed each time. 594 595 Returns: 596 Pass if True 597 Fail if False 598 599 TAGS: LE, Advertising, Concurrency 600 Priority: 1 601 """ 602 advertise_timeout_s = 4 603 num_iterations = 4 604 test_result = True 605 606 for i in range(0, num_iterations): 607 advertise_callback, advertise_data, advertise_settings = ( 608 generate_ble_advertise_objects(self.adv_ad.droid)) 609 610 self.adv_ad.droid.bleSetAdvertiseSettingsTimeout( 611 advertise_timeout_s * 1000) 612 613 self.adv_ad.droid.bleStartBleAdvertising( 614 advertise_callback, advertise_data, advertise_settings) 615 try: 616 self.adv_ad.ed.pop_event( 617 adv_succ.format(advertise_callback), self.default_timeout) 618 except Empty as error: 619 self.log.error("Test failed with Empty error: {}".format( 620 error)) 621 test_result = False 622 except concurrent.futures._base.TimeoutError as error: 623 self.log.debug( 624 "Test failed, filtering callback onSuccess never occurred: {}". 625 format(error)) 626 627 if not test_result: 628 return test_result 629 630 time.sleep(advertise_timeout_s + 1) 631 632 return test_result 633