class DirectHciTest(GdBaseTestClass): def setup_class(self): super().setup_class(dut_module='HCI', cert_module='HAL') def setup_test(self): super().setup_test() self.dut_hci = PyHci(self.dut, acl_streaming=True) self.cert_hal = PyHal(self.cert) self.cert_hal.send_hci_command(ResetBuilder()) def teardown_test(self): self.dut_hci.close() self.cert_hal.close() super().teardown_test() def enqueue_acl_data(self, handle, pb_flag, b_flag, data): acl = AclBuilder(handle, pb_flag, b_flag, RawBuilder(data)) self.dut.hci.SendAcl(common.Data(payload=bytes(acl.Serialize()))) def test_local_hci_cmd_and_event(self): # Loopback mode responds with ACL and SCO connection complete self.dut_hci.register_for_events(EventCode.LOOPBACK_COMMAND) self.dut_hci.send_command( WriteLoopbackModeBuilder(LoopbackMode.ENABLE_LOCAL)) self.dut_hci.send_command(ReadLocalNameBuilder()) assertThat(self.dut_hci.get_event_stream()).emits( HciMatchers.LoopbackOf(ReadLocalNameBuilder())) def test_inquiry_from_dut(self): self.dut_hci.register_for_events(EventCode.INQUIRY_RESULT) self.cert_hal.enable_inquiry_and_page_scan() lap = Lap() lap.lap = 0x33 self.dut_hci.send_command(InquiryBuilder(lap, 0x30, 0xff)) assertThat(self.dut_hci.get_event_stream()).emits( HciMatchers.EventWithCode(EventCode.INQUIRY_RESULT)) def test_le_ad_scan_cert_advertises(self): self.dut_hci.register_for_le_events( SubeventCode.EXTENDED_ADVERTISING_REPORT, SubeventCode.ADVERTISING_REPORT) # DUT Scans self.dut_hci.send_command( LeSetRandomAddressBuilder('0D:05:04:03:02:01')) phy_scan_params = PhyScanParameters() phy_scan_params.le_scan_interval = 6553 phy_scan_params.le_scan_window = 6553 phy_scan_params.le_scan_type = LeScanType.ACTIVE self.dut_hci.send_command( LeSetExtendedScanParametersBuilder( OwnAddressType.RANDOM_DEVICE_ADDRESS, LeScanningFilterPolicy.ACCEPT_ALL, 1, [phy_scan_params])) self.dut_hci.send_command( LeSetExtendedScanEnableBuilder(Enable.ENABLED, FilterDuplicates.DISABLED, 0, 0)) # CERT Advertises advertising_handle = 0 self.cert_hal.send_hci_command( LeSetExtendedAdvertisingLegacyParametersBuilder( advertising_handle, LegacyAdvertisingProperties.ADV_IND, 512, 768, 7, OwnAddressType.RANDOM_DEVICE_ADDRESS, PeerAddressType.PUBLIC_DEVICE_OR_IDENTITY_ADDRESS, 'A6:A5:A4:A3:A2:A1', AdvertisingFilterPolicy.ALL_DEVICES, 0xF7, 1, # SID Enable.DISABLED # Scan request notification )) self.cert_hal.send_hci_command( LeSetExtendedAdvertisingRandomAddressBuilder( advertising_handle, '0C:05:04:03:02:01')) gap_name = GapData() gap_name.data_type = GapDataType.COMPLETE_LOCAL_NAME gap_name.data = list(bytes(b'Im_A_Cert')) self.cert_hal.send_hci_command( LeSetExtendedAdvertisingDataBuilder( advertising_handle, Operation.COMPLETE_ADVERTISEMENT, FragmentPreference.CONTROLLER_SHOULD_NOT, [gap_name])) gap_short_name = GapData() gap_short_name.data_type = GapDataType.SHORTENED_LOCAL_NAME gap_short_name.data = list(bytes(b'Im_A_C')) self.cert_hal.send_hci_command( LeSetExtendedAdvertisingScanResponseBuilder( advertising_handle, Operation.COMPLETE_ADVERTISEMENT, FragmentPreference.CONTROLLER_SHOULD_NOT, [gap_short_name])) enabled_set = EnabledSet() enabled_set.advertising_handle = 0 enabled_set.duration = 0 enabled_set.max_extended_advertising_events = 0 self.cert_hal.send_hci_command( LeSetExtendedAdvertisingEnableBuilder(Enable.ENABLED, [enabled_set])) assertThat(self.dut_hci.get_le_event_stream()).emits( lambda packet: b'Im_A_Cert' in packet.payload) self.cert_hal.send_hci_command( LeSetExtendedAdvertisingEnableBuilder(Enable.DISABLED, [enabled_set])) self.dut_hci.send_command( LeSetExtendedScanEnableBuilder(Enable.DISABLED, FilterDuplicates.DISABLED, 0, 0)) def _verify_le_connection_complete(self): cert_conn_complete_capture = HalCaptures.LeConnectionCompleteCapture() assertThat(self.cert_hal.get_hci_event_stream()).emits( cert_conn_complete_capture) cert_handle = cert_conn_complete_capture.get().GetConnectionHandle() dut_conn_complete_capture = HciCaptures.LeConnectionCompleteCapture() assertThat(self.dut_hci.get_le_event_stream()).emits( dut_conn_complete_capture) dut_handle = dut_conn_complete_capture.get().GetConnectionHandle() return (dut_handle, cert_handle) @staticmethod def _create_phy_scan_params(): phy_scan_params = LeCreateConnPhyScanParameters() phy_scan_params.scan_interval = 0x60 phy_scan_params.scan_window = 0x30 phy_scan_params.conn_interval_min = 0x18 phy_scan_params.conn_interval_max = 0x28 phy_scan_params.conn_latency = 0 phy_scan_params.supervision_timeout = 0x1f4 phy_scan_params.min_ce_length = 0 phy_scan_params.max_ce_length = 0 return phy_scan_params def test_le_connection_dut_advertises(self): self.dut_hci.register_for_le_events( SubeventCode.CONNECTION_COMPLETE, SubeventCode.ADVERTISING_SET_TERMINATED, SubeventCode.ENHANCED_CONNECTION_COMPLETE, SubeventCode.READ_REMOTE_FEATURES_COMPLETE) # Cert Connects self.cert_hal.send_hci_command( LeSetRandomAddressBuilder('0C:05:04:03:02:01')) phy_scan_params = DirectHciTest._create_phy_scan_params() self.cert_hal.send_hci_command( LeExtendedCreateConnectionBuilder( InitiatorFilterPolicy.USE_PEER_ADDRESS, OwnAddressType.RANDOM_DEVICE_ADDRESS, AddressType.RANDOM_DEVICE_ADDRESS, '0D:05:04:03:02:01', 1, [phy_scan_params])) advertisement = self.dut_hci.create_advertisement( 0, '0D:05:04:03:02:01') advertisement.set_data(b'Im_The_DUT') advertisement.set_scan_response(b'Im_The_D') advertisement.start() (dut_handle, cert_handle) = self._verify_le_connection_complete() self.dut_hci.send_command(LeReadRemoteFeaturesBuilder(dut_handle)) assertThat(self.dut_hci.get_le_event_stream()).emits( lambda packet: packet.payload[0] == int( EventCode.LE_META_EVENT) and packet.payload[2] == int( SubeventCode.READ_REMOTE_FEATURES_COMPLETE)) # Send ACL Data self.enqueue_acl_data( dut_handle, PacketBoundaryFlag.FIRST_NON_AUTOMATICALLY_FLUSHABLE, BroadcastFlag.POINT_TO_POINT, bytes(b'Just SomeAclData')) self.cert_hal.send_acl_first(cert_handle, bytes(b'Just SomeMoreAclData')) assertThat( self.cert_hal.get_acl_stream()).emits(lambda packet: logging.debug( packet.payload) or b'SomeAclData' in packet.payload) assertThat(self.dut_hci.get_raw_acl_stream()).emits( lambda packet: logging.debug( packet.payload) or b'SomeMoreAclData' in packet.payload) def test_le_connect_list_connection_cert_advertises(self): self.dut_hci.register_for_le_events( SubeventCode.CONNECTION_COMPLETE, SubeventCode.ENHANCED_CONNECTION_COMPLETE) # DUT Connects self.dut_hci.send_command( LeSetRandomAddressBuilder('0D:05:04:03:02:01')) self.dut_hci.send_command( LeAddDeviceToConnectListBuilder(ConnectListAddressType.RANDOM, '0C:05:04:03:02:01')) phy_scan_params = DirectHciTest._create_phy_scan_params() self.dut_hci.send_command( LeExtendedCreateConnectionBuilder( InitiatorFilterPolicy.USE_CONNECT_LIST, OwnAddressType.RANDOM_DEVICE_ADDRESS, AddressType.RANDOM_DEVICE_ADDRESS, 'BA:D5:A4:A3:A2:A1', 1, [phy_scan_params])) advertisement = self.cert_hal.create_advertisement( 1, '0C:05:04:03:02:01', min_interval=512, max_interval=768, peer_address='A6:A5:A4:A3:A2:A1', tx_power=0x7f, sid=0) advertisement.set_data(b'Im_A_Cert') advertisement.start() # LeConnectionComplete self._verify_le_connection_complete() def test_connection_dut_connects(self): self.dut_hci.send_command(WritePageTimeoutBuilder(0x4000)) self.cert_hal.enable_inquiry_and_page_scan() address = self.cert_hal.read_own_address() self.dut_hci.initiate_connection(address) cert_acl = self.cert_hal.accept_connection() dut_acl = self.dut_hci.complete_connection() # Send ACL Data dut_acl.send_first(b'Just SomeAclData') cert_acl.send_first(b'Just SomeMoreAclData') assertThat(self.cert_hal.get_acl_stream()).emits( lambda packet: b'SomeAclData' in packet.payload) assertThat(self.dut_hci.get_raw_acl_stream()).emits( lambda packet: b'SomeMoreAclData' in packet.payload) def test_connection_cert_connects(self): self.cert_hal.send_hci_command(WritePageTimeoutBuilder(0x4000)) self.dut_hci.enable_inquiry_and_page_scan() address = self.dut_hci.read_own_address() self.cert_hal.initiate_connection(address) dut_acl = self.dut_hci.accept_connection() cert_acl = self.cert_hal.complete_connection() # Send ACL Data dut_acl.send_first(b'This is just SomeAclData') cert_acl.send_first(b'This is just SomeMoreAclData') assertThat(self.cert_hal.get_acl_stream()).emits( lambda packet: b'SomeAclData' in packet.payload) assertThat(self.dut_hci.get_raw_acl_stream()).emits( lambda packet: b'SomeMoreAclData' in packet.payload)
class AclManagerTest(GdBaseTestClass): def setup_class(self): super().setup_class(dut_module='HCI_INTERFACES', cert_module='HCI') # todo: move into GdBaseTestClass, based on modules inited def setup_test(self): super().setup_test() self.cert_hci = PyHci(self.cert, acl_streaming=True) self.dut_acl_manager = PyAclManager(self.dut) def teardown_test(self): self.cert_hci.close() self.dut_acl_manager.close() super().teardown_test() def test_dut_connects(self): self.cert_hci.enable_inquiry_and_page_scan() cert_address = self.cert_hci.read_own_address() with self.dut_acl_manager.initiate_connection(cert_address) as dut_acl: cert_acl = self.cert_hci.accept_connection() cert_acl.send_first( b'\x26\x00\x07\x00This is just SomeAclData from the Cert') dut_acl.wait_for_connection_complete() dut_acl.send( b'\x29\x00\x07\x00This is just SomeMoreAclData from the DUT') assertThat(cert_acl).emits( lambda packet: b'SomeMoreAclData' in packet.data) assertThat(dut_acl).emits( lambda packet: b'SomeAclData' in packet.payload) def test_cert_connects(self): dut_address = self.dut.hci_controller.GetMacAddressSimple() self.dut.neighbor.EnablePageScan( neighbor_facade.EnableMsg(enabled=True)) self.dut_acl_manager.listen_for_incoming_connections() self.cert_hci.initiate_connection(dut_address) dut_acl = self.dut_acl_manager.accept_connection() cert_acl = self.cert_hci.complete_connection() dut_acl.send( b'\x29\x00\x07\x00This is just SomeMoreAclData from the DUT') cert_acl.send_first( b'\x26\x00\x07\x00This is just SomeAclData from the Cert') assertThat(cert_acl).emits( lambda packet: b'SomeMoreAclData' in packet.data) assertThat(dut_acl).emits( lambda packet: b'SomeAclData' in packet.payload) def test_recombination_l2cap_packet(self): self.cert_hci.enable_inquiry_and_page_scan() cert_address = self.cert_hci.read_own_address() with self.dut_acl_manager.initiate_connection(cert_address) as dut_acl: cert_acl = self.cert_hci.accept_connection() cert_acl.send_first(b'\x06\x00\x07\x00Hello') cert_acl.send_continuing(b'!') cert_acl.send_first(b'\xe8\x03\x07\x00' + b'Hello' * 200) dut_acl.wait_for_connection_complete() assertThat(dut_acl).emits( lambda packet: b'Hello!' in packet.payload, lambda packet: b'Hello' * 200 in packet.payload).inOrder()
class AclManagerTest(GdBaseTestClass): def setup_class(self): super().setup_class(dut_module='HCI_INTERFACES', cert_module='HCI') # todo: move into GdBaseTestClass, based on modules inited def setup_test(self): super().setup_test() self.cert_hci = PyHci(self.cert, acl_streaming=True) self.dut_acl_manager = PyAclManager(self.dut) def teardown_test(self): self.cert_hci.close() super().teardown_test() def test_dut_connects(self): self.cert_hci.enable_inquiry_and_page_scan() cert_address = self.cert_hci.read_own_address() self.dut_acl_manager.initiate_connection(cert_address) cert_acl = self.cert_hci.accept_connection() with self.dut_acl_manager.complete_outgoing_connection() as dut_acl: cert_acl.send_first( b'\x26\x00\x07\x00This is just SomeAclData from the Cert') dut_acl.send( b'\x29\x00\x07\x00This is just SomeMoreAclData from the DUT') assertThat(cert_acl).emits( lambda packet: b'SomeMoreAclData' in packet.payload) assertThat(dut_acl).emits( lambda packet: b'SomeAclData' in packet.payload) def test_cert_connects(self): dut_address = self.dut.hci_controller.GetMacAddressSimple() self.dut.neighbor.EnablePageScan( neighbor_facade.EnableMsg(enabled=True)) self.dut_acl_manager.listen_for_an_incoming_connection() self.cert_hci.initiate_connection(dut_address) with self.dut_acl_manager.complete_incoming_connection() as dut_acl: cert_acl = self.cert_hci.complete_connection() dut_acl.send( b'\x29\x00\x07\x00This is just SomeMoreAclData from the DUT') cert_acl.send_first( b'\x26\x00\x07\x00This is just SomeAclData from the Cert') assertThat(cert_acl).emits( lambda packet: b'SomeMoreAclData' in packet.payload) assertThat(dut_acl).emits( lambda packet: b'SomeAclData' in packet.payload) def test_reject_broadcast(self): dut_address = self.dut.hci_controller.GetMacAddressSimple() self.dut.neighbor.EnablePageScan( neighbor_facade.EnableMsg(enabled=True)) self.dut_acl_manager.listen_for_an_incoming_connection() self.cert_hci.initiate_connection(dut_address) with self.dut_acl_manager.complete_incoming_connection() as dut_acl: cert_acl = self.cert_hci.complete_connection() cert_acl.send( hci_packets.PacketBoundaryFlag.FIRST_AUTOMATICALLY_FLUSHABLE, hci_packets.BroadcastFlag.ACTIVE_PERIPHERAL_BROADCAST, b'\x26\x00\x07\x00This is a Broadcast from the Cert') assertThat(dut_acl).emitsNone() cert_acl.send( hci_packets.PacketBoundaryFlag.FIRST_AUTOMATICALLY_FLUSHABLE, hci_packets.BroadcastFlag.POINT_TO_POINT, b'\x26\x00\x07\x00This is just SomeAclData from the Cert') assertThat(dut_acl).emits( lambda packet: b'SomeAclData' in packet.payload) def test_cert_connects_disconnects(self): dut_address = self.dut.hci_controller.GetMacAddressSimple() self.dut.neighbor.EnablePageScan( neighbor_facade.EnableMsg(enabled=True)) self.dut_acl_manager.listen_for_an_incoming_connection() self.cert_hci.initiate_connection(dut_address) with self.dut_acl_manager.complete_incoming_connection() as dut_acl: cert_acl = self.cert_hci.complete_connection() dut_acl.send( b'\x29\x00\x07\x00This is just SomeMoreAclData from the DUT') cert_acl.send_first( b'\x26\x00\x07\x00This is just SomeAclData from the Cert') assertThat(cert_acl).emits( lambda packet: b'SomeMoreAclData' in packet.payload) assertThat(dut_acl).emits( lambda packet: b'SomeAclData' in packet.payload) dut_acl.disconnect( hci_packets.DisconnectReason.REMOTE_USER_TERMINATED_CONNECTION) dut_acl.wait_for_disconnection_complete() def test_recombination_l2cap_packet(self): self.cert_hci.enable_inquiry_and_page_scan() cert_address = self.cert_hci.read_own_address() self.dut_acl_manager.initiate_connection(cert_address) cert_acl = self.cert_hci.accept_connection() with self.dut_acl_manager.complete_outgoing_connection() as dut_acl: cert_acl.send_first(b'\x06\x00\x07\x00Hello') cert_acl.send_continuing(b'!') cert_acl.send_first(b'\xe8\x03\x07\x00' + b'Hello' * 200) assertThat(dut_acl).emits( lambda packet: b'Hello!' in packet.payload, lambda packet: b'Hello' * 200 in packet.payload).inOrder()