In Falcon mode, the SPL loads a FIT image containing a Linux kernel instead of U-boot proper, and it may need to fall back to loading U-boot if Linux is unavailable.
Add support for images containing a Linux kernel, optionally on a different UFS LUN and/or offset vs. U-boot proper to enable fallback. Signed-off-by: Alexey Charkov <[email protected]> --- common/spl/Kconfig | 18 +++++++++++++++++ common/spl/spl_ufs.c | 57 ++++++++++++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 73 insertions(+), 2 deletions(-) diff --git a/common/spl/Kconfig b/common/spl/Kconfig index cc819344cad2..2ce48eb981d2 100644 --- a/common/spl/Kconfig +++ b/common/spl/Kconfig @@ -1635,6 +1635,16 @@ config SPL_UFS_RAW_U_BOOT_DEVNUM SCSI device number, which might differ from the UFS LUN if you have multiple SCSI devices attached and recognized by the SPL. +config SPL_UFS_RAW_OS_DEVNUM + int "Falcon mode: SCSI device number of the UFS device to load OS from" + depends on SPL_UFS_SUPPORT && SPL_OS_BOOT + default SPL_UFS_RAW_U_BOOT_DEVNUM + help + UFS devices are usually configured with multiple LUNs, which present + themselves as sequentially numbered SCSI devices. This option controls + which of them SPL will attempt to load a Falcon OS payload from. + It allows putting Linux FITs on a different UFS LU than U-Boot. + config SPL_UFS_RAW_U_BOOT_SECTOR hex "Address on the UFS to load U-Boot from" depends on SPL_UFS_SUPPORT @@ -1643,6 +1653,14 @@ config SPL_UFS_RAW_U_BOOT_SECTOR Address on the block device to load U-Boot from. Units: UFS sectors (1 sector = 4096 bytes). +config SPL_UFS_RAW_OS_SECTOR + hex "Falcon mode: address on the UFS to load OS image from" + depends on SPL_UFS_SUPPORT && SPL_OS_BOOT + default SPL_UFS_RAW_U_BOOT_SECTOR + help + Address on the block device to load a Falcon OS FIT/image from. + Units: UFS sectors (1 sector = 4096 bytes). + config SPL_WATCHDOG bool "Support watchdog drivers" imply SPL_WDT if !HW_WATCHDOG diff --git a/common/spl/spl_ufs.c b/common/spl/spl_ufs.c index cef1843f40f3..5c7ec869ab0c 100644 --- a/common/spl/spl_ufs.c +++ b/common/spl/spl_ufs.c @@ -20,13 +20,37 @@ static ulong spl_ufs_load_read(struct spl_load_info *load, ulong off, ulong size return blk_dread(bd, sector, count, buf) << bd->log2blksz; } +static int spl_ufs_load_image_raw_os(struct spl_image_info *spl_image, + struct spl_boot_device *bootdev, + struct spl_load_info *load, + struct blk_desc *bd) +{ + ulong sector = config_opt_enabled(CONFIG_SPL_OS_BOOT, + CONFIG_SPL_UFS_RAW_OS_SECTOR, 0); + int err; + + err = spl_load(spl_image, bootdev, load, 0, + sector << bd->log2blksz); + if (err) + return err; + + if (spl_image->os != IH_OS_LINUX && spl_image->os != IH_OS_TEE && + spl_image->os != IH_OS_ARM_TRUSTED_FIRMWARE) { + puts("Expected OS image is not found\n"); + return -ENOENT; + } + + return 0; +} + static int spl_ufs_load_image(struct spl_image_info *spl_image, struct spl_boot_device *bootdev) { unsigned long sector = CONFIG_SPL_UFS_RAW_U_BOOT_SECTOR; int devnum = CONFIG_SPL_UFS_RAW_U_BOOT_DEVNUM; - struct spl_load_info load; - struct blk_desc *bd; + int os_devnum; + struct spl_load_info load, os_load; + struct blk_desc *bd, *os_bd; int err; /* try to recognize storage devices immediately */ @@ -36,6 +60,35 @@ static int spl_ufs_load_image(struct spl_image_info *spl_image, return -ENODEV; spl_load_init(&load, spl_ufs_load_read, bd, bd->blksz); + if (IS_ENABLED(CONFIG_SPL_OS_BOOT) && !spl_start_uboot()) { + os_devnum = config_opt_enabled(CONFIG_SPL_OS_BOOT, + CONFIG_SPL_UFS_RAW_OS_DEVNUM, + CONFIG_SPL_UFS_RAW_U_BOOT_DEVNUM); + if (os_devnum == devnum) { + os_bd = bd; + os_load = load; + } else { + os_bd = blk_get_devnum_by_uclass_id(UCLASS_SCSI, os_devnum); + if (!os_bd) { + puts("spl_ufs_load_image: UFS OS device not found\n"); + if (IS_ENABLED(CONFIG_SPL_OS_BOOT_SECURE)) + return -ENODEV; + goto load_uboot; + } + spl_load_init(&os_load, spl_ufs_load_read, os_bd, os_bd->blksz); + } + + err = spl_ufs_load_image_raw_os(spl_image, bootdev, &os_load, os_bd); + if (!err) + return 0; + + puts("spl_ufs_load_image: Failed to load falcon payload\n"); + log_debug("(error=%d)\n", err); + if (IS_ENABLED(CONFIG_SPL_OS_BOOT_SECURE)) + return err; + } + +load_uboot: err = spl_load(spl_image, bootdev, &load, 0, sector << bd->log2blksz); if (err) { puts("spl_ufs_load_image: ufs block read error\n"); -- 2.53.0

