xref: /openbmc/u-boot/common/spl/spl_ubi.c (revision 04ce5427bd3914cab8be78513275a20ab878520a)
14549e789STom Rini // SPDX-License-Identifier: GPL-2.0+ OR BSD-3-Clause
2bf55cd4fSLadislav Michl /*
3bf55cd4fSLadislav Michl  * Copyright (C) 2016
4bf55cd4fSLadislav Michl  * Ladislav Michl <ladis@linux-mips.org>
5bf55cd4fSLadislav Michl  */
6bf55cd4fSLadislav Michl 
7bf55cd4fSLadislav Michl #include <common.h>
8bf55cd4fSLadislav Michl #include <config.h>
9bf55cd4fSLadislav Michl #include <nand.h>
10bf55cd4fSLadislav Michl #include <onenand_uboot.h>
11bf55cd4fSLadislav Michl #include <ubispl.h>
12bf55cd4fSLadislav Michl #include <spl.h>
13bf55cd4fSLadislav Michl 
spl_ubi_load_image(struct spl_image_info * spl_image,struct spl_boot_device * bootdev)142a2ee2acSSimon Glass int spl_ubi_load_image(struct spl_image_info *spl_image,
152a2ee2acSSimon Glass 		       struct spl_boot_device *bootdev)
16bf55cd4fSLadislav Michl {
17bf55cd4fSLadislav Michl 	struct image_header *header;
18bf55cd4fSLadislav Michl 	struct ubispl_info info;
19bf55cd4fSLadislav Michl 	struct ubispl_load volumes[2];
20bf55cd4fSLadislav Michl 	int ret = 1;
21bf55cd4fSLadislav Michl 
22ecdfd69aSSimon Glass 	switch (bootdev->boot_device) {
23bf55cd4fSLadislav Michl #ifdef CONFIG_SPL_NAND_SUPPORT
24bf55cd4fSLadislav Michl 	case BOOT_DEVICE_NAND:
25bf55cd4fSLadislav Michl 		nand_init();
26bf55cd4fSLadislav Michl 		info.read = nand_spl_read_block;
27bf55cd4fSLadislav Michl 		info.peb_size = CONFIG_SYS_NAND_BLOCK_SIZE;
28bf55cd4fSLadislav Michl 		break;
29bf55cd4fSLadislav Michl #endif
30bf55cd4fSLadislav Michl #ifdef CONFIG_SPL_ONENAND_SUPPORT
31bf55cd4fSLadislav Michl 	case BOOT_DEVICE_ONENAND:
32bf55cd4fSLadislav Michl 		info.read = onenand_spl_read_block;
33bf55cd4fSLadislav Michl 		info.peb_size = CONFIG_SYS_ONENAND_BLOCK_SIZE;
34bf55cd4fSLadislav Michl 		break;
35bf55cd4fSLadislav Michl #endif
36bf55cd4fSLadislav Michl 	default:
37bf55cd4fSLadislav Michl 		goto out;
38bf55cd4fSLadislav Michl 	}
39bf55cd4fSLadislav Michl 	info.ubi = (struct ubi_scan_info *)CONFIG_SPL_UBI_INFO_ADDR;
40ffc282e1SLadislav Michl 	info.fastmap = IS_ENABLED(CONFIG_MTD_UBI_FASTMAP);
41bf55cd4fSLadislav Michl 
42bf55cd4fSLadislav Michl 	info.peb_offset = CONFIG_SPL_UBI_PEB_OFFSET;
43bf55cd4fSLadislav Michl 	info.vid_offset = CONFIG_SPL_UBI_VID_OFFSET;
44bf55cd4fSLadislav Michl 	info.leb_start = CONFIG_SPL_UBI_LEB_START;
45bf55cd4fSLadislav Michl 	info.peb_count = CONFIG_SPL_UBI_MAX_PEBS - info.peb_offset;
46bf55cd4fSLadislav Michl 
47bf55cd4fSLadislav Michl #ifdef CONFIG_SPL_OS_BOOT
48bf55cd4fSLadislav Michl 	if (!spl_start_uboot()) {
49bf55cd4fSLadislav Michl 		volumes[0].vol_id = CONFIG_SPL_UBI_LOAD_KERNEL_ID;
50bf55cd4fSLadislav Michl 		volumes[0].load_addr = (void *)CONFIG_SYS_LOAD_ADDR;
51bf55cd4fSLadislav Michl 		volumes[1].vol_id = CONFIG_SPL_UBI_LOAD_ARGS_ID;
52bf55cd4fSLadislav Michl 		volumes[1].load_addr = (void *)CONFIG_SYS_SPL_ARGS_ADDR;
53bf55cd4fSLadislav Michl 
54bf55cd4fSLadislav Michl 		ret = ubispl_load_volumes(&info, volumes, 2);
55bf55cd4fSLadislav Michl 		if (!ret) {
56bf55cd4fSLadislav Michl 			header = (struct image_header *)volumes[0].load_addr;
572a2ee2acSSimon Glass 			spl_parse_image_header(spl_image, header);
58bf55cd4fSLadislav Michl 			puts("Linux loaded.\n");
59bf55cd4fSLadislav Michl 			goto out;
60bf55cd4fSLadislav Michl 		}
61bf55cd4fSLadislav Michl 		puts("Loading Linux failed, falling back to U-Boot.\n");
62bf55cd4fSLadislav Michl 	}
63bf55cd4fSLadislav Michl #endif
64*04ce5427SMarek Vasut 	header = spl_get_load_buffer(-sizeof(*header), sizeof(header));
65bf55cd4fSLadislav Michl 	volumes[0].vol_id = CONFIG_SPL_UBI_LOAD_MONITOR_ID;
66bf55cd4fSLadislav Michl 	volumes[0].load_addr = (void *)header;
67bf55cd4fSLadislav Michl 
68bf55cd4fSLadislav Michl 	ret = ubispl_load_volumes(&info, volumes, 1);
69bf55cd4fSLadislav Michl 	if (!ret)
702a2ee2acSSimon Glass 		spl_parse_image_header(spl_image, header);
71bf55cd4fSLadislav Michl out:
72bf55cd4fSLadislav Michl #ifdef CONFIG_SPL_NAND_SUPPORT
73ecdfd69aSSimon Glass 	if (bootdev->boot_device == BOOT_DEVICE_NAND)
74bf55cd4fSLadislav Michl 		nand_deselect();
75bf55cd4fSLadislav Michl #endif
76bf55cd4fSLadislav Michl 	return ret;
77bf55cd4fSLadislav Michl }
787d7dd821SSimon Glass /* Use priorty 0 so that Ubi will override NAND and ONENAND methods */
79ebc4ef61SSimon Glass SPL_LOAD_IMAGE_METHOD("NAND", 0, BOOT_DEVICE_NAND, spl_ubi_load_image);
80ebc4ef61SSimon Glass SPL_LOAD_IMAGE_METHOD("OneNAND", 0, BOOT_DEVICE_ONENAND, spl_ubi_load_image);
81