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 suite for GATT over BR/EDR. 18""" 19 20import time 21from queue import Empty 22 23from acts.test_decorators import test_tracker_info 24from acts_contrib.test_utils.bt.BluetoothBaseTest import BluetoothBaseTest 25from acts_contrib.test_utils.bt.bt_test_utils import reset_bluetooth 26from acts_contrib.test_utils.bt.bt_constants import gatt_characteristic 27from acts_contrib.test_utils.bt.bt_constants import gatt_service_types 28from acts_contrib.test_utils.bt.bt_constants import gatt_transport 29from acts_contrib.test_utils.bt.bt_constants import gatt_cb_strings 30from acts_contrib.test_utils.bt.bt_gatt_utils import GattTestUtilsError 31from acts_contrib.test_utils.bt.bt_gatt_utils import disconnect_gatt_connection 32from acts_contrib.test_utils.bt.bt_gatt_utils import log_gatt_server_uuids 33from acts_contrib.test_utils.bt.bt_gatt_utils import orchestrate_gatt_connection 34from acts_contrib.test_utils.bt.bt_gatt_utils import setup_gatt_characteristics 35from acts_contrib.test_utils.bt.bt_gatt_utils import setup_gatt_connection 36from acts_contrib.test_utils.bt.bt_gatt_utils import setup_gatt_descriptors 37from acts_contrib.test_utils.bt.bt_gatt_utils import setup_multiple_services 38from acts_contrib.test_utils.bt.bt_test_utils import setup_multiple_devices_for_bt_test 39from acts_contrib.test_utils.bt.bt_test_utils import take_btsnoop_logs 40 41 42class GattOverBrEdrTest(BluetoothBaseTest): 43 adv_instances = [] 44 bluetooth_gatt_list = [] 45 gatt_server_list = [] 46 default_timeout = 10 47 default_discovery_timeout = 3 48 per_droid_mac_address = None 49 50 def setup_class(self): 51 super(BluetoothBaseTest, self).setup_class() 52 self.cen_ad = self.android_devices[0] 53 self.per_ad = self.android_devices[1] 54 55 self.per_droid_mac_address = self.per_ad.droid.bluetoothGetLocalAddress( 56 ) 57 if not self.per_droid_mac_address: 58 return False 59 return True 60 61 def setup_test(self): 62 super(BluetoothBaseTest, self).setup_test() 63 bluetooth_gatt_list = [] 64 self.gatt_server_list = [] 65 self.adv_instances = [] 66 67 def teardown_test(self): 68 for bluetooth_gatt in self.bluetooth_gatt_list: 69 self.cen_ad.droid.gattClientClose(bluetooth_gatt) 70 for gatt_server in self.gatt_server_list: 71 self.per_ad.droid.gattServerClose(gatt_server) 72 return True 73 74 def on_fail(self, test_name, begin_time): 75 take_btsnoop_logs(self.android_devices, self, test_name) 76 reset_bluetooth(self.android_devices) 77 78 def _orchestrate_gatt_disconnection(self, bluetooth_gatt, gatt_callback): 79 self.log.info("Disconnecting from peripheral device.") 80 try: 81 disconnect_gatt_connection(self.cen_ad, bluetooth_gatt, 82 gatt_callback) 83 if bluetooth_gatt in self.bluetooth_gatt_list: 84 self.bluetooth_gatt_list.remove(bluetooth_gatt) 85 except GattTestUtilsError as err: 86 self.log.error(err) 87 return False 88 return True 89 90 def _find_service_added_event(self, gatt_server_callback, uuid): 91 event = self.per_ad.ed.pop_event( 92 gatt_cb_strings['serv_added'].format(gatt_server_callback), 93 self.default_timeout) 94 if event['data']['serviceUuid'].lower() != uuid.lower(): 95 self.log.info("Uuid mismatch. Found: {}, Expected {}.".format( 96 event['data']['serviceUuid'], uuid)) 97 return False 98 return True 99 100 @BluetoothBaseTest.bt_test_wrap 101 @test_tracker_info(uuid='32d32c87-911e-4f14-9654-29fe1431e995') 102 def test_gatt_bredr_connect(self): 103 """Test GATT connection over BR/EDR. 104 105 Test establishing a gatt connection between a GATT server and GATT 106 client. 107 108 Steps: 109 1. Start a generic advertisement. 110 2. Start a generic scanner. 111 3. Find the advertisement and extract the mac address. 112 4. Stop the first scanner. 113 5. Create a GATT connection between the scanner and advertiser. 114 6. Disconnect the GATT connection. 115 116 Expected Result: 117 Verify that a connection was established and then disconnected 118 successfully. 119 120 Returns: 121 Pass if True 122 Fail if False 123 124 TAGS: BR/EDR, Filtering, GATT, Scanning 125 Priority: 0 126 """ 127 gatt_server_cb = self.per_ad.droid.gattServerCreateGattServerCallback() 128 gatt_server = self.per_ad.droid.gattServerOpenGattServer( 129 gatt_server_cb) 130 self.gatt_server_list.append(gatt_server) 131 try: 132 bluetooth_gatt, gatt_callback, adv_callback = ( 133 orchestrate_gatt_connection(self.cen_ad, self.per_ad, 134 gatt_transport['bredr'], 135 self.per_droid_mac_address)) 136 self.bluetooth_gatt_list.append(bluetooth_gatt) 137 except GattTestUtilsError as err: 138 self.log.error(err) 139 return False 140 return self._orchestrate_gatt_disconnection(bluetooth_gatt, 141 gatt_callback) 142 143 @BluetoothBaseTest.bt_test_wrap 144 @test_tracker_info(uuid='357b697b-a52c-4c2a-997c-00876a018f37') 145 def test_gatt_bredr_connect_trigger_on_read_rssi(self): 146 """Test GATT connection over BR/EDR read RSSI. 147 148 Test establishing a gatt connection between a GATT server and GATT 149 client then read the RSSI. 150 151 Steps: 152 1. Start a generic advertisement. 153 2. Start a generic scanner. 154 3. Find the advertisement and extract the mac address. 155 4. Stop the first scanner. 156 5. Create a GATT connection between the scanner and advertiser. 157 6. From the scanner, request to read the RSSI of the advertiser. 158 7. Disconnect the GATT connection. 159 160 Expected Result: 161 Verify that a connection was established and then disconnected 162 successfully. Verify that the RSSI was ready correctly. 163 164 Returns: 165 Pass if True 166 Fail if False 167 168 TAGS: BR/EDR, Scanning, GATT, RSSI 169 Priority: 1 170 """ 171 gatt_server_cb = self.per_ad.droid.gattServerCreateGattServerCallback() 172 gatt_server = self.per_ad.droid.gattServerOpenGattServer( 173 gatt_server_cb) 174 self.gatt_server_list.append(gatt_server) 175 try: 176 bluetooth_gatt, gatt_callback, adv_callback = ( 177 orchestrate_gatt_connection(self.cen_ad, self.per_ad, 178 gatt_transport['bredr'], 179 self.per_droid_mac_address)) 180 self.bluetooth_gatt_list.append(bluetooth_gatt) 181 except GattTestUtilsError as err: 182 self.log.error(err) 183 return False 184 if self.cen_ad.droid.gattClientReadRSSI(bluetooth_gatt): 185 self.cen_ad.ed.pop_event( 186 gatt_cb_strings['rd_remote_rssi'].format(gatt_callback), 187 self.default_timeout) 188 return self._orchestrate_gatt_disconnection(bluetooth_gatt, 189 gatt_callback) 190 191 @BluetoothBaseTest.bt_test_wrap 192 @test_tracker_info(uuid='dee9ef28-b872-428a-821b-cc62f27ba936') 193 def test_gatt_bredr_connect_trigger_on_services_discovered(self): 194 """Test GATT connection and discover services of peripheral. 195 196 Test establishing a gatt connection between a GATT server and GATT 197 client the discover all services from the connected device. 198 199 Steps: 200 1. Start a generic advertisement. 201 2. Start a generic scanner. 202 3. Find the advertisement and extract the mac address. 203 4. Stop the first scanner. 204 5. Create a GATT connection between the scanner and advertiser. 205 6. From the scanner (central device), discover services. 206 7. Disconnect the GATT connection. 207 208 Expected Result: 209 Verify that a connection was established and then disconnected 210 successfully. Verify that the service were discovered. 211 212 Returns: 213 Pass if True 214 Fail if False 215 216 TAGS: BR/EDR, Scanning, GATT, Services 217 Priority: 1 218 """ 219 gatt_server_cb = self.per_ad.droid.gattServerCreateGattServerCallback() 220 gatt_server = self.per_ad.droid.gattServerOpenGattServer( 221 gatt_server_cb) 222 self.gatt_server_list.append(gatt_server) 223 try: 224 bluetooth_gatt, gatt_callback, adv_callback = ( 225 orchestrate_gatt_connection(self.cen_ad, self.per_ad, 226 gatt_transport['bredr'], 227 self.per_droid_mac_address)) 228 self.bluetooth_gatt_list.append(bluetooth_gatt) 229 except GattTestUtilsError as err: 230 self.log.error(err) 231 return False 232 discovered_services_index = -1 233 if self.cen_ad.droid.gattClientDiscoverServices(bluetooth_gatt): 234 event = self.cen_ad.ed.pop_event( 235 gatt_cb_strings['gatt_serv_disc'].format(gatt_callback), 236 self.default_timeout) 237 discovered_services_index = event['data']['ServicesIndex'] 238 return self._orchestrate_gatt_disconnection(bluetooth_gatt, 239 gatt_callback) 240 241 @BluetoothBaseTest.bt_test_wrap 242 @test_tracker_info(uuid='01883bdd-0cf8-48fb-bf15-467bbd4f065b') 243 def test_gatt_bredr_connect_trigger_on_services_discovered_iterate_attributes( 244 self): 245 """Test GATT connection and iterate peripherals attributes. 246 247 Test establishing a gatt connection between a GATT server and GATT 248 client and iterate over all the characteristics and descriptors of the 249 discovered services. 250 251 Steps: 252 1. Start a generic advertisement. 253 2. Start a generic scanner. 254 3. Find the advertisement and extract the mac address. 255 4. Stop the first scanner. 256 5. Create a GATT connection between the scanner and advertiser. 257 6. From the scanner (central device), discover services. 258 7. Iterate over all the characteristics and descriptors of the 259 discovered features. 260 8. Disconnect the GATT connection. 261 262 Expected Result: 263 Verify that a connection was established and then disconnected 264 successfully. Verify that the services, characteristics, and descriptors 265 were discovered. 266 267 Returns: 268 Pass if True 269 Fail if False 270 271 TAGS: BR/EDR, Scanning, GATT, Services 272 Characteristics, Descriptors 273 Priority: 1 274 """ 275 gatt_server_cb = self.per_ad.droid.gattServerCreateGattServerCallback() 276 gatt_server = self.per_ad.droid.gattServerOpenGattServer( 277 gatt_server_cb) 278 self.gatt_server_list.append(gatt_server) 279 try: 280 bluetooth_gatt, gatt_callback, adv_callback = ( 281 orchestrate_gatt_connection(self.cen_ad, self.per_ad, 282 gatt_transport['bredr'], 283 self.per_droid_mac_address)) 284 self.bluetooth_gatt_list.append(bluetooth_gatt) 285 except GattTestUtilsError as err: 286 self.log.error(err) 287 return False 288 discovered_services_index = -1 289 if self.cen_ad.droid.gattClientDiscoverServices(bluetooth_gatt): 290 event = self.cen_ad.ed.pop_event( 291 gatt_cb_strings['gatt_serv_disc'].format(gatt_callback), 292 self.default_timeout) 293 discovered_services_index = event['data']['ServicesIndex'] 294 log_gatt_server_uuids(self.cen_ad, discovered_services_index) 295 return self._orchestrate_gatt_disconnection(bluetooth_gatt, 296 gatt_callback) 297 298 @BluetoothBaseTest.bt_test_wrap 299 @test_tracker_info(uuid='d4277bee-da99-4f48-8a4d-f81b5389da18') 300 def test_gatt_bredr_connect_with_service_uuid_variations(self): 301 """Test GATT connection with multiple service uuids. 302 303 Test establishing a gatt connection between a GATT server and GATT 304 client with multiple service uuid variations. 305 306 Steps: 307 1. Start a generic advertisement. 308 2. Start a generic scanner. 309 3. Find the advertisement and extract the mac address. 310 4. Stop the first scanner. 311 5. Create a GATT connection between the scanner and advertiser. 312 6. From the scanner (central device), discover services. 313 7. Verify that all the service uuid variations are found. 314 8. Disconnect the GATT connection. 315 316 Expected Result: 317 Verify that a connection was established and then disconnected 318 successfully. Verify that the service uuid variations are found. 319 320 Returns: 321 Pass if True 322 Fail if False 323 324 TAGS: BR/EDR, Scanning, GATT, Services 325 Priority: 2 326 """ 327 gatt_server_cb = self.per_ad.droid.gattServerCreateGattServerCallback() 328 gatt_server = self.per_ad.droid.gattServerOpenGattServer( 329 gatt_server_cb) 330 self.gatt_server_list.append(gatt_server) 331 try: 332 gatt_server_callback, gatt_server = setup_multiple_services( 333 self.per_ad) 334 self.gatt_server_list.append(gatt_server) 335 except GattTestUtilsError as err: 336 self.log.error(err) 337 return False 338 try: 339 bluetooth_gatt, gatt_callback, adv_callback = ( 340 orchestrate_gatt_connection(self.cen_ad, self.per_ad, 341 gatt_transport['bredr'], 342 self.per_droid_mac_address)) 343 self.bluetooth_gatt_list.append(bluetooth_gatt) 344 except GattTestUtilsError as err: 345 self.log.error(err) 346 return False 347 discovered_services_index = -1 348 if self.cen_ad.droid.gattClientDiscoverServices(bluetooth_gatt): 349 event = self.cen_ad.ed.pop_event( 350 gatt_cb_strings['gatt_serv_disc'].format(gatt_callback), 351 self.default_timeout) 352 discovered_services_index = event['data']['ServicesIndex'] 353 log_gatt_server_uuids(self.cen_ad, discovered_services_index) 354 return self._orchestrate_gatt_disconnection(bluetooth_gatt, 355 gatt_callback) 356 357 @BluetoothBaseTest.bt_test_wrap 358 @test_tracker_info(uuid='15c726dc-788a-4400-9a90-8c6866b24a3a') 359 def test_gatt_bredr_connect_multiple_iterations(self): 360 """Test GATT connections multiple times. 361 362 Test establishing a gatt connection between a GATT server and GATT 363 client with multiple iterations. 364 365 Steps: 366 1. Start a generic advertisement. 367 2. Start a generic scanner. 368 3. Find the advertisement and extract the mac address. 369 4. Stop the first scanner. 370 5. Create a GATT connection between the scanner and advertiser. 371 6. Disconnect the GATT connection. 372 7. Repeat steps 5 and 6 twenty times. 373 374 Expected Result: 375 Verify that a connection was established and then disconnected 376 successfully twenty times. 377 378 Returns: 379 Pass if True 380 Fail if False 381 382 TAGS: BR/EDR, Scanning, GATT, Stress 383 Priority: 1 384 """ 385 gatt_server_cb = self.per_ad.droid.gattServerCreateGattServerCallback() 386 gatt_server = self.per_ad.droid.gattServerOpenGattServer( 387 gatt_server_cb) 388 self.gatt_server_list.append(gatt_server) 389 autoconnect = False 390 mac_address = self.per_ad.droid.bluetoothGetLocalAddress() 391 for i in range(20): 392 try: 393 bluetooth_gatt, gatt_callback, adv_callback = ( 394 orchestrate_gatt_connection(self.cen_ad, self.per_ad, 395 gatt_transport['bredr'], 396 self.per_droid_mac_address)) 397 self.bluetooth_gatt_list.append(bluetooth_gatt) 398 except GattTestUtilsError as err: 399 self.log.error(err) 400 return False 401 self.log.info("Disconnecting from peripheral device.") 402 test_result = self._orchestrate_gatt_disconnection(bluetooth_gatt, 403 gatt_callback) 404 if not test_result: 405 self.log.info("Failed to disconnect from peripheral device.") 406 return False 407 return True 408 409 @BluetoothBaseTest.bt_test_wrap 410 @test_tracker_info(uuid='6ec766ca-6358-48ff-9d85-ede4d2756546') 411 def test_bredr_write_descriptor_stress(self): 412 """Test GATT connection writing and reading descriptors. 413 414 Test establishing a gatt connection between a GATT server and GATT 415 client with multiple service uuid variations. 416 417 Steps: 418 1. Start a generic advertisement. 419 2. Start a generic scanner. 420 3. Find the advertisement and extract the mac address. 421 4. Stop the first scanner. 422 5. Create a GATT connection between the scanner and advertiser. 423 6. Discover services. 424 7. Write data to the descriptors of each characteristic 100 times. 425 8. Read the data sent to the descriptors. 426 9. Disconnect the GATT connection. 427 428 Expected Result: 429 Each descriptor in each characteristic is written and read 100 times. 430 431 Returns: 432 Pass if True 433 Fail if False 434 435 TAGS: BR/EDR, Scanning, GATT, Stress, Characteristics, Descriptors 436 Priority: 1 437 """ 438 try: 439 gatt_server_callback, gatt_server = setup_multiple_services( 440 self.per_ad) 441 self.gatt_server_list.append(gatt_server) 442 except GattTestUtilsError as err: 443 self.log.error(err) 444 return False 445 try: 446 bluetooth_gatt, gatt_callback, adv_callback = ( 447 orchestrate_gatt_connection(self.cen_ad, self.per_ad, 448 gatt_transport['bredr'], 449 self.per_droid_mac_address)) 450 self.bluetooth_gatt_list.append(bluetooth_gatt) 451 except GattTestUtilsError as err: 452 self.log.error(err) 453 return False 454 if self.cen_ad.droid.gattClientDiscoverServices(bluetooth_gatt): 455 try: 456 event = self.cen_ad.ed.pop_event( 457 gatt_cb_strings['gatt_serv_disc'].format(gatt_callback), 458 self.default_timeout) 459 except Empty as err: 460 self.log.error("Event not found: {}".format(err)) 461 return False 462 discovered_services_index = event['data']['ServicesIndex'] 463 else: 464 self.log.info("Failed to discover services.") 465 return False 466 services_count = self.cen_ad.droid.gattClientGetDiscoveredServicesCount( 467 discovered_services_index) 468 469 connected_device_list = self.per_ad.droid.gattServerGetConnectedDevices( 470 gatt_server) 471 if len(connected_device_list) == 0: 472 self.log.info("No devices connected from peripheral.") 473 return False 474 bt_device_id = 0 475 status = 1 476 offset = 1 477 test_value = [1, 2, 3, 4, 5, 6, 7] 478 test_value_return = [1, 2, 3] 479 for i in range(services_count): 480 characteristic_uuids = ( 481 self.cen_ad.droid.gattClientGetDiscoveredCharacteristicUuids( 482 discovered_services_index, i)) 483 for characteristic in characteristic_uuids: 484 descriptor_uuids = ( 485 self.cen_ad.droid.gattClientGetDiscoveredDescriptorUuids( 486 discovered_services_index, i, characteristic)) 487 for _ in range(100): 488 for descriptor in descriptor_uuids: 489 self.cen_ad.droid.gattClientDescriptorSetValue( 490 bluetooth_gatt, discovered_services_index, i, 491 characteristic, descriptor, test_value) 492 self.cen_ad.droid.gattClientWriteDescriptor( 493 bluetooth_gatt, discovered_services_index, i, 494 characteristic, descriptor) 495 event = self.per_ad.ed.pop_event(gatt_cb_strings[ 496 'desc_write_req'].format(gatt_server_callback), 497 self.default_timeout) 498 self.log.info( 499 "onDescriptorWriteRequest event found: {}".format( 500 event)) 501 request_id = event['data']['requestId'] 502 found_value = event['data']['value'] 503 if found_value != test_value: 504 self.log.info("Values didn't match. Found: {}, " 505 "Expected: {}".format(found_value, 506 test_value)) 507 return False 508 self.per_ad.droid.gattServerSendResponse( 509 gatt_server, bt_device_id, request_id, status, 510 offset, test_value_return) 511 self.log.info( 512 "onDescriptorWrite event found: {}".format( 513 self.cen_ad.ed.pop_event(gatt_cb_strings[ 514 'desc_write'].format( 515 gatt_callback), self.default_timeout))) 516 return True 517