2020-04-21 07:28:41 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2020 Stefan Roese <sr@denx.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
2020-05-10 17:40:01 +00:00
|
|
|
#include <image.h>
|
2020-05-10 17:40:05 +00:00
|
|
|
#include <log.h>
|
2020-04-21 07:28:45 +00:00
|
|
|
#include <malloc.h>
|
2023-10-14 20:47:55 +00:00
|
|
|
#include <mapmem.h>
|
2023-05-29 12:04:06 +00:00
|
|
|
#include <asm/sections.h>
|
2020-04-21 07:28:41 +00:00
|
|
|
#include <spl.h>
|
|
|
|
|
2020-04-21 07:28:45 +00:00
|
|
|
#include <lzma/LzmaTypes.h>
|
|
|
|
#include <lzma/LzmaDec.h>
|
|
|
|
#include <lzma/LzmaTools.h>
|
|
|
|
|
|
|
|
#define LZMA_LEN (1 << 20)
|
|
|
|
|
2023-05-29 12:04:06 +00:00
|
|
|
static void spl_parse_legacy_validate(uintptr_t start, uintptr_t size)
|
|
|
|
{
|
|
|
|
uintptr_t spl_start = (uintptr_t)_start;
|
2023-10-14 20:47:37 +00:00
|
|
|
uintptr_t spl_end = (uintptr_t)&_image_binary_end;
|
2023-05-29 12:04:06 +00:00
|
|
|
uintptr_t end = start + size;
|
|
|
|
|
|
|
|
if ((start >= spl_start && start < spl_end) ||
|
|
|
|
(end > spl_start && end <= spl_end) ||
|
|
|
|
(start < spl_start && end >= spl_end) ||
|
|
|
|
(start > end && end > spl_start))
|
|
|
|
panic("SPL: Image overlaps SPL\n");
|
|
|
|
|
|
|
|
if (size > CONFIG_SYS_BOOTM_LEN)
|
|
|
|
panic("SPL: Image too large\n");
|
|
|
|
}
|
|
|
|
|
2020-04-21 07:28:41 +00:00
|
|
|
int spl_parse_legacy_header(struct spl_image_info *spl_image,
|
2022-09-07 02:26:52 +00:00
|
|
|
const struct legacy_img_hdr *header)
|
2020-04-21 07:28:41 +00:00
|
|
|
{
|
2022-09-07 02:26:52 +00:00
|
|
|
u32 header_size = sizeof(struct legacy_img_hdr);
|
2020-04-21 07:28:41 +00:00
|
|
|
|
|
|
|
/* check uImage header CRC */
|
2020-04-21 07:28:42 +00:00
|
|
|
if (IS_ENABLED(CONFIG_SPL_LEGACY_IMAGE_CRC_CHECK) &&
|
|
|
|
!image_check_hcrc(header)) {
|
2020-04-21 07:28:41 +00:00
|
|
|
puts("SPL: Image header CRC check failed!\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (spl_image->flags & SPL_COPY_PAYLOAD_ONLY) {
|
|
|
|
/*
|
|
|
|
* On some system (e.g. powerpc), the load-address and
|
|
|
|
* entry-point is located at address 0. We can't load
|
|
|
|
* to 0-0x40. So skip header in this case.
|
|
|
|
*/
|
|
|
|
spl_image->load_addr = image_get_load(header);
|
|
|
|
spl_image->entry_point = image_get_ep(header);
|
|
|
|
spl_image->size = image_get_data_size(header);
|
|
|
|
} else {
|
|
|
|
spl_image->entry_point = image_get_ep(header);
|
|
|
|
/* Load including the header */
|
|
|
|
spl_image->load_addr = image_get_load(header) -
|
|
|
|
header_size;
|
|
|
|
spl_image->size = image_get_data_size(header) +
|
|
|
|
header_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_SPL_LEGACY_IMAGE_CRC_CHECK
|
|
|
|
/* store uImage data length and CRC to check later */
|
|
|
|
spl_image->dcrc_data = image_get_load(header);
|
|
|
|
spl_image->dcrc_length = image_get_data_size(header);
|
|
|
|
spl_image->dcrc = image_get_dcrc(header);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
spl_image->os = image_get_os(header);
|
|
|
|
spl_image->name = image_get_name(header);
|
|
|
|
debug(SPL_TPL_PROMPT
|
|
|
|
"payload image: %32s load addr: 0x%lx size: %d\n",
|
|
|
|
spl_image->name, spl_image->load_addr, spl_image->size);
|
|
|
|
|
2023-05-29 12:04:06 +00:00
|
|
|
spl_parse_legacy_validate(spl_image->load_addr, spl_image->size);
|
|
|
|
spl_parse_legacy_validate(spl_image->entry_point, 0);
|
|
|
|
|
2020-04-21 07:28:41 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2020-04-21 07:28:43 +00:00
|
|
|
|
2020-04-21 07:28:45 +00:00
|
|
|
/*
|
|
|
|
* This function is added explicitly to avoid code size increase, when
|
|
|
|
* no compression method is enabled. The compiler will optimize the
|
|
|
|
* following switch/case statement in spl_load_legacy_img() away due to
|
|
|
|
* Dead Code Elimination.
|
|
|
|
*/
|
2022-09-07 02:26:52 +00:00
|
|
|
static inline int spl_image_get_comp(const struct legacy_img_hdr *hdr)
|
2020-04-21 07:28:45 +00:00
|
|
|
{
|
|
|
|
if (IS_ENABLED(CONFIG_SPL_LZMA))
|
|
|
|
return image_get_comp(hdr);
|
|
|
|
|
|
|
|
return IH_COMP_NONE;
|
|
|
|
}
|
|
|
|
|
2020-04-21 07:28:43 +00:00
|
|
|
int spl_load_legacy_img(struct spl_image_info *spl_image,
|
2022-01-14 13:31:38 +00:00
|
|
|
struct spl_boot_device *bootdev,
|
2022-09-29 10:11:28 +00:00
|
|
|
struct spl_load_info *load, ulong offset,
|
|
|
|
struct legacy_img_hdr *hdr)
|
2020-04-21 07:28:43 +00:00
|
|
|
{
|
2020-04-21 07:28:45 +00:00
|
|
|
__maybe_unused SizeT lzma_len;
|
|
|
|
__maybe_unused void *src;
|
|
|
|
ulong dataptr;
|
2020-04-21 07:28:43 +00:00
|
|
|
int ret;
|
|
|
|
|
2022-05-20 03:23:58 +00:00
|
|
|
/*
|
|
|
|
* If the payload is compressed, the decompressed data should be
|
|
|
|
* directly write to its load address.
|
|
|
|
*/
|
2022-09-29 10:11:28 +00:00
|
|
|
if (spl_image_get_comp(hdr) != IH_COMP_NONE)
|
2022-05-20 03:23:58 +00:00
|
|
|
spl_image->flags |= SPL_COPY_PAYLOAD_ONLY;
|
|
|
|
|
2022-09-29 10:11:28 +00:00
|
|
|
ret = spl_parse_image_header(spl_image, bootdev, hdr);
|
2020-04-21 07:28:43 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* Read image */
|
2022-09-29 10:11:28 +00:00
|
|
|
switch (spl_image_get_comp(hdr)) {
|
2020-04-21 07:28:45 +00:00
|
|
|
case IH_COMP_NONE:
|
2022-09-29 10:11:28 +00:00
|
|
|
dataptr = offset;
|
2022-05-20 03:23:58 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Image header will be skipped only if SPL_COPY_PAYLOAD_ONLY
|
|
|
|
* is set
|
|
|
|
*/
|
|
|
|
if (spl_image->flags & SPL_COPY_PAYLOAD_ONLY)
|
2022-12-09 11:40:21 +00:00
|
|
|
dataptr += sizeof(*hdr);
|
2022-05-20 03:23:58 +00:00
|
|
|
|
2020-04-21 07:28:45 +00:00
|
|
|
load->read(load, dataptr, spl_image->size,
|
2023-10-14 20:47:55 +00:00
|
|
|
map_sysmem(spl_image->load_addr, spl_image->size));
|
2020-04-21 07:28:45 +00:00
|
|
|
break;
|
|
|
|
|
2023-11-04 20:37:43 +00:00
|
|
|
case IH_COMP_LZMA: {
|
|
|
|
ulong overhead, size;
|
|
|
|
|
2020-04-21 07:28:45 +00:00
|
|
|
lzma_len = LZMA_LEN;
|
|
|
|
|
2022-05-20 03:23:58 +00:00
|
|
|
/* dataptr points to compressed payload */
|
2023-11-08 16:48:43 +00:00
|
|
|
dataptr = ALIGN_DOWN(sizeof(*hdr), spl_get_bl_len(load));
|
2023-11-04 20:37:43 +00:00
|
|
|
overhead = sizeof(*hdr) - dataptr;
|
2023-11-08 16:48:43 +00:00
|
|
|
size = ALIGN(spl_image->size + overhead, spl_get_bl_len(load));
|
2023-11-04 20:37:43 +00:00
|
|
|
dataptr += offset;
|
2022-05-20 03:23:58 +00:00
|
|
|
|
2020-04-21 07:28:45 +00:00
|
|
|
debug("LZMA: Decompressing %08lx to %08lx\n",
|
|
|
|
dataptr, spl_image->load_addr);
|
2023-11-04 20:37:43 +00:00
|
|
|
src = malloc(size);
|
2020-04-21 07:28:45 +00:00
|
|
|
if (!src) {
|
|
|
|
printf("Unable to allocate %d bytes for LZMA\n",
|
|
|
|
spl_image->size);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2023-11-04 20:37:43 +00:00
|
|
|
load->read(load, dataptr, size, src);
|
2023-10-14 20:47:55 +00:00
|
|
|
ret = lzmaBuffToBuffDecompress(map_sysmem(spl_image->load_addr,
|
|
|
|
spl_image->size),
|
2023-11-04 20:37:43 +00:00
|
|
|
&lzma_len, src + overhead,
|
|
|
|
spl_image->size);
|
2020-04-21 07:28:45 +00:00
|
|
|
if (ret) {
|
|
|
|
printf("LZMA decompression error: %d\n", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
spl_image->size = lzma_len;
|
|
|
|
break;
|
2023-11-04 20:37:43 +00:00
|
|
|
}
|
2020-04-21 07:28:45 +00:00
|
|
|
default:
|
|
|
|
debug("Compression method %s is not supported\n",
|
2022-09-29 10:11:28 +00:00
|
|
|
genimg_get_comp_short_name(image_get_comp(hdr)));
|
2020-04-21 07:28:45 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2020-04-21 07:28:43 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|