2018-05-06 21:58:06 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
2012-08-27 10:50:59 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 Stefan Roese <sr@denx.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
|
|
|
#include <spl.h>
|
|
|
|
|
2018-06-26 17:10:03 +00:00
|
|
|
static ulong spl_nor_load_read(struct spl_load_info *load, ulong sector,
|
|
|
|
ulong count, void *buf)
|
|
|
|
{
|
|
|
|
debug("%s: sector %lx, count %lx, buf %p\n",
|
|
|
|
__func__, sector, count, buf);
|
|
|
|
memcpy(buf, (void *)sector, count);
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2019-09-23 02:18:42 +00:00
|
|
|
unsigned long __weak spl_nor_get_uboot_base(void)
|
|
|
|
{
|
|
|
|
return CONFIG_SYS_UBOOT_BASE;
|
|
|
|
}
|
|
|
|
|
2016-09-25 00:20:13 +00:00
|
|
|
static int spl_nor_load_image(struct spl_image_info *spl_image,
|
|
|
|
struct spl_boot_device *bootdev)
|
2012-08-27 10:50:59 +00:00
|
|
|
{
|
2016-04-28 22:44:54 +00:00
|
|
|
int ret;
|
2018-06-26 17:10:03 +00:00
|
|
|
__maybe_unused const struct image_header *header;
|
|
|
|
__maybe_unused struct spl_load_info load;
|
|
|
|
|
2012-08-27 10:50:59 +00:00
|
|
|
/*
|
|
|
|
* Loading of the payload to SDRAM is done with skipping of
|
|
|
|
* the mkimage header in this SPL NOR driver
|
|
|
|
*/
|
2016-09-25 00:20:13 +00:00
|
|
|
spl_image->flags |= SPL_COPY_PAYLOAD_ONLY;
|
2012-08-27 10:50:59 +00:00
|
|
|
|
2015-01-08 10:23:35 +00:00
|
|
|
#ifdef CONFIG_SPL_OS_BOOT
|
|
|
|
if (!spl_start_uboot()) {
|
2012-08-27 10:50:59 +00:00
|
|
|
/*
|
|
|
|
* Load Linux from its location in NOR flash to its defined
|
|
|
|
* location in SDRAM
|
|
|
|
*/
|
2015-01-08 10:23:35 +00:00
|
|
|
header = (const struct image_header *)CONFIG_SYS_OS_BASE;
|
2018-06-26 17:10:03 +00:00
|
|
|
#ifdef CONFIG_SPL_LOAD_FIT
|
|
|
|
if (image_get_magic(header) == FDT_MAGIC) {
|
|
|
|
debug("Found FIT\n");
|
|
|
|
load.bl_len = 1;
|
|
|
|
load.read = spl_nor_load_read;
|
2012-08-27 10:50:59 +00:00
|
|
|
|
2018-06-26 17:10:03 +00:00
|
|
|
ret = spl_load_simple_fit(spl_image, &load,
|
|
|
|
CONFIG_SYS_OS_BASE,
|
|
|
|
(void *)header);
|
|
|
|
|
2019-10-15 08:28:45 +00:00
|
|
|
#if defined CONFIG_SYS_SPL_ARGS_ADDR && defined CONFIG_CMD_SPL_NOR_OFS
|
|
|
|
memcpy((void *)CONFIG_SYS_SPL_ARGS_ADDR,
|
|
|
|
(void *)CONFIG_CMD_SPL_NOR_OFS,
|
|
|
|
CONFIG_CMD_SPL_WRITE_SIZE);
|
|
|
|
#endif
|
2018-06-26 17:10:03 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
2015-01-08 10:23:35 +00:00
|
|
|
if (image_get_os(header) == IH_OS_LINUX) {
|
|
|
|
/* happy - was a Linux */
|
2012-08-27 10:50:59 +00:00
|
|
|
|
2016-09-25 00:20:13 +00:00
|
|
|
ret = spl_parse_image_header(spl_image, header);
|
2016-04-28 22:44:54 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
2015-01-08 10:23:35 +00:00
|
|
|
|
2016-09-25 00:20:13 +00:00
|
|
|
memcpy((void *)spl_image->load_addr,
|
2015-01-08 10:23:35 +00:00
|
|
|
(void *)(CONFIG_SYS_OS_BASE +
|
|
|
|
sizeof(struct image_header)),
|
2016-09-25 00:20:13 +00:00
|
|
|
spl_image->size);
|
2018-06-26 17:10:04 +00:00
|
|
|
#ifdef CONFIG_SYS_FDT_BASE
|
2017-04-07 22:38:13 +00:00
|
|
|
spl_image->arg = (void *)CONFIG_SYS_FDT_BASE;
|
2018-06-26 17:10:04 +00:00
|
|
|
#endif
|
2015-01-08 10:23:35 +00:00
|
|
|
|
2015-11-08 15:11:49 +00:00
|
|
|
return 0;
|
2015-01-08 10:23:35 +00:00
|
|
|
} else {
|
|
|
|
puts("The Expected Linux image was not found.\n"
|
|
|
|
"Please check your NOR configuration.\n"
|
|
|
|
"Trying to start u-boot now...\n");
|
|
|
|
}
|
2012-08-27 10:50:59 +00:00
|
|
|
}
|
2015-01-08 10:23:35 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Load real U-Boot from its location in NOR flash to its
|
|
|
|
* defined location in SDRAM
|
|
|
|
*/
|
2018-06-26 17:10:03 +00:00
|
|
|
#ifdef CONFIG_SPL_LOAD_FIT
|
2019-09-23 02:18:42 +00:00
|
|
|
header = (const struct image_header *)spl_nor_get_uboot_base();
|
2018-06-26 17:10:03 +00:00
|
|
|
if (image_get_magic(header) == FDT_MAGIC) {
|
|
|
|
debug("Found FIT format U-Boot\n");
|
|
|
|
load.bl_len = 1;
|
|
|
|
load.read = spl_nor_load_read;
|
|
|
|
ret = spl_load_simple_fit(spl_image, &load,
|
2019-09-23 02:18:42 +00:00
|
|
|
spl_nor_get_uboot_base(),
|
2018-06-26 17:10:03 +00:00
|
|
|
(void *)header);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
2019-09-23 02:18:48 +00:00
|
|
|
if (IS_ENABLED(CONFIG_SPL_LOAD_IMX_CONTAINER)) {
|
|
|
|
load.bl_len = 1;
|
|
|
|
load.read = spl_nor_load_read;
|
|
|
|
return spl_load_imx_container(spl_image, &load,
|
|
|
|
spl_nor_get_uboot_base());
|
|
|
|
}
|
|
|
|
|
2016-09-25 00:20:13 +00:00
|
|
|
ret = spl_parse_image_header(spl_image,
|
2019-09-23 02:18:42 +00:00
|
|
|
(const struct image_header *)spl_nor_get_uboot_base());
|
2016-04-28 22:44:54 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
2015-01-08 10:23:35 +00:00
|
|
|
|
2016-09-25 00:20:13 +00:00
|
|
|
memcpy((void *)(unsigned long)spl_image->load_addr,
|
2019-09-23 02:18:42 +00:00
|
|
|
(void *)(spl_nor_get_uboot_base() + sizeof(struct image_header)),
|
2016-09-25 00:20:13 +00:00
|
|
|
spl_image->size);
|
2015-11-08 15:11:49 +00:00
|
|
|
|
|
|
|
return 0;
|
2012-08-27 10:50:59 +00:00
|
|
|
}
|
2016-11-30 22:30:50 +00:00
|
|
|
SPL_LOAD_IMAGE_METHOD("NOR", 0, BOOT_DEVICE_NOR, spl_nor_load_image);
|