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