xref: /OK3568_Linux_fs/u-boot/common/spl/spl_mtd_blk.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (c) 2019 Fuzhou Rockchip Electronics Co., Ltd
4  */
5 
6 #include <common.h>
7 #include <blk.h>
8 #include <boot_rkimg.h>
9 #include <dm.h>
10 #include <errno.h>
11 #include <image.h>
12 #include <malloc.h>
13 #include <mtd_blk.h>
14 #include <part.h>
15 #include <spl.h>
16 #include <spl_ab.h>
17 #include <spl_rkfw.h>
18 #include <asm/u-boot.h>
19 #include <dm/device-internal.h>
20 #include <linux/compiler.h>
21 #include <linux/mtd/mtd.h>
22 
spl_mtd_get_device_index(u32 boot_device)23 static int spl_mtd_get_device_index(u32 boot_device)
24 {
25 	switch (boot_device) {
26 	case BOOT_DEVICE_MTD_BLK_NAND:
27 		return 0;
28 	case BOOT_DEVICE_MTD_BLK_SPI_NAND:
29 		return 1;
30 	case BOOT_DEVICE_MTD_BLK_SPI_NOR:
31 		return 2;
32 	}
33 
34 #ifdef CONFIG_SPL_LIBCOMMON_SUPPORT
35 	printf("spl: unsupported mtd boot device.\n");
36 #endif
37 
38 	return -ENODEV;
39 }
40 
find_mtd_device(int dev_num)41 struct blk_desc *find_mtd_device(int dev_num)
42 {
43 	struct udevice *dev;
44 	struct blk_desc *desc;
45 	int ret;
46 
47 	ret = blk_find_device(IF_TYPE_MTD, dev_num, &dev);
48 
49 	if (ret) {
50 #if !defined(CONFIG_SPL_BUILD) || defined(CONFIG_SPL_LIBCOMMON_SUPPORT)
51 		printf("MTD Device %d not found\n", dev_num);
52 #endif
53 		return NULL;
54 	}
55 
56 	ret = device_probe(dev);
57 	if (ret) {
58 #if !defined(CONFIG_SPL_BUILD) || defined(CONFIG_SPL_LIBCOMMON_SUPPORT)
59 		printf("MTD Device %d not found\n", dev_num);
60 #endif
61 		return NULL;
62 	}
63 
64 	desc = dev_get_uclass_platdata(dev);
65 	if (!desc)
66 		return NULL;
67 
68 	return desc;
69 }
70 
mtd_spl_load_read(struct spl_load_info * load,ulong sector,ulong count,void * buf)71 static ulong mtd_spl_load_read(struct spl_load_info *load, ulong sector,
72 			       ulong count, void *buf)
73 {
74 	struct blk_desc *desc = load->dev;
75 
76 	return blk_dread(desc, sector, count, buf);
77 }
78 
79 #ifdef CONFIG_SPL_LOAD_RKFW
spl_mtd_load_rkfw(struct spl_image_info * spl_image,struct blk_desc * desc)80 int spl_mtd_load_rkfw(struct spl_image_info *spl_image, struct blk_desc *desc)
81 {
82 	struct spl_load_info load;
83 	int ret;
84 
85 	load.dev = desc;
86 	load.priv = NULL;
87 	load.filename = NULL;
88 	load.bl_len = desc->blksz;
89 	load.read = mtd_spl_load_read;
90 
91 	ret = spl_load_rkfw_image(spl_image, &load);
92 	if (ret) {
93 #ifdef CONFIG_SPL_LIBCOMMON_SUPPORT
94 		puts("spl_mtd_load_rkfw: mtd block read error\n");
95 #endif
96 		return -1;
97 	}
98 
99 	return 0;
100 }
101 #endif
102 
spl_mtd_load_image(struct spl_image_info * spl_image,struct spl_boot_device * bootdev)103 int spl_mtd_load_image(struct spl_image_info *spl_image,
104 		       struct spl_boot_device *bootdev)
105 {
106 	struct image_header *header;
107 	struct blk_desc *desc;
108 	int ret = -1;
109 	lbaint_t image_sector = CONFIG_MTD_BLK_U_BOOT_OFFS;
110 
111 	desc = find_mtd_device(spl_mtd_get_device_index(bootdev->boot_device));
112 	if (!desc)
113 		return -ENODEV;
114 
115 #ifdef CONFIG_SPL_LIBDISK_SUPPORT
116 	disk_partition_t info;
117 
118 	mtd_blk_map_partitions(desc);
119 	ret = part_get_info_by_name(desc, PART_UBOOT, &info);
120 	if (ret > 0)
121 		image_sector = info.start;
122 
123 #endif
124 	if (IS_ENABLED(CONFIG_SPL_LOAD_FIT)) {
125 		header = (struct image_header *)(CONFIG_SYS_TEXT_BASE -
126 					 sizeof(struct image_header));
127 		ret = blk_dread(desc, image_sector, 1, header);
128 		if (ret != 1)
129 			return -ENODEV;
130 
131 #ifdef CONFIG_SPL_FIT_IMAGE_MULTIPLE
132 		if (image_get_magic(header) == FDT_MAGIC ||
133 		    CONFIG_SPL_FIT_IMAGE_MULTIPLE > 1) {
134 #else
135 		if (image_get_magic(header) == FDT_MAGIC) {
136 #endif
137 			struct spl_load_info load;
138 
139 			load.dev = desc;
140 			load.priv = NULL;
141 			load.filename = NULL;
142 			load.bl_len = desc->blksz;
143 			load.read = mtd_spl_load_read;
144 
145 			ret = spl_load_simple_fit(spl_image, &load,
146 						  image_sector,
147 						  header);
148 		}
149 	}
150 
151 	if (!ret)
152 		return 0;
153 
154 	if (IS_ENABLED(CONFIG_SPL_LOAD_RKFW)) {
155 #ifdef CONFIG_SPL_LOAD_RKFW
156 		ret = spl_mtd_load_rkfw(spl_image, desc);
157 #endif
158 	}
159 
160 	return ret;
161 }
162 
163 SPL_LOAD_IMAGE_METHOD("MTD0", 0, BOOT_DEVICE_MTD_BLK_NAND, spl_mtd_load_image);
164 SPL_LOAD_IMAGE_METHOD("MTD1", 0, BOOT_DEVICE_MTD_BLK_SPI_NAND, spl_mtd_load_image);
165 SPL_LOAD_IMAGE_METHOD("MTD2", 0, BOOT_DEVICE_MTD_BLK_SPI_NOR, spl_mtd_load_image);
166