blob: 4729ccf07feb903f4e235d9df51b9d29058f5c43 [file] [log] [blame]
# SPDX-License-Identifier: GPL-2.0
# (C) Copyright 2023, Advanced Micro Devices, Inc.
import pytest
import u_boot_utils
import test_net
import re
"""
Note: This test relies on boardenv_* containing configuration values to define
which the network environment available for testing. Without this, this test
will be automatically skipped.
For example:
# Details regarding a boot image file that may be read from a TFTP server. This
# variable may be omitted or set to None if TFTP boot testing is not possible
# or desired.
env__net_tftp_bootable_file = {
'fn': 'image.ub',
'addr': 0x10000000,
'size': 5058624,
'crc32': 'c2244b26',
'pattern': 'Linux',
'config': 'config@2',
'timeout': 50000,
'check_type': 'boot_error',
'check_pattern': 'ERROR',
}
# False or omitted if a TFTP boot test should be tested.
# If TFTP boot testing is not possible or desired, set this variable to True.
# For example: If FIT image is not proper to boot
env__tftp_boot_test_skip = False
# Here is the example of FIT image configurations:
configurations {
default = "config@1";
config@1 {
description = "Boot Linux kernel with config@1";
kernel = "kernel@0";
fdt = "fdt@0";
ramdisk = "ramdisk@0";
hash@1 {
algo = "sha1";
};
};
config@2 {
description = "Boot Linux kernel with config@2";
kernel = "kernel@1";
fdt = "fdt@1";
ramdisk = "ramdisk@1";
hash@1 {
algo = "sha1";
};
};
};
# Details regarding a file that may be read from a TFTP server. This variable
# may be omitted or set to None if PXE testing is not possible or desired.
env__net_pxe_bootable_file = {
'fn': 'default',
'addr': 0x10000000,
'size': 74,
'timeout': 50000,
'pattern': 'Linux',
'valid_label': '1',
'invalid_label': '2',
'exp_str_invalid': 'Skipping install for failure retrieving',
'local_label': '3',
'exp_str_local': 'missing environment variable: localcmd',
'empty_label': '4',
'exp_str_empty': 'No kernel given, skipping boot',
'check_type': 'boot_error',
'check_pattern': 'ERROR',
}
# False or omitted if a PXE boot test should be tested.
# If PXE boot testing is not possible or desired, set this variable to True.
# For example: If pxe configuration file is not proper to boot
env__pxe_boot_test_skip = False
# Here is the example of pxe configuration file ordered based on the execution
# flow:
1) /tftpboot/pxelinux.cfg/default-arm-zynqmp
menu include pxelinux.cfg/default-arm
timeout 50
default Linux
2) /tftpboot/pxelinux.cfg/default-arm
menu title Linux boot selections
menu include pxelinux.cfg/default
label install
menu label Invalid boot
kernel kernels/install.bin
append console=ttyAMA0,38400 debug earlyprintk
initrd initrds/uzInitrdDebInstall
label local
menu label Local boot
append root=/dev/sdb1
localboot 1
label boot
menu label Empty boot
3) /tftpboot/pxelinux.cfg/default
label Linux
menu label Boot kernel
kernel Image
fdt system.dtb
initrd rootfs.cpio.gz.u-boot
"""
def setup_networking(u_boot_console):
test_net.test_net_dhcp(u_boot_console)
if not test_net.net_set_up:
test_net.test_net_setup_static(u_boot_console)
def setup_tftpboot_boot(u_boot_console):
f = u_boot_console.config.env.get('env__net_tftp_bootable_file', None)
if not f:
pytest.skip('No TFTP bootable file to read')
setup_networking(u_boot_console)
addr = f.get('addr', None)
if not addr:
addr = u_boot_utils.find_ram_base(u_boot_console)
fn = f['fn']
timeout = f.get('timeout', 50000)
with u_boot_console.temporary_timeout(timeout):
output = u_boot_console.run_command('tftpboot %x %s' % (addr, fn))
expected_text = 'Bytes transferred = '
sz = f.get('size', None)
if sz:
expected_text += '%d' % sz
assert expected_text in output
expected_crc = f.get('crc32', None)
output = u_boot_console.run_command('crc32 %x $filesize' % addr)
if expected_crc:
assert expected_crc in output
pattern = f.get('pattern')
chk_type = f.get('check_type', 'boot_error')
chk_pattern = re.compile(f.get('check_pattern', 'ERROR'))
config = f.get('config', None)
return addr, timeout, pattern, chk_type, chk_pattern, config
@pytest.mark.buildconfigspec('cmd_net')
def test_net_tftpboot_boot(u_boot_console):
"""Boot the loaded image
A boot file (fit image) is downloaded from the TFTP server and booted using
bootm command with the default fit configuration, its boot log pattern are
validated.
The details of the file to download are provided by the boardenv_* file;
see the comment at the beginning of this file.
"""
if u_boot_console.config.env.get('env__tftp_boot_test_skip', True):
pytest.skip('TFTP boot test is not enabled!')
addr, timeout, pattern, chk_type, chk_pattern, imcfg = setup_tftpboot_boot(
u_boot_console
)
if imcfg:
bootcmd = 'bootm %x#%s' % (addr, imcfg)
else:
bootcmd = 'bootm %x' % addr
with u_boot_console.enable_check(
chk_type, chk_pattern
), u_boot_console.temporary_timeout(timeout):
try:
# wait_for_prompt=False makes the core code not wait for the U-Boot
# prompt code to be seen, since it won't be on a successful kernel
# boot
u_boot_console.run_command(bootcmd, wait_for_prompt=False)
# Wait for boot log pattern
u_boot_console.wait_for(pattern)
finally:
# This forces the console object to be shutdown, so any subsequent
# test will reset the board back into U-Boot. We want to force this
# no matter whether the kernel boot passed or failed.
u_boot_console.drain_console()
u_boot_console.cleanup_spawn()
def setup_pxe_boot(u_boot_console):
f = u_boot_console.config.env.get('env__net_pxe_bootable_file', None)
if not f:
pytest.skip('No PXE bootable file to read')
setup_networking(u_boot_console)
bootfile = u_boot_console.run_command('echo $bootfile')
if not bootfile:
bootfile = '<NULL>'
return f, bootfile
@pytest.mark.buildconfigspec('cmd_net')
@pytest.mark.buildconfigspec('cmd_pxe')
def test_net_pxe_boot(u_boot_console):
"""Test the pxe boot command.
A pxe configuration file is downloaded from the TFTP server and interpreted
to boot the images mentioned in pxe configuration file.
The details of the file to download are provided by the boardenv_* file;
see the comment at the beginning of this file.
"""
if u_boot_console.config.env.get('env__pxe_boot_test_skip', True):
pytest.skip('PXE boot test is not enabled!')
f, bootfile = setup_pxe_boot(u_boot_console)
addr = f.get('addr', None)
timeout = f.get('timeout', u_boot_console.p.timeout)
fn = f['fn']
if addr:
u_boot_console.run_command('setenv pxefile_addr_r %x' % addr)
with u_boot_console.temporary_timeout(timeout):
output = u_boot_console.run_command('pxe get')
expected_text = 'Bytes transferred = '
sz = f.get('size', None)
if sz:
expected_text += '%d' % sz
assert 'TIMEOUT' not in output
assert expected_text in output
assert f"Config file '{bootfile}' found" in output
pattern = f.get('pattern')
chk_type = f.get('check_type', 'boot_error')
chk_pattern = re.compile(f.get('check_pattern', 'ERROR'))
if not addr:
pxe_boot_cmd = 'pxe boot'
else:
pxe_boot_cmd = 'pxe boot %x' % addr
with u_boot_console.enable_check(
chk_type, chk_pattern
), u_boot_console.temporary_timeout(timeout):
try:
u_boot_console.run_command(pxe_boot_cmd, wait_for_prompt=False)
u_boot_console.wait_for(pattern)
finally:
u_boot_console.drain_console()
u_boot_console.cleanup_spawn()
@pytest.mark.buildconfigspec('cmd_net')
@pytest.mark.buildconfigspec('cmd_pxe')
def test_net_pxe_boot_config(u_boot_console):
"""Test the pxe boot command by selecting different combination of labels
A pxe configuration file is downloaded from the TFTP server and interpreted
to boot the images mentioned in pxe configuration file.
The details of the file to download are provided by the boardenv_* file;
see the comment at the beginning of this file.
"""
if u_boot_console.config.env.get('env__pxe_boot_test_skip', True):
pytest.skip('PXE boot test is not enabled!')
f, bootfile = setup_pxe_boot(u_boot_console)
addr = f.get('addr', None)
timeout = f.get('timeout', u_boot_console.p.timeout)
fn = f['fn']
local_label = f['local_label']
empty_label = f['empty_label']
exp_str_local = f['exp_str_local']
exp_str_empty = f['exp_str_empty']
if addr:
u_boot_console.run_command('setenv pxefile_addr_r %x' % addr)
with u_boot_console.temporary_timeout(timeout):
output = u_boot_console.run_command('pxe get')
expected_text = 'Bytes transferred = '
sz = f.get('size', None)
if sz:
expected_text += '%d' % sz
assert 'TIMEOUT' not in output
assert expected_text in output
assert f"Config file '{bootfile}' found" in output
pattern = f.get('pattern')
chk_type = f.get('check_type', 'boot_error')
chk_pattern = re.compile(f.get('check_pattern', 'ERROR'))
if not addr:
pxe_boot_cmd = 'pxe boot'
else:
pxe_boot_cmd = 'pxe boot %x' % addr
with u_boot_console.enable_check(
chk_type, chk_pattern
), u_boot_console.temporary_timeout(timeout):
try:
u_boot_console.run_command(pxe_boot_cmd, wait_for_prompt=False)
# pxe config is loaded where multiple labels are there and need to
# select particular label to boot and check for expected string
# In this case, local label is selected and it should look for
# localcmd env variable and if that variable is not defined it
# should not boot it and come out to u-boot prompt
u_boot_console.wait_for('Enter choice:')
u_boot_console.run_command(local_label, wait_for_prompt=False)
expected_str = u_boot_console.p.expect([exp_str_local])
assert (
expected_str == 0
), f'Expected string: {exp_str_local} did not match!'
# In this case, empty label is selected and it should look for
# kernel image path and if it is not set it should fail it and load
# default label to boot
u_boot_console.run_command(pxe_boot_cmd, wait_for_prompt=False)
u_boot_console.wait_for('Enter choice:')
u_boot_console.run_command(empty_label, wait_for_prompt=False)
expected_str = u_boot_console.p.expect([exp_str_empty])
assert (
expected_str == 0
), f'Expected string: {exp_str_empty} did not match!'
u_boot_console.wait_for(pattern)
finally:
u_boot_console.drain_console()
u_boot_console.cleanup_spawn()
@pytest.mark.buildconfigspec('cmd_net')
@pytest.mark.buildconfigspec('cmd_pxe')
def test_net_pxe_boot_config_invalid(u_boot_console):
"""Test the pxe boot command by selecting invalid label
A pxe configuration file is downloaded from the TFTP server and interpreted
to boot the images mentioned in pxe configuration file.
The details of the file to download are provided by the boardenv_* file;
see the comment at the beginning of this file.
"""
if u_boot_console.config.env.get('env__pxe_boot_test_skip', True):
pytest.skip('PXE boot test is not enabled!')
f, bootfile = setup_pxe_boot(u_boot_console)
addr = f.get('addr', None)
timeout = f.get('timeout', u_boot_console.p.timeout)
fn = f['fn']
invalid_label = f['invalid_label']
exp_str_invalid = f['exp_str_invalid']
if addr:
u_boot_console.run_command('setenv pxefile_addr_r %x' % addr)
with u_boot_console.temporary_timeout(timeout):
output = u_boot_console.run_command('pxe get')
expected_text = 'Bytes transferred = '
sz = f.get('size', None)
if sz:
expected_text += '%d' % sz
assert 'TIMEOUT' not in output
assert expected_text in output
assert f"Config file '{bootfile}' found" in output
pattern = f.get('pattern')
if not addr:
pxe_boot_cmd = 'pxe boot'
else:
pxe_boot_cmd = 'pxe boot %x' % addr
with u_boot_console.temporary_timeout(timeout):
try:
u_boot_console.run_command(pxe_boot_cmd, wait_for_prompt=False)
# pxe config is loaded where multiple labels are there and need to
# select particular label to boot and check for expected string
# In this case invalid label is selected, it should load invalid
# label and if it fails it should load the default label to boot
u_boot_console.wait_for('Enter choice:')
u_boot_console.run_command(invalid_label, wait_for_prompt=False)
expected_str = u_boot_console.p.expect([exp_str_invalid])
assert (
expected_str == 0
), f'Expected string: {exp_str_invalid} did not match!'
u_boot_console.wait_for(pattern)
finally:
u_boot_console.drain_console()
u_boot_console.cleanup_spawn()