twister: add flash-before option

Add option to flash board before attach serial.

Current implementation performs the following sequence:

1. Open serial port to listen to board log output
2. Flash device

In case of ESP32 where it uses the same serial port
for both operations, flashing needs to come first.

This PR adds a twister option named --flash-before
which enables the process above, allowing tests to be
performed properly.

Signed-off-by: Lucas Tamborrino <lucas.tamborrino@espressif.com>
This commit is contained in:
Lucas Tamborrino 2023-04-27 10:44:56 -03:00 committed by Anas Nashif
parent bcd1d19322
commit d65b6bc199
4 changed files with 48 additions and 6 deletions

View file

@ -178,6 +178,12 @@ Artificially long but functional example:
when flash operation also executes test case on the platform. when flash operation also executes test case on the platform.
""") """)
parser.add_argument("--flash-before", action="store_true", default=False,
help="""Flash device before attaching to serial port.
This is useful for devices that share the same port for programming
and serial console, where flash must come first.
""")
test_or_build.add_argument( test_or_build.add_argument(
"-b", "--build-only", action="store_true", default="--prep-artifacts-for-testing" in sys.argv, "-b", "--build-only", action="store_true", default="--prep-artifacts-for-testing" in sys.argv,
help="Only build the code, do not attempt to run the code on targets.") help="Only build the code, do not attempt to run the code on targets.")
@ -796,6 +802,14 @@ def parse_arguments(parser, args, options = None):
logger.error("--device-flash-with-test requires --device_testing") logger.error("--device-flash-with-test requires --device_testing")
sys.exit(1) sys.exit(1)
if options.flash_before and options.device_flash_with_test:
logger.error("--device-flash-with-test does not apply when --flash-before is used")
sys.exit(1)
if options.flash_before and options.device_serial_pty:
logger.error("--device-serial-pty cannot be used when --flash-before is set (for now)")
sys.exit(1)
if options.shuffle_tests and options.subset is None: if options.shuffle_tests and options.subset is None:
logger.error("--shuffle-tests requires --subset") logger.error("--shuffle-tests requires --subset")
sys.exit(1) sys.exit(1)

View file

@ -378,6 +378,10 @@ class DeviceHandler(Handler):
# from the test. # from the test.
harness.capture_coverage = True harness.capture_coverage = True
# Wait for serial connection
while not ser.isOpen():
time.sleep(0.1)
# Clear serial leftover. # Clear serial leftover.
ser.reset_input_buffer() ser.reset_input_buffer()
@ -647,9 +651,13 @@ class DeviceHandler(Handler):
if hardware.flash_with_test: if hardware.flash_with_test:
flash_timeout += self.get_test_timeout() flash_timeout += self.get_test_timeout()
serial_port = None
if hardware.flash_before is False:
serial_port = serial_device
try: try:
ser = self._create_serial_connection( ser = self._create_serial_connection(
serial_device, serial_port,
hardware.baud, hardware.baud,
flash_timeout, flash_timeout,
serial_pty, serial_pty,
@ -703,6 +711,15 @@ class DeviceHandler(Handler):
if post_flash_script: if post_flash_script:
self.run_custom_script(post_flash_script, 30) self.run_custom_script(post_flash_script, 30)
# Connect to device after flashing it
if hardware.flash_before:
try:
logger.debug(f"Attach serial device {serial_device} @ {hardware.baud} baud")
ser.port = serial_device
ser.open()
except serial.SerialException:
return
if not flash_error: if not flash_error:
# Always wait at most the test timeout here after flashing. # Always wait at most the test timeout here after flashing.
t.join(self.get_test_timeout()) t.join(self.get_test_timeout())

View file

@ -49,7 +49,8 @@ class DUT(object):
post_flash_script=None, post_flash_script=None,
runner=None, runner=None,
flash_timeout=60, flash_timeout=60,
flash_with_test=False): flash_with_test=False,
flash_before=False):
self.serial = serial self.serial = serial
self.baud = serial_baud or 115200 self.baud = serial_baud or 115200
@ -63,6 +64,7 @@ class DUT(object):
self.product = product self.product = product
self.runner = runner self.runner = runner
self.runner_params = runner_params self.runner_params = runner_params
self.flash_before = flash_before
self.fixtures = [] self.fixtures = []
self.post_flash_script = post_flash_script self.post_flash_script = post_flash_script
self.post_script = post_script self.post_script = post_script
@ -177,7 +179,8 @@ class HardwareMap:
False, False,
baud=self.options.device_serial_baud, baud=self.options.device_serial_baud,
flash_timeout=self.options.device_flash_timeout, flash_timeout=self.options.device_flash_timeout,
flash_with_test=self.options.device_flash_with_test flash_with_test=self.options.device_flash_with_test,
flash_before=self.options.flash_before,
) )
elif self.options.device_serial_pty: elif self.options.device_serial_pty:
@ -186,7 +189,8 @@ class HardwareMap:
self.options.pre_script, self.options.pre_script,
True, True,
flash_timeout=self.options.device_flash_timeout, flash_timeout=self.options.device_flash_timeout,
flash_with_test=self.options.device_flash_with_test flash_with_test=self.options.device_flash_with_test,
flash_before=False,
) )
# the fixtures given by twister command explicitly should be assigned to each DUT # the fixtures given by twister command explicitly should be assigned to each DUT
@ -207,9 +211,9 @@ class HardwareMap:
print(tabulate(table, headers=header, tablefmt="github")) print(tabulate(table, headers=header, tablefmt="github"))
def add_device(self, serial, platform, pre_script, is_pty, baud=None, flash_timeout=60, flash_with_test=False): def add_device(self, serial, platform, pre_script, is_pty, baud=None, flash_timeout=60, flash_with_test=False, flash_before=False):
device = DUT(platform=platform, connected=True, pre_script=pre_script, serial_baud=baud, device = DUT(platform=platform, connected=True, pre_script=pre_script, serial_baud=baud,
flash_timeout=flash_timeout, flash_with_test=flash_with_test flash_timeout=flash_timeout, flash_with_test=flash_with_test, flash_before=flash_before
) )
if is_pty: if is_pty:
device.serial_pty = serial device.serial_pty = serial
@ -229,6 +233,9 @@ class HardwareMap:
flash_with_test = dut.get('flash_with_test') flash_with_test = dut.get('flash_with_test')
if flash_with_test is None: if flash_with_test is None:
flash_with_test = self.options.device_flash_with_test flash_with_test = self.options.device_flash_with_test
flash_before = dut.get('flash_before')
if flash_before is None:
flash_before = self.options.flash_before and (not (flash_with_test or serial_pty))
platform = dut.get('platform') platform = dut.get('platform')
id = dut.get('id') id = dut.get('id')
runner = dut.get('runner') runner = dut.get('runner')
@ -251,6 +258,7 @@ class HardwareMap:
serial_baud=baud, serial_baud=baud,
connected=connected, connected=connected,
pre_script=pre_script, pre_script=pre_script,
flash_before=flash_before,
post_script=post_script, post_script=post_script,
post_flash_script=post_flash_script, post_flash_script=post_flash_script,
flash_timeout=flash_timeout, flash_timeout=flash_timeout,

View file

@ -61,3 +61,6 @@ sequence:
"flash_with_test": "flash_with_test":
type: bool type: bool
required: false required: false
"flash_before":
type: bool
required: false