blob: b9d86426d25f8385cbd75777676c80761ff53f92 [file] [log] [blame]
#!/usr/bin/env python3
#
# Copyright 2019 - The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from datetime import timedelta
import os
import sys
import logging
from cert.gd_base_test_facade_only import GdFacadeOnlyBaseTestClass
from cert.event_callback_stream import EventCallbackStream
from cert.event_asserts import EventAsserts
from google.protobuf import empty_pb2 as empty_proto
from facade import rootservice_pb2 as facade_rootservice
from hal import facade_pb2 as hal_facade
from hci.facade import facade_pb2 as hci_facade
from bluetooth_packets_python3 import hci_packets
import bluetooth_packets_python3 as bt_packets
class DirectHciTest(GdFacadeOnlyBaseTestClass):
def setup_test(self):
self.device_under_test.rootservice.StartStack(
facade_rootservice.StartStackRequest(
module_under_test=facade_rootservice.BluetoothModule.Value(
'HCI'),))
self.cert_device.rootservice.StartStack(
facade_rootservice.StartStackRequest(
module_under_test=facade_rootservice.BluetoothModule.Value(
'HAL'),))
self.device_under_test.wait_channel_ready()
self.cert_device.wait_channel_ready()
self.cert_device.hal.SendHciCommand(
hal_facade.HciCommandPacket(
payload=bytes(hci_packets.ResetBuilder().Serialize())))
def teardown_test(self):
self.device_under_test.rootservice.StopStack(
facade_rootservice.StopStackRequest())
self.cert_device.rootservice.StopStack(
facade_rootservice.StopStackRequest())
def register_for_event(self, event_code):
msg = hci_facade.EventCodeMsg(code=int(event_code))
self.device_under_test.hci.RegisterEventHandler(msg)
def register_for_le_event(self, event_code):
msg = hci_facade.LeSubeventCodeMsg(code=int(event_code))
self.device_under_test.hci.RegisterLeEventHandler(msg)
def enqueue_hci_command(self, command, expect_complete):
cmd_bytes = bytes(command.Serialize())
cmd = hci_facade.CommandMsg(command=cmd_bytes)
if (expect_complete):
self.device_under_test.hci.EnqueueCommandWithComplete(cmd)
else:
self.device_under_test.hci.EnqueueCommandWithStatus(cmd)
def send_hal_hci_command(self, command):
self.cert_device.hal.SendHciCommand(
hal_facade.HciCommandPacket(payload=bytes(command.Serialize())))
def enqueue_acl_data(self, handle, pb_flag, b_flag, acl):
acl_msg = hci_facade.AclMsg(
handle=int(handle),
packet_boundary_flag=int(pb_flag),
broadcast_flag=int(b_flag),
data=acl)
self.device_under_test.hci.SendAclData(acl_msg)
def send_hal_acl_data(self, handle, pb_flag, b_flag, acl):
lower = handle & 0xff
upper = (handle >> 8) & 0xf
upper = upper | int(pb_flag) & 0x3
upper = upper | ((int(b_flag) & 0x3) << 2)
lower_length = len(acl) & 0xff
upper_length = (len(acl) & 0xff00) >> 8
concatenated = bytes([lower, upper, lower_length, upper_length] +
list(acl))
self.cert_device.hal.SendHciAcl(
hal_facade.HciAclPacket(payload=concatenated))
def test_local_hci_cmd_and_event(self):
# Loopback mode responds with ACL and SCO connection complete
self.register_for_event(hci_packets.EventCode.CONNECTION_COMPLETE)
self.register_for_event(hci_packets.EventCode.LOOPBACK_COMMAND)
self.register_for_event(
hci_packets.EventCode.CONNECTION_PACKET_TYPE_CHANGED)
with EventCallbackStream(
self.device_under_test.hci.FetchEvents(
empty_proto.Empty())) as hci_event_stream:
hci_event_asserts = EventAsserts(hci_event_stream)
self.enqueue_hci_command(
hci_packets.WriteLoopbackModeBuilder(
hci_packets.LoopbackMode.ENABLE_LOCAL), True)
cmd2loop = hci_packets.ReadLocalNameBuilder()
self.enqueue_hci_command(cmd2loop, True)
looped_bytes = bytes(cmd2loop.Serialize())
hci_event_asserts.assert_event_occurs(
lambda packet: looped_bytes in packet.event)
def test_inquiry_from_dut(self):
self.register_for_event(hci_packets.EventCode.INQUIRY_RESULT)
with EventCallbackStream(
self.device_under_test.hci.FetchEvents(
empty_proto.Empty())) as hci_event_stream:
hci_event_asserts = EventAsserts(hci_event_stream)
self.send_hal_hci_command(
hci_packets.WriteScanEnableBuilder(
hci_packets.ScanEnable.INQUIRY_AND_PAGE_SCAN))
lap = hci_packets.Lap()
lap.lap = 0x33
self.enqueue_hci_command(
hci_packets.InquiryBuilder(lap, 0x30, 0xff), False)
hci_event_asserts.assert_event_occurs(
lambda packet: b'\x02\x0f' in packet.event
# Expecting an HCI Event (code 0x02, length 0x0f)
)
def test_le_ad_scan_cert_advertises(self):
self.register_for_le_event(
hci_packets.SubeventCode.EXTENDED_ADVERTISING_REPORT)
self.register_for_le_event(hci_packets.SubeventCode.ADVERTISING_REPORT)
with EventCallbackStream(
self.device_under_test.hci.FetchLeSubevents(
empty_proto.Empty())) as hci_le_event_stream:
hci_event_asserts = EventAsserts(hci_le_event_stream)
# DUT Scans
self.enqueue_hci_command(
hci_packets.LeSetRandomAddressBuilder('0D:05:04:03:02:01'),
True)
phy_scan_params = hci_packets.PhyScanParameters()
phy_scan_params.le_scan_interval = 6553
phy_scan_params.le_scan_window = 6553
phy_scan_params.le_scan_type = hci_packets.LeScanType.ACTIVE
self.enqueue_hci_command(
hci_packets.LeSetExtendedScanParametersBuilder(
hci_packets.AddressType.RANDOM_DEVICE_ADDRESS,
hci_packets.LeSetScanningFilterPolicy.ACCEPT_ALL, 1,
[phy_scan_params]), True)
self.enqueue_hci_command(
hci_packets.LeSetExtendedScanEnableBuilder(
hci_packets.Enable.ENABLED,
hci_packets.FilterDuplicates.DISABLED, 0, 0), True)
# CERT Advertises
advertising_handle = 0
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingLegacyParametersBuilder(
advertising_handle,
hci_packets.LegacyAdvertisingProperties.ADV_IND,
512,
768,
7,
hci_packets.OwnAddressType.RANDOM_DEVICE_ADDRESS,
hci_packets.PeerAddressType.
PUBLIC_DEVICE_OR_IDENTITY_ADDRESS,
'A6:A5:A4:A3:A2:A1',
hci_packets.AdvertisingFilterPolicy.ALL_DEVICES,
0xF7,
1, # SID
hci_packets.Enable.DISABLED # Scan request notification
))
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingRandomAddressBuilder(
advertising_handle, '0C:05:04:03:02:01'))
gap_name = hci_packets.GapData()
gap_name.data_type = hci_packets.GapDataType.COMPLETE_LOCAL_NAME
gap_name.data = list(bytes(b'Im_A_Cert!')) # TODO: Fix and remove !
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingDataBuilder(
advertising_handle,
hci_packets.Operation.COMPLETE_ADVERTISEMENT,
hci_packets.FragmentPreference.CONTROLLER_SHOULD_NOT,
[gap_name]))
gap_short_name = hci_packets.GapData()
gap_short_name.data_type = hci_packets.GapDataType.SHORTENED_LOCAL_NAME
gap_short_name.data = list(bytes(b'Im_A_C'))
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingScanResponseBuilder(
advertising_handle,
hci_packets.Operation.COMPLETE_ADVERTISEMENT,
hci_packets.FragmentPreference.CONTROLLER_SHOULD_NOT,
[gap_short_name]))
enabled_set = hci_packets.EnabledSet()
enabled_set.advertising_handle = 0
enabled_set.duration = 0
enabled_set.max_extended_advertising_events = 0
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingEnableBuilder(
hci_packets.Enable.ENABLED, [enabled_set]))
hci_event_asserts.assert_event_occurs(
lambda packet: b'Im_A_Cert' in packet.event)
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingEnableBuilder(
hci_packets.Enable.DISABLED, [enabled_set]))
self.enqueue_hci_command(
hci_packets.LeSetExtendedScanEnableBuilder(
hci_packets.Enable.DISABLED,
hci_packets.FilterDuplicates.DISABLED, 0, 0), True)
def test_le_connection_dut_advertises(self):
self.register_for_le_event(hci_packets.SubeventCode.CONNECTION_COMPLETE)
with EventCallbackStream(self.device_under_test.hci.FetchLeSubevents(empty_proto.Empty())) as le_event_stream, \
EventCallbackStream(self.device_under_test.hci.FetchEvents(empty_proto.Empty())) as event_stream, \
EventCallbackStream(self.device_under_test.hci.FetchAclPackets(empty_proto.Empty())) as acl_data_stream, \
EventCallbackStream(self.cert_device.hal.FetchHciEvent(empty_proto.Empty())) as cert_hci_event_stream, \
EventCallbackStream(self.cert_device.hal.FetchHciAcl(empty_proto.Empty())) as cert_acl_data_stream:
le_event_asserts = EventAsserts(le_event_stream)
event_asserts = EventAsserts(event_stream)
cert_hci_event_asserts = EventAsserts(cert_hci_event_stream)
acl_data_asserts = EventAsserts(acl_data_stream)
cert_acl_data_asserts = EventAsserts(cert_acl_data_stream)
# Cert Connects
self.send_hal_hci_command(
hci_packets.LeSetRandomAddressBuilder('0C:05:04:03:02:01'))
phy_scan_params = hci_packets.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
self.send_hal_hci_command(
hci_packets.LeExtendedCreateConnectionBuilder(
hci_packets.InitiatorFilterPolicy.USE_PEER_ADDRESS,
hci_packets.OwnAddressType.RANDOM_DEVICE_ADDRESS,
hci_packets.AddressType.RANDOM_DEVICE_ADDRESS,
'0D:05:04:03:02:01', 1, [phy_scan_params]))
# DUT Advertises
advertising_handle = 0
self.enqueue_hci_command(
hci_packets.LeSetExtendedAdvertisingLegacyParametersBuilder(
advertising_handle,
hci_packets.LegacyAdvertisingProperties.ADV_IND,
400,
450,
7,
hci_packets.OwnAddressType.RANDOM_DEVICE_ADDRESS,
hci_packets.PeerAddressType.
PUBLIC_DEVICE_OR_IDENTITY_ADDRESS,
'00:00:00:00:00:00',
hci_packets.AdvertisingFilterPolicy.ALL_DEVICES,
0xF8,
1, #SID
hci_packets.Enable.DISABLED # Scan request notification
),
True)
self.enqueue_hci_command(
hci_packets.LeSetExtendedAdvertisingRandomAddressBuilder(
advertising_handle, '0D:05:04:03:02:01'), True)
gap_name = hci_packets.GapData()
gap_name.data_type = hci_packets.GapDataType.COMPLETE_LOCAL_NAME
gap_name.data = list(
bytes(b'Im_The_DUT!')) # TODO: Fix and remove !
self.enqueue_hci_command(
hci_packets.LeSetExtendedAdvertisingDataBuilder(
advertising_handle,
hci_packets.Operation.COMPLETE_ADVERTISEMENT,
hci_packets.FragmentPreference.CONTROLLER_SHOULD_NOT,
[gap_name]), True)
gap_short_name = hci_packets.GapData()
gap_short_name.data_type = hci_packets.GapDataType.SHORTENED_LOCAL_NAME
gap_short_name.data = list(bytes(b'Im_The_D'))
self.enqueue_hci_command(
hci_packets.LeSetExtendedAdvertisingScanResponseBuilder(
advertising_handle,
hci_packets.Operation.COMPLETE_ADVERTISEMENT,
hci_packets.FragmentPreference.CONTROLLER_SHOULD_NOT,
[gap_short_name]), True)
enabled_set = hci_packets.EnabledSet()
enabled_set.advertising_handle = advertising_handle
enabled_set.duration = 0
enabled_set.max_extended_advertising_events = 0
self.enqueue_hci_command(
hci_packets.LeSetExtendedAdvertisingEnableBuilder(
hci_packets.Enable.ENABLED, [enabled_set]), True)
# Check for success of Enable
event_asserts.assert_event_occurs(
lambda packet: b'\x0e\x04\x01\x39\x20\x00' in packet.event)
conn_handle = 0xfff
def event_handle(packet):
packet_bytes = packet.event
if b'\x3e\x13\x01\x00' in packet_bytes:
nonlocal conn_handle
cc_view = hci_packets.LeConnectionCompleteView(
hci_packets.LeMetaEventView(
hci_packets.EventPacketView(
bt_packets.PacketViewLittleEndian(
list(packet_bytes)))))
conn_handle = cc_view.GetConnectionHandle()
return True
return False
def payload_handle(packet):
packet_bytes = packet.payload
if b'\x3e\x13\x01\x00' in packet_bytes:
nonlocal conn_handle
cc_view = hci_packets.LeConnectionCompleteView(
hci_packets.LeMetaEventView(
hci_packets.EventPacketView(
bt_packets.PacketViewLittleEndian(
list(packet_bytes)))))
conn_handle = cc_view.GetConnectionHandle()
return True
return False
cert_hci_event_asserts.assert_event_occurs(payload_handle)
cert_handle = conn_handle
conn_handle = 0xfff
le_event_asserts.assert_event_occurs(event_handle)
dut_handle = conn_handle
if dut_handle == 0xfff:
logging.warning("Failed to get the DUT handle")
return False
if cert_handle == 0xfff:
logging.warning("Failed to get the CERT handle")
return False
# Send ACL Data
self.enqueue_acl_data(
dut_handle, hci_packets.PacketBoundaryFlag.
FIRST_NON_AUTOMATICALLY_FLUSHABLE,
hci_packets.BroadcastFlag.POINT_TO_POINT,
bytes(b'Just SomeAclData'))
self.send_hal_acl_data(
cert_handle, hci_packets.PacketBoundaryFlag.
FIRST_NON_AUTOMATICALLY_FLUSHABLE,
hci_packets.BroadcastFlag.POINT_TO_POINT,
bytes(b'Just SomeMoreAclData'))
cert_acl_data_asserts.assert_event_occurs(
lambda packet: logging.debug(packet.payload) or b'SomeAclData' in packet.payload
)
acl_data_asserts.assert_event_occurs(
lambda packet: logging.debug(packet.data) or b'SomeMoreAclData' in packet.data
)
def test_le_white_list_connection_cert_advertises(self):
self.register_for_le_event(hci_packets.SubeventCode.CONNECTION_COMPLETE)
with EventCallbackStream(self.device_under_test.hci.FetchLeSubevents(empty_proto.Empty())) as le_event_stream, \
EventCallbackStream(self.cert_device.hal.FetchHciEvent(empty_proto.Empty())) as cert_hci_event_stream:
le_event_asserts = EventAsserts(le_event_stream)
cert_hci_event_asserts = EventAsserts(cert_hci_event_stream)
# DUT Connects
self.enqueue_hci_command(
hci_packets.LeSetRandomAddressBuilder('0D:05:04:03:02:01'),
True)
self.enqueue_hci_command(
hci_packets.LeAddDeviceToWhiteListBuilder(
hci_packets.WhiteListAddressType.RANDOM,
'0C:05:04:03:02:01'), True)
phy_scan_params = hci_packets.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
self.enqueue_hci_command(
hci_packets.LeExtendedCreateConnectionBuilder(
hci_packets.InitiatorFilterPolicy.USE_WHITE_LIST,
hci_packets.OwnAddressType.RANDOM_DEVICE_ADDRESS,
hci_packets.AddressType.RANDOM_DEVICE_ADDRESS,
'BA:D5:A4:A3:A2:A1', 1, [phy_scan_params]), False)
# CERT Advertises
advertising_handle = 1
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingLegacyParametersBuilder(
advertising_handle,
hci_packets.LegacyAdvertisingProperties.ADV_IND,
512,
768,
7,
hci_packets.OwnAddressType.RANDOM_DEVICE_ADDRESS,
hci_packets.PeerAddressType.
PUBLIC_DEVICE_OR_IDENTITY_ADDRESS,
'A6:A5:A4:A3:A2:A1',
hci_packets.AdvertisingFilterPolicy.ALL_DEVICES,
0x7F,
0, # SID
hci_packets.Enable.DISABLED # Scan request notification
))
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingRandomAddressBuilder(
advertising_handle, '0C:05:04:03:02:01'))
gap_name = hci_packets.GapData()
gap_name.data_type = hci_packets.GapDataType.COMPLETE_LOCAL_NAME
gap_name.data = list(bytes(b'Im_A_Cert!')) # TODO: Fix and remove !
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingDataBuilder(
advertising_handle,
hci_packets.Operation.COMPLETE_ADVERTISEMENT,
hci_packets.FragmentPreference.CONTROLLER_SHOULD_NOT,
[gap_name]))
enabled_set = hci_packets.EnabledSet()
enabled_set.advertising_handle = 1
enabled_set.duration = 0
enabled_set.max_extended_advertising_events = 0
self.send_hal_hci_command(
hci_packets.LeSetExtendedAdvertisingEnableBuilder(
hci_packets.Enable.ENABLED, [enabled_set]))
# LeConnectionComplete
cert_hci_event_asserts.assert_event_occurs(
lambda packet: b'\x3e\x13\x01\x00' in packet.payload,
timeout=timedelta(seconds=20))
le_event_asserts.assert_event_occurs(
lambda packet: b'\x3e\x13\x01\x00' in packet.event)
def test_connection_dut_connects(self):
self.register_for_event(hci_packets.EventCode.CONNECTION_COMPLETE)
self.register_for_event(
hci_packets.EventCode.CONNECTION_PACKET_TYPE_CHANGED)
self.enqueue_hci_command(
hci_packets.WritePageTimeoutBuilder(0x4000), True)
with EventCallbackStream(self.device_under_test.hci.FetchEvents(empty_proto.Empty())) as hci_event_stream, \
EventCallbackStream(self.device_under_test.hci.FetchAclPackets(empty_proto.Empty())) as acl_data_stream, \
EventCallbackStream(self.cert_device.hal.FetchHciEvent(empty_proto.Empty())) as cert_hci_event_stream, \
EventCallbackStream(self.cert_device.hal.FetchHciAcl(empty_proto.Empty())) as cert_acl_data_stream:
cert_hci_event_asserts = EventAsserts(cert_hci_event_stream)
hci_event_asserts = EventAsserts(hci_event_stream)
acl_data_asserts = EventAsserts(acl_data_stream)
cert_acl_data_asserts = EventAsserts(cert_acl_data_stream)
address = hci_packets.Address()
def get_address_from_complete(packet):
packet_bytes = packet.payload
if b'\x0e\x0a\x01\x09\x10' in packet_bytes:
nonlocal address
addr_view = hci_packets.ReadBdAddrCompleteView(
hci_packets.CommandCompleteView(
hci_packets.EventPacketView(
bt_packets.PacketViewLittleEndian(
list(packet_bytes)))))
address = addr_view.GetBdAddr()
return True
return False
# CERT Enables scans and gets its address
self.send_hal_hci_command(hci_packets.ReadBdAddrBuilder())
cert_hci_event_asserts.assert_event_occurs(
get_address_from_complete)
self.send_hal_hci_command(
hci_packets.WriteScanEnableBuilder(
hci_packets.ScanEnable.INQUIRY_AND_PAGE_SCAN))
# DUT Connects
self.enqueue_hci_command(
hci_packets.CreateConnectionBuilder(
address,
0xcc18, # Packet Type
hci_packets.PageScanRepetitionMode.R0,
0,
hci_packets.ClockOffsetValid.INVALID,
hci_packets.CreateConnectionRoleSwitch.ALLOW_ROLE_SWITCH),
False)
# Cert Accepts
connection_request = None
def get_connect_request(packet):
if b'\x04\x0a' in packet.payload:
nonlocal connection_request
connection_request = hci_packets.ConnectionRequestView(
hci_packets.EventPacketView(
bt_packets.PacketViewLittleEndian(
list(packet.payload))))
return True
return False
# Cert Accepts
cert_hci_event_asserts.assert_event_occurs(
get_connect_request, timeout=timedelta(seconds=20))
self.send_hal_hci_command(
hci_packets.AcceptConnectionRequestBuilder(
connection_request.GetBdAddr(),
hci_packets.AcceptConnectionRequestRole.REMAIN_SLAVE))
conn_handle = 0xfff
def get_handle(packet_bytes):
if b'\x03\x0b\x00' in packet_bytes:
nonlocal conn_handle
cc_view = hci_packets.ConnectionCompleteView(
hci_packets.EventPacketView(
bt_packets.PacketViewLittleEndian(
list(packet_bytes))))
conn_handle = cc_view.GetConnectionHandle()
return True
return False
def event_handle(packet):
packet_bytes = packet.event
return get_handle(packet_bytes)
def payload_handle(packet):
packet_bytes = packet.payload
return get_handle(packet_bytes)
cert_hci_event_asserts.assert_event_occurs(payload_handle)
cert_handle = conn_handle
conn_handle = 0xfff
hci_event_asserts.assert_event_occurs(event_handle)
dut_handle = conn_handle
if dut_handle == 0xfff:
logging.warning("Failed to get the DUT handle")
return False
if cert_handle == 0xfff:
logging.warning("Failed to get the CERT handle")
return False
# Send ACL Data
self.enqueue_acl_data(
dut_handle, hci_packets.PacketBoundaryFlag.
FIRST_NON_AUTOMATICALLY_FLUSHABLE,
hci_packets.BroadcastFlag.POINT_TO_POINT,
bytes(b'Just SomeAclData'))
self.send_hal_acl_data(
cert_handle, hci_packets.PacketBoundaryFlag.
FIRST_NON_AUTOMATICALLY_FLUSHABLE,
hci_packets.BroadcastFlag.POINT_TO_POINT,
bytes(b'Just SomeMoreAclData'))
cert_acl_data_asserts.assert_event_occurs(
lambda packet: b'SomeAclData' in packet.payload)
acl_data_asserts.assert_event_occurs(
lambda packet: b'SomeMoreAclData' in packet.data)
def test_connection_cert_connects(self):
self.register_for_event(hci_packets.EventCode.CONNECTION_COMPLETE)
self.register_for_event(
hci_packets.EventCode.CONNECTION_PACKET_TYPE_CHANGED)
self.register_for_event(hci_packets.EventCode.CONNECTION_REQUEST)
self.send_hal_hci_command(hci_packets.WritePageTimeoutBuilder(0x4000))
with EventCallbackStream(self.device_under_test.hci.FetchEvents(empty_proto.Empty())) as hci_event_stream, \
EventCallbackStream(self.device_under_test.hci.FetchAclPackets(empty_proto.Empty())) as acl_data_stream, \
EventCallbackStream(self.cert_device.hal.FetchHciEvent(empty_proto.Empty())) as cert_hci_event_stream, \
EventCallbackStream(self.cert_device.hal.FetchHciAcl(empty_proto.Empty())) as cert_acl_data_stream:
hci_event_asserts = EventAsserts(hci_event_stream)
cert_hci_event_asserts = EventAsserts(cert_hci_event_stream)
acl_data_asserts = EventAsserts(acl_data_stream)
cert_acl_data_asserts = EventAsserts(cert_acl_data_stream)
address = hci_packets.Address()
def get_address_from_complete(packet):
packet_bytes = packet.event
if b'\x0e\x0a\x01\x09\x10' in packet_bytes:
nonlocal address
addr_view = hci_packets.ReadBdAddrCompleteView(
hci_packets.CommandCompleteView(
hci_packets.EventPacketView(
bt_packets.PacketViewLittleEndian(
list(packet_bytes)))))
address = addr_view.GetBdAddr()
return True
return False
# DUT Enables scans and gets its address
self.enqueue_hci_command(
hci_packets.WriteScanEnableBuilder(
hci_packets.ScanEnable.INQUIRY_AND_PAGE_SCAN), True)
self.enqueue_hci_command(hci_packets.ReadBdAddrBuilder(), True)
hci_event_asserts.assert_event_occurs(get_address_from_complete)
# Cert Connects
self.send_hal_hci_command(
hci_packets.CreateConnectionBuilder(
address,
0xcc18, # Packet Type
hci_packets.PageScanRepetitionMode.R0,
0,
hci_packets.ClockOffsetValid.INVALID,
hci_packets.CreateConnectionRoleSwitch.ALLOW_ROLE_SWITCH))
# DUT Accepts
connection_request = None
def get_connect_request(packet):
if b'\x04\x0a' in packet.event:
nonlocal connection_request
connection_request = hci_packets.ConnectionRequestView(
hci_packets.EventPacketView(
bt_packets.PacketViewLittleEndian(
list(packet.event))))
return True
return False
hci_event_asserts.assert_event_occurs(
get_connect_request, timeout=timedelta(seconds=20))
self.enqueue_hci_command(
hci_packets.AcceptConnectionRequestBuilder(
connection_request.GetBdAddr(),
hci_packets.AcceptConnectionRequestRole.REMAIN_SLAVE),
False)
conn_handle = 0xfff
def get_handle(packet_bytes):
if b'\x03\x0b\x00' in packet_bytes:
nonlocal conn_handle
cc_view = hci_packets.ConnectionCompleteView(
hci_packets.EventPacketView(
bt_packets.PacketViewLittleEndian(
list(packet_bytes))))
conn_handle = cc_view.GetConnectionHandle()
return True
return False
def event_handle(packet):
packet_bytes = packet.event
return get_handle(packet_bytes)
def payload_handle(packet):
packet_bytes = packet.payload
return get_handle(packet_bytes)
cert_hci_event_asserts.assert_event_occurs(payload_handle)
cert_handle = conn_handle
conn_handle = 0xfff
hci_event_asserts.assert_event_occurs(event_handle)
dut_handle = conn_handle
if dut_handle == 0xfff:
logging.warning("Failed to get the DUT handle")
return False
if cert_handle == 0xfff:
logging.warning("Failed to get the CERT handle")
return False
# Send ACL Data
self.enqueue_acl_data(
dut_handle, hci_packets.PacketBoundaryFlag.
FIRST_NON_AUTOMATICALLY_FLUSHABLE,
hci_packets.BroadcastFlag.POINT_TO_POINT,
bytes(b'This is just SomeAclData'))
self.send_hal_acl_data(
cert_handle, hci_packets.PacketBoundaryFlag.
FIRST_NON_AUTOMATICALLY_FLUSHABLE,
hci_packets.BroadcastFlag.POINT_TO_POINT,
bytes(b'This is just SomeMoreAclData'))
cert_acl_data_asserts.assert_event_occurs(
lambda packet: b'SomeAclData' in packet.payload)
acl_data_asserts.assert_event_occurs(
lambda packet: b'SomeMoreAclData' in packet.data)