]>
Commit | Line | Data |
---|---|---|
83d290c5 | 1 | // SPDX-License-Identifier: GPL-2.0+ |
32b11273 CR |
2 | /* |
3 | * Copyright (C) 2011 OMICRON electronics GmbH | |
4 | * | |
a430fa06 | 5 | * based on drivers/mtd/nand/raw/nand_spl_load.c |
32b11273 CR |
6 | * |
7 | * Copyright (C) 2011 | |
8 | * Heiko Schocher, DENX Software Engineering, [email protected]. | |
32b11273 CR |
9 | */ |
10 | ||
11 | #include <common.h> | |
4d72caa5 | 12 | #include <image.h> |
f7ae49fc | 13 | #include <log.h> |
ff0960f9 | 14 | #include <spi.h> |
32b11273 | 15 | #include <spi_flash.h> |
36afd451 | 16 | #include <errno.h> |
a4cc1c48 | 17 | #include <spl.h> |
401d1c4f | 18 | #include <asm/global_data.h> |
32b11273 | 19 | |
2bac55bc PT |
20 | DECLARE_GLOBAL_DATA_PTR; |
21 | ||
fa1a73fa TR |
22 | #ifdef CONFIG_SPL_OS_BOOT |
23 | /* | |
24 | * Load the kernel, check for a valid header we can parse, and if found load | |
25 | * the kernel and then device tree. | |
26 | */ | |
2a2ee2ac SG |
27 | static int spi_load_image_os(struct spl_image_info *spl_image, |
28 | struct spi_flash *flash, | |
fa1a73fa TR |
29 | struct image_header *header) |
30 | { | |
7e0f2267 MV |
31 | int err; |
32 | ||
fa1a73fa | 33 | /* Read for a header, parse or error out. */ |
51c12319 | 34 | spi_flash_read(flash, CONFIG_SYS_SPI_KERNEL_OFFS, sizeof(*header), |
fa1a73fa TR |
35 | (void *)header); |
36 | ||
37 | if (image_get_magic(header) != IH_MAGIC) | |
38 | return -1; | |
39 | ||
2a2ee2ac | 40 | err = spl_parse_image_header(spl_image, header); |
7e0f2267 MV |
41 | if (err) |
42 | return err; | |
fa1a73fa TR |
43 | |
44 | spi_flash_read(flash, CONFIG_SYS_SPI_KERNEL_OFFS, | |
2a2ee2ac | 45 | spl_image->size, (void *)spl_image->load_addr); |
fa1a73fa TR |
46 | |
47 | /* Read device tree. */ | |
48 | spi_flash_read(flash, CONFIG_SYS_SPI_ARGS_OFFS, | |
49 | CONFIG_SYS_SPI_ARGS_SIZE, | |
50 | (void *)CONFIG_SYS_SPL_ARGS_ADDR); | |
51 | ||
52 | return 0; | |
53 | } | |
54 | #endif | |
55 | ||
00d55956 LV |
56 | static ulong spl_spi_fit_read(struct spl_load_info *load, ulong sector, |
57 | ulong count, void *buf) | |
58 | { | |
59 | struct spi_flash *flash = load->dev; | |
60 | ulong ret; | |
61 | ||
62 | ret = spi_flash_read(flash, sector, count, buf); | |
63 | if (!ret) | |
64 | return count; | |
65 | else | |
66 | return 0; | |
67 | } | |
ec649330 PF |
68 | |
69 | unsigned int __weak spl_spi_get_uboot_offs(struct spi_flash *flash) | |
70 | { | |
71 | return CONFIG_SYS_SPI_U_BOOT_OFFS; | |
72 | } | |
73 | ||
32b11273 CR |
74 | /* |
75 | * The main entry for SPI booting. It's necessary that SDRAM is already | |
76 | * configured and available since this code loads the main U-Boot image | |
77 | * from SPI into SDRAM and starts it from there. | |
78 | */ | |
2a2ee2ac SG |
79 | static int spl_spi_load_image(struct spl_image_info *spl_image, |
80 | struct spl_boot_device *bootdev) | |
32b11273 | 81 | { |
36afd451 | 82 | int err = 0; |
ec649330 | 83 | unsigned int payload_offs; |
32b11273 | 84 | struct spi_flash *flash; |
a4cc1c48 | 85 | struct image_header *header; |
32b11273 CR |
86 | |
87 | /* | |
88 | * Load U-Boot image from SPI flash into RAM | |
b0cc1b84 PD |
89 | * In DM mode: defaults speed and mode will be |
90 | * taken from DT when available | |
32b11273 CR |
91 | */ |
92 | ||
88e34e5f NK |
93 | flash = spi_flash_probe(CONFIG_SF_DEFAULT_BUS, |
94 | CONFIG_SF_DEFAULT_CS, | |
95 | CONFIG_SF_DEFAULT_SPEED, | |
96 | CONFIG_SF_DEFAULT_MODE); | |
32b11273 | 97 | if (!flash) { |
a4cc1c48 | 98 | puts("SPI probe failed.\n"); |
36afd451 | 99 | return -ENODEV; |
32b11273 CR |
100 | } |
101 | ||
ec649330 PF |
102 | payload_offs = spl_spi_get_uboot_offs(flash); |
103 | ||
51c12319 | 104 | header = spl_get_load_buffer(-sizeof(*header), sizeof(*header)); |
32b11273 | 105 | |
2bac55bc PT |
106 | #if CONFIG_IS_ENABLED(OF_CONTROL) && !CONFIG_IS_ENABLED(OF_PLATDATA) |
107 | payload_offs = fdtdec_get_config_int(gd->fdt_blob, | |
108 | "u-boot,spl-payload-offset", | |
109 | payload_offs); | |
110 | #endif | |
111 | ||
fa1a73fa | 112 | #ifdef CONFIG_SPL_OS_BOOT |
2a2ee2ac | 113 | if (spl_start_uboot() || spi_load_image_os(spl_image, flash, header)) |
fa1a73fa TR |
114 | #endif |
115 | { | |
116 | /* Load u-boot, mkimage header is 64 bytes. */ | |
51c12319 | 117 | err = spi_flash_read(flash, payload_offs, sizeof(*header), |
36afd451 | 118 | (void *)header); |
a7044900 SG |
119 | if (err) { |
120 | debug("%s: Failed to read from SPI flash (err=%d)\n", | |
121 | __func__, err); | |
36afd451 | 122 | return err; |
a7044900 | 123 | } |
36afd451 | 124 | |
26ad648f MV |
125 | if (IS_ENABLED(CONFIG_SPL_LOAD_FIT_FULL) && |
126 | image_get_magic(header) == FDT_MAGIC) { | |
127 | err = spi_flash_read(flash, payload_offs, | |
128 | roundup(fdt_totalsize(header), 4), | |
129 | (void *)CONFIG_SYS_LOAD_ADDR); | |
130 | if (err) | |
131 | return err; | |
132 | err = spl_parse_image_header(spl_image, | |
133 | (struct image_header *)CONFIG_SYS_LOAD_ADDR); | |
134 | } else if (IS_ENABLED(CONFIG_SPL_LOAD_FIT) && | |
135 | image_get_magic(header) == FDT_MAGIC) { | |
00d55956 LV |
136 | struct spl_load_info load; |
137 | ||
138 | debug("Found FIT\n"); | |
139 | load.dev = flash; | |
140 | load.priv = NULL; | |
141 | load.filename = NULL; | |
142 | load.bl_len = 1; | |
143 | load.read = spl_spi_fit_read; | |
f4d7d859 | 144 | err = spl_load_simple_fit(spl_image, &load, |
2bac55bc | 145 | payload_offs, |
00d55956 | 146 | header); |
d9bd2f4a PF |
147 | } else if (IS_ENABLED(CONFIG_SPL_LOAD_IMX_CONTAINER)) { |
148 | struct spl_load_info load; | |
149 | ||
150 | load.dev = flash; | |
151 | load.priv = NULL; | |
152 | load.filename = NULL; | |
153 | load.bl_len = 1; | |
154 | load.read = spl_spi_fit_read; | |
155 | ||
156 | err = spl_load_imx_container(spl_image, &load, | |
157 | payload_offs); | |
00d55956 | 158 | } else { |
2a2ee2ac | 159 | err = spl_parse_image_header(spl_image, header); |
00d55956 LV |
160 | if (err) |
161 | return err; | |
2bac55bc | 162 | err = spi_flash_read(flash, payload_offs, |
2a2ee2ac SG |
163 | spl_image->size, |
164 | (void *)spl_image->load_addr); | |
00d55956 | 165 | } |
fa1a73fa | 166 | } |
36afd451 NK |
167 | |
168 | return err; | |
32b11273 | 169 | } |
139db7af | 170 | /* Use priorty 1 so that boards can override this */ |
ebc4ef61 | 171 | SPL_LOAD_IMAGE_METHOD("SPI", 1, BOOT_DEVICE_SPI, spl_spi_load_image); |