enhance serial port transport

This commit is contained in:
Gilles Boccon-Gibod
2025-09-25 18:31:14 +02:00
parent 34e0f293c2
commit aa1d7933da
4 changed files with 74 additions and 8 deletions

View File

@@ -550,7 +550,7 @@ class Host(utils.EventEmitter):
logger.debug( logger.debug(
'HCI LE flow control: ' 'HCI LE flow control: '
f'le_acl_data_packet_length={le_acl_data_packet_length},' f'le_acl_data_packet_length={le_acl_data_packet_length},'
f'total_num_le_acl_data_packets={total_num_le_acl_data_packets}' f'total_num_le_acl_data_packets={total_num_le_acl_data_packets},'
f'iso_data_packet_length={iso_data_packet_length},' f'iso_data_packet_length={iso_data_packet_length},'
f'total_num_iso_data_packets={total_num_iso_data_packets}' f'total_num_iso_data_packets={total_num_iso_data_packets}'
) )

View File

@@ -131,7 +131,11 @@ def publish_grpc_port(grpc_port: int, instance_number: int) -> bool:
def cleanup(): def cleanup():
logger.debug("removing .ini file") logger.debug("removing .ini file")
ini_file.unlink() try:
ini_file.unlink()
except OSError as error:
# Don't log at exception level, since this may happen normally.
logger.debug(f'failed to remove .ini file ({error})')
atexit.register(cleanup) atexit.register(cleanup)
return True return True

View File

@@ -28,25 +28,56 @@ from bumble.transport.common import StreamPacketSink, StreamPacketSource, Transp
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
# -----------------------------------------------------------------------------
# Constants
# -----------------------------------------------------------------------------
DEFAULT_POST_OPEN_DELAY = 0.5 # in seconds
# -----------------------------------------------------------------------------
# Classes and Functions
# -----------------------------------------------------------------------------
# -----------------------------------------------------------------------------
class SerialPacketSource(StreamPacketSource):
def __init__(self) -> None:
super().__init__()
self._ready = asyncio.Event()
async def wait_until_ready(self) -> None:
await self._ready.wait()
def connection_made(self, transport: asyncio.BaseTransport) -> None:
logger.debug('connection made')
self._ready.set()
def connection_lost(self, exc: Exception | None) -> None:
logger.debug('connection lost')
self.on_transport_lost()
# ----------------------------------------------------------------------------- # -----------------------------------------------------------------------------
async def open_serial_transport(spec: str) -> Transport: async def open_serial_transport(spec: str) -> Transport:
''' '''
Open a serial port transport. Open a serial port transport.
The parameter string has this syntax: The parameter string has this syntax:
<device-path>[,<speed>][,rtscts][,dsrdtr] <device-path>[,<speed>][,rtscts][,dsrdtr][,delay]
When <speed> is omitted, the default value of 1000000 is used When <speed> is omitted, the default value of 1000000 is used
When "rtscts" is specified, RTS/CTS hardware flow control is enabled When "rtscts" is specified, RTS/CTS hardware flow control is enabled
When "dsrdtr" is specified, DSR/DTR hardware flow control is enabled When "dsrdtr" is specified, DSR/DTR hardware flow control is enabled
When "delay" is specified, a short delay is added after opening the port
Examples: Examples:
/dev/tty.usbmodem0006839912172 /dev/tty.usbmodem0006839912172
/dev/tty.usbmodem0006839912172,1000000 /dev/tty.usbmodem0006839912172,1000000
/dev/tty.usbmodem0006839912172,rtscts /dev/tty.usbmodem0006839912172,rtscts
/dev/tty.usbmodem0006839912172,rtscts,delay
''' '''
speed = 1000000 speed = 1000000
rtscts = False rtscts = False
dsrdtr = False dsrdtr = False
delay = 0.0
if ',' in spec: if ',' in spec:
parts = spec.split(',') parts = spec.split(',')
device = parts[0] device = parts[0]
@@ -55,13 +86,16 @@ async def open_serial_transport(spec: str) -> Transport:
rtscts = True rtscts = True
elif part == 'dsrdtr': elif part == 'dsrdtr':
dsrdtr = True dsrdtr = True
elif part == 'delay':
delay = DEFAULT_POST_OPEN_DELAY
elif part.isnumeric(): elif part.isnumeric():
speed = int(part) speed = int(part)
else: else:
device = spec device = spec
serial_transport, packet_source = await serial_asyncio.create_serial_connection( serial_transport, packet_source = await serial_asyncio.create_serial_connection(
asyncio.get_running_loop(), asyncio.get_running_loop(),
StreamPacketSource, SerialPacketSource,
device, device,
baudrate=speed, baudrate=speed,
rtscts=rtscts, rtscts=rtscts,
@@ -69,4 +103,23 @@ async def open_serial_transport(spec: str) -> Transport:
) )
packet_sink = StreamPacketSink(serial_transport) packet_sink = StreamPacketSink(serial_transport)
logger.debug('waiting for the port to be ready')
await packet_source.wait_until_ready()
logger.debug('port is ready')
# Try to assert DTR
assert serial_transport.serial is not None
try:
serial_transport.serial.dtr = True
logger.debug(
f"DSR={serial_transport.serial.dsr}, DTR={serial_transport.serial.dtr}"
)
except Exception as e:
logger.warning(f'could not assert DTR: {e}')
# Wait a bit after opening the port, if requested
if delay > 0.0:
logger.debug(f'waiting {delay} seconds after opening the port')
await asyncio.sleep(delay)
return Transport(packet_source, packet_sink) return Transport(packet_source, packet_sink)

View File

@@ -4,9 +4,18 @@ SERIAL TRANSPORT
The serial transport implements sending/receiving HCI packets over a UART (a.k.a serial port). The serial transport implements sending/receiving HCI packets over a UART (a.k.a serial port).
## Moniker ## Moniker
The moniker syntax for a serial transport is: `serial:<device-path>[,<speed>]` The moniker syntax for a serial transport is:
When `<speed>` is omitted, the default value of 1000000 is used `<device-path>[,<speed>][,rtscts][,dsrdtr][,delay]`
When `<speed>` is omitted, the default value of 1000000 is used.
When `rtscts` is specified, RTS/CTS hardware flow control is enabled.
When `dsrdtr` is specified, DSR/DTR hardware flow control is enabled.
When `delay` is specified, a short delay is added after opening the port.
!!! example !!! example
`serial:/dev/tty.usbmodem0006839912172,1000000` ```
Opens the serial port `/dev/tty.usbmodem0006839912172` at `1000000`bps /dev/tty.usbmodem0006839912172
/dev/tty.usbmodem0006839912172,1000000
/dev/tty.usbmodem0006839912172,rtscts
/dev/tty.usbmodem0006839912172,rtscts,delay
```