mirror of
https://github.com/AsahiLinux/u-boot
synced 2024-12-21 02:33:07 +00:00
3854296801
This reverts commit 1f63ee6566
.
As reported by Ye Li on ML:
1. Removing the image_offset will break secondary (redundant) boot support for sd and emmc.
2. When booting from emmc boot partition, the image_offset is 0. But the flash.bin
generated by mkimage with imximage-8mp-lpddr4.cfg is for sd. It expects to be burn at 32KB offset.
The fit offset 0x60000 has already included the 32KB offset. So when you burn this flash.bin
to emmc boot partition at offset 0, the fit offset should subtract the 32KB (0x60000 - 0x8000).
Signed-off-by: Stefano Babic <sbabic@denx.de>
293 lines
6.3 KiB
C
293 lines
6.3 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* Copyright 2019 NXP
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <errno.h>
|
|
#include <image.h>
|
|
#include <log.h>
|
|
#include <linux/libfdt.h>
|
|
#include <spl.h>
|
|
|
|
#include <asm/arch/sys_proto.h>
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
static int is_boot_from_stream_device(u32 boot)
|
|
{
|
|
u32 interface;
|
|
|
|
interface = boot >> 16;
|
|
if (interface >= BT_DEV_TYPE_USB)
|
|
return 1;
|
|
|
|
if (interface == BT_DEV_TYPE_MMC && (boot & 1))
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ulong spl_romapi_read_seekable(struct spl_load_info *load,
|
|
ulong sector, ulong count,
|
|
void *buf)
|
|
{
|
|
u32 pagesize = *(u32 *)load->priv;
|
|
volatile gd_t *pgd = gd;
|
|
ulong byte = count * pagesize;
|
|
int ret;
|
|
u32 offset;
|
|
|
|
offset = sector * pagesize;
|
|
|
|
debug("ROM API load from 0x%x, size 0x%x\n", offset, (u32)byte);
|
|
|
|
ret = g_rom_api->download_image(buf, offset, byte,
|
|
((uintptr_t)buf) ^ offset ^ byte);
|
|
gd = pgd;
|
|
|
|
if (ret == ROM_API_OKAY)
|
|
return count;
|
|
|
|
printf("ROM API Failure when load 0x%x\n", offset);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int spl_romapi_load_image_seekable(struct spl_image_info *spl_image,
|
|
struct spl_boot_device *bootdev,
|
|
u32 rom_bt_dev)
|
|
{
|
|
volatile gd_t *pgd = gd;
|
|
int ret;
|
|
u32 offset;
|
|
u32 pagesize, size;
|
|
struct image_header *header;
|
|
u32 image_offset;
|
|
|
|
ret = g_rom_api->query_boot_infor(QUERY_IVT_OFF, &offset,
|
|
((uintptr_t)&offset) ^ QUERY_IVT_OFF);
|
|
ret |= g_rom_api->query_boot_infor(QUERY_PAGE_SZ, &pagesize,
|
|
((uintptr_t)&pagesize) ^ QUERY_PAGE_SZ);
|
|
ret |= g_rom_api->query_boot_infor(QUERY_IMG_OFF, &image_offset,
|
|
((uintptr_t)&image_offset) ^ QUERY_IMG_OFF);
|
|
|
|
gd = pgd;
|
|
|
|
if (ret != ROM_API_OKAY) {
|
|
puts("ROMAPI: Failure query boot infor pagesize/offset\n");
|
|
return -1;
|
|
}
|
|
|
|
header = (struct image_header *)(CONFIG_SPL_IMX_ROMAPI_LOADADDR);
|
|
|
|
printf("image offset 0x%x, pagesize 0x%x, ivt offset 0x%x\n",
|
|
image_offset, pagesize, offset);
|
|
|
|
if (((rom_bt_dev >> 16) & 0xff) == BT_DEV_TYPE_FLEXSPINOR)
|
|
offset = CONFIG_SYS_MMCSD_RAW_MODE_U_BOOT_SECTOR * 512;
|
|
else
|
|
offset = image_offset +
|
|
CONFIG_SYS_MMCSD_RAW_MODE_U_BOOT_SECTOR * 512 - 0x8000;
|
|
|
|
size = ALIGN(sizeof(struct image_header), pagesize);
|
|
ret = g_rom_api->download_image((u8 *)header, offset, size,
|
|
((uintptr_t)header) ^ offset ^ size);
|
|
gd = pgd;
|
|
|
|
if (ret != ROM_API_OKAY) {
|
|
printf("ROMAPI: download failure offset 0x%x size 0x%x\n",
|
|
offset, size);
|
|
return -1;
|
|
}
|
|
|
|
if (IS_ENABLED(CONFIG_SPL_LOAD_FIT) &&
|
|
image_get_magic(header) == FDT_MAGIC) {
|
|
struct spl_load_info load;
|
|
|
|
memset(&load, 0, sizeof(load));
|
|
load.bl_len = pagesize;
|
|
load.read = spl_romapi_read_seekable;
|
|
load.priv = &pagesize;
|
|
return spl_load_simple_fit(spl_image, &load,
|
|
offset / pagesize, header);
|
|
} else {
|
|
/* TODO */
|
|
puts("Can't support legacy image\n");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ulong spl_ram_load_read(struct spl_load_info *load, ulong sector,
|
|
ulong count, void *buf)
|
|
{
|
|
memcpy(buf, (void *)(sector), count);
|
|
|
|
if (load->priv) {
|
|
ulong *p = (ulong *)load->priv;
|
|
ulong total = sector + count;
|
|
|
|
if (total > *p)
|
|
*p = total;
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
static ulong get_fit_image_size(void *fit)
|
|
{
|
|
struct spl_image_info spl_image;
|
|
struct spl_load_info spl_load_info;
|
|
ulong last = (ulong)fit;
|
|
|
|
memset(&spl_load_info, 0, sizeof(spl_load_info));
|
|
spl_load_info.bl_len = 1;
|
|
spl_load_info.read = spl_ram_load_read;
|
|
spl_load_info.priv = &last;
|
|
|
|
spl_load_simple_fit(&spl_image, &spl_load_info,
|
|
(uintptr_t)fit, fit);
|
|
|
|
return last - (ulong)fit;
|
|
}
|
|
|
|
u8 *search_fit_header(u8 *p, int size)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < size; i += 4)
|
|
if (genimg_get_format(p + i) == IMAGE_FORMAT_FIT)
|
|
return p + i;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static int spl_romapi_load_image_stream(struct spl_image_info *spl_image,
|
|
struct spl_boot_device *bootdev)
|
|
{
|
|
struct spl_load_info load;
|
|
volatile gd_t *pgd = gd;
|
|
u32 pagesize, pg;
|
|
int ret;
|
|
int i = 0;
|
|
u8 *p = (u8 *)CONFIG_SPL_IMX_ROMAPI_LOADADDR;
|
|
u8 *pfit = NULL;
|
|
int imagesize;
|
|
int total;
|
|
|
|
ret = g_rom_api->query_boot_infor(QUERY_PAGE_SZ, &pagesize,
|
|
((uintptr_t)&pagesize) ^ QUERY_PAGE_SZ);
|
|
gd = pgd;
|
|
|
|
if (ret != ROM_API_OKAY)
|
|
puts("failure at query_boot_info\n");
|
|
|
|
pg = pagesize;
|
|
if (pg < 1024)
|
|
pg = 1024;
|
|
|
|
for (i = 0; i < 640; i++) {
|
|
ret = g_rom_api->download_image(p, 0, pg,
|
|
((uintptr_t)p) ^ pg);
|
|
gd = pgd;
|
|
|
|
if (ret != ROM_API_OKAY) {
|
|
puts("Steam(USB) download failure\n");
|
|
return -1;
|
|
}
|
|
|
|
pfit = search_fit_header(p, pg);
|
|
p += pg;
|
|
|
|
if (pfit)
|
|
break;
|
|
}
|
|
|
|
if (!pfit) {
|
|
puts("Can't found uboot FIT image in 640K range \n");
|
|
return -1;
|
|
}
|
|
|
|
if (p - pfit < sizeof(struct fdt_header)) {
|
|
ret = g_rom_api->download_image(p, 0, pg, ((uintptr_t)p) ^ pg);
|
|
gd = pgd;
|
|
|
|
if (ret != ROM_API_OKAY) {
|
|
puts("Steam(USB) download failure\n");
|
|
return -1;
|
|
}
|
|
|
|
p += pg;
|
|
}
|
|
|
|
imagesize = fit_get_size(pfit);
|
|
printf("Find FIT header 0x&%p, size %d\n", pfit, imagesize);
|
|
|
|
if (p - pfit < imagesize) {
|
|
imagesize -= p - pfit;
|
|
/*need pagesize hear after ROM fix USB problme*/
|
|
imagesize += pg - 1;
|
|
imagesize /= pg;
|
|
imagesize *= pg;
|
|
|
|
printf("Need continue download %d\n", imagesize);
|
|
|
|
ret = g_rom_api->download_image(p, 0, imagesize,
|
|
((uintptr_t)p) ^ imagesize);
|
|
gd = pgd;
|
|
|
|
p += imagesize;
|
|
|
|
if (ret != ROM_API_OKAY) {
|
|
printf("Failure download %d\n", imagesize);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
total = get_fit_image_size(pfit);
|
|
total += 3;
|
|
total &= ~0x3;
|
|
|
|
imagesize = total - (p - pfit);
|
|
|
|
imagesize += pagesize - 1;
|
|
imagesize /= pagesize;
|
|
imagesize *= pagesize;
|
|
|
|
printf("Download %d, total fit %d\n", imagesize, total);
|
|
|
|
ret = g_rom_api->download_image(p, 0, imagesize,
|
|
((uintptr_t)p) ^ imagesize);
|
|
if (ret != ROM_API_OKAY)
|
|
printf("ROM download failure %d\n", imagesize);
|
|
|
|
memset(&load, 0, sizeof(load));
|
|
load.bl_len = 1;
|
|
load.read = spl_ram_load_read;
|
|
|
|
return spl_load_simple_fit(spl_image, &load, (ulong)pfit, pfit);
|
|
}
|
|
|
|
int board_return_to_bootrom(struct spl_image_info *spl_image,
|
|
struct spl_boot_device *bootdev)
|
|
{
|
|
volatile gd_t *pgd = gd;
|
|
int ret;
|
|
u32 boot;
|
|
|
|
ret = g_rom_api->query_boot_infor(QUERY_BT_DEV, &boot,
|
|
((uintptr_t)&boot) ^ QUERY_BT_DEV);
|
|
gd = pgd;
|
|
|
|
if (ret != ROM_API_OKAY) {
|
|
puts("ROMAPI: failure at query_boot_info\n");
|
|
return -1;
|
|
}
|
|
|
|
if (is_boot_from_stream_device(boot))
|
|
return spl_romapi_load_image_stream(spl_image, bootdev);
|
|
|
|
return spl_romapi_load_image_seekable(spl_image, bootdev, boot);
|
|
}
|