blob: 5270401db0347270a23b7b33a27898b1d0d04ae9 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Stefan Roese33d34642012-08-27 12:50:59 +02002/*
3 * Copyright (C) 2012 Stefan Roese <sr@denx.de>
Stefan Roese33d34642012-08-27 12:50:59 +02004 */
5
6#include <common.h>
Simon Glass4d72caa2020-05-10 11:40:01 -06007#include <image.h>
Simon Glassf7ae49f2020-05-10 11:40:05 -06008#include <log.h>
Stefan Roese33d34642012-08-27 12:50:59 +02009#include <spl.h>
10
York Sun6ecd8202018-06-26 10:10:03 -070011static ulong spl_nor_load_read(struct spl_load_info *load, ulong sector,
12 ulong count, void *buf)
13{
14 debug("%s: sector %lx, count %lx, buf %p\n",
15 __func__, sector, count, buf);
16 memcpy(buf, (void *)sector, count);
17
18 return count;
19}
York Sun6ecd8202018-06-26 10:10:03 -070020
Peng Fan07d900a2019-09-23 10:18:42 +080021unsigned long __weak spl_nor_get_uboot_base(void)
22{
23 return CONFIG_SYS_UBOOT_BASE;
24}
25
Simon Glass2a2ee2a2016-09-24 18:20:13 -060026static int spl_nor_load_image(struct spl_image_info *spl_image,
27 struct spl_boot_device *bootdev)
Stefan Roese33d34642012-08-27 12:50:59 +020028{
York Sun6ecd8202018-06-26 10:10:03 -070029 __maybe_unused const struct image_header *header;
30 __maybe_unused struct spl_load_info load;
31
Stefan Roese33d34642012-08-27 12:50:59 +020032 /*
33 * Loading of the payload to SDRAM is done with skipping of
34 * the mkimage header in this SPL NOR driver
35 */
Simon Glass2a2ee2a2016-09-24 18:20:13 -060036 spl_image->flags |= SPL_COPY_PAYLOAD_ONLY;
Stefan Roese33d34642012-08-27 12:50:59 +020037
Masahiro Yamada9f9d8702015-01-08 19:23:35 +090038#ifdef CONFIG_SPL_OS_BOOT
39 if (!spl_start_uboot()) {
Stefan Roese33d34642012-08-27 12:50:59 +020040 /*
41 * Load Linux from its location in NOR flash to its defined
42 * location in SDRAM
43 */
Masahiro Yamada9f9d8702015-01-08 19:23:35 +090044 header = (const struct image_header *)CONFIG_SYS_OS_BASE;
York Sun6ecd8202018-06-26 10:10:03 -070045#ifdef CONFIG_SPL_LOAD_FIT
46 if (image_get_magic(header) == FDT_MAGIC) {
Stefan Roesef1b0f152020-04-21 09:28:44 +020047 int ret;
48
York Sun6ecd8202018-06-26 10:10:03 -070049 debug("Found FIT\n");
50 load.bl_len = 1;
51 load.read = spl_nor_load_read;
Stefan Roese33d34642012-08-27 12:50:59 +020052
York Sun6ecd8202018-06-26 10:10:03 -070053 ret = spl_load_simple_fit(spl_image, &load,
54 CONFIG_SYS_OS_BASE,
55 (void *)header);
56
Lukasz Majewskia4a16c92019-10-15 10:28:45 +020057#if defined CONFIG_SYS_SPL_ARGS_ADDR && defined CONFIG_CMD_SPL_NOR_OFS
58 memcpy((void *)CONFIG_SYS_SPL_ARGS_ADDR,
59 (void *)CONFIG_CMD_SPL_NOR_OFS,
60 CONFIG_CMD_SPL_WRITE_SIZE);
61#endif
York Sun6ecd8202018-06-26 10:10:03 -070062 return ret;
63 }
64#endif
Masahiro Yamada9f9d8702015-01-08 19:23:35 +090065 if (image_get_os(header) == IH_OS_LINUX) {
66 /* happy - was a Linux */
Stefan Roesef1b0f152020-04-21 09:28:44 +020067 int ret;
Stefan Roese33d34642012-08-27 12:50:59 +020068
Simon Glass2a2ee2a2016-09-24 18:20:13 -060069 ret = spl_parse_image_header(spl_image, header);
Marek Vasut7e0f2262016-04-29 00:44:54 +020070 if (ret)
71 return ret;
Masahiro Yamada9f9d8702015-01-08 19:23:35 +090072
Simon Glass2a2ee2a2016-09-24 18:20:13 -060073 memcpy((void *)spl_image->load_addr,
Masahiro Yamada9f9d8702015-01-08 19:23:35 +090074 (void *)(CONFIG_SYS_OS_BASE +
75 sizeof(struct image_header)),
Simon Glass2a2ee2a2016-09-24 18:20:13 -060076 spl_image->size);
York Sun14acea02018-06-26 10:10:04 -070077#ifdef CONFIG_SYS_FDT_BASE
Vikas Manocha5bf52502017-04-07 15:38:13 -070078 spl_image->arg = (void *)CONFIG_SYS_FDT_BASE;
York Sun14acea02018-06-26 10:10:04 -070079#endif
Masahiro Yamada9f9d8702015-01-08 19:23:35 +090080
Nikita Kiryanov36afd452015-11-08 17:11:49 +020081 return 0;
Masahiro Yamada9f9d8702015-01-08 19:23:35 +090082 } else {
83 puts("The Expected Linux image was not found.\n"
84 "Please check your NOR configuration.\n"
85 "Trying to start u-boot now...\n");
86 }
Stefan Roese33d34642012-08-27 12:50:59 +020087 }
Masahiro Yamada9f9d8702015-01-08 19:23:35 +090088#endif
89
90 /*
91 * Load real U-Boot from its location in NOR flash to its
92 * defined location in SDRAM
93 */
York Sun6ecd8202018-06-26 10:10:03 -070094#ifdef CONFIG_SPL_LOAD_FIT
Peng Fan07d900a2019-09-23 10:18:42 +080095 header = (const struct image_header *)spl_nor_get_uboot_base();
York Sun6ecd8202018-06-26 10:10:03 -070096 if (image_get_magic(header) == FDT_MAGIC) {
97 debug("Found FIT format U-Boot\n");
98 load.bl_len = 1;
99 load.read = spl_nor_load_read;
Stefan Roesef1b0f152020-04-21 09:28:44 +0200100 return spl_load_simple_fit(spl_image, &load,
101 spl_nor_get_uboot_base(),
102 (void *)header);
York Sun6ecd8202018-06-26 10:10:03 -0700103 }
104#endif
Peng Fan3bc4f1d2019-09-23 10:18:48 +0800105 if (IS_ENABLED(CONFIG_SPL_LOAD_IMX_CONTAINER)) {
106 load.bl_len = 1;
107 load.read = spl_nor_load_read;
108 return spl_load_imx_container(spl_image, &load,
109 spl_nor_get_uboot_base());
110 }
111
Stefan Roese2fc91ed2020-04-21 09:28:43 +0200112 /* Legacy image handling */
113 if (IS_ENABLED(CONFIG_SPL_LEGACY_IMAGE_SUPPORT)) {
114 load.bl_len = 1;
115 load.read = spl_nor_load_read;
116 return spl_load_legacy_img(spl_image, &load,
117 spl_nor_get_uboot_base());
118 }
Nikita Kiryanov36afd452015-11-08 17:11:49 +0200119
120 return 0;
Stefan Roese33d34642012-08-27 12:50:59 +0200121}
Simon Glassebc4ef62016-11-30 15:30:50 -0700122SPL_LOAD_IMAGE_METHOD("NOR", 0, BOOT_DEVICE_NOR, spl_nor_load_image);