mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-07 10:48:54 +00:00
636da2039a
Enable the support for boot image header version 3 and 4 using abootimg command. In order to use version 3 or 4: 1- Vendor boot image address should be given to abootimg cmd. abootimg addr $1 $vendor_boot_load_addr 2- "ramdisk_addr_r" env variable (ramdisk address) should be set to host the ramdisk : generic ramdisk + vendor ramdisk Replace "struct andr_boot_img_hdr_v0*" by "void *" in some functions since v3 and v4 are now supported as well. Signed-off-by: Safae Ouajih <souajih@baylibre.com> Reviewed-by: Mattijs Korpershoek <mkorpershoek@baylibre.com> Tested-by: Mattijs Korpershoek <mkorpershoek@baylibre.com>
293 lines
6.7 KiB
C
293 lines
6.7 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* (C) Copyright 2020
|
|
* Sam Protsenko <joe.skb7@gmail.com>
|
|
*/
|
|
|
|
#include <android_image.h>
|
|
#include <common.h>
|
|
#include <command.h>
|
|
#include <image.h>
|
|
#include <mapmem.h>
|
|
|
|
#define abootimg_addr() \
|
|
(_abootimg_addr == -1 ? image_load_addr : _abootimg_addr)
|
|
|
|
/* Please use abootimg_addr() macro to obtain the boot image address */
|
|
static ulong _abootimg_addr = -1;
|
|
static ulong _avendor_bootimg_addr = -1;
|
|
|
|
ulong get_abootimg_addr(void)
|
|
{
|
|
return (_abootimg_addr == -1 ? image_load_addr : _abootimg_addr);
|
|
}
|
|
|
|
ulong get_avendor_bootimg_addr(void)
|
|
{
|
|
return _avendor_bootimg_addr;
|
|
}
|
|
|
|
static int abootimg_get_ver(int argc, char *const argv[])
|
|
{
|
|
const struct andr_boot_img_hdr_v0 *hdr;
|
|
int res = CMD_RET_SUCCESS;
|
|
|
|
if (argc > 1)
|
|
return CMD_RET_USAGE;
|
|
|
|
hdr = map_sysmem(abootimg_addr(), sizeof(*hdr));
|
|
if (!is_android_boot_image_header(hdr)) {
|
|
printf("Error: Boot Image header is incorrect\n");
|
|
res = CMD_RET_FAILURE;
|
|
goto exit;
|
|
}
|
|
|
|
if (argc == 0)
|
|
printf("%u\n", hdr->header_version);
|
|
else
|
|
env_set_ulong(argv[0], hdr->header_version);
|
|
|
|
exit:
|
|
unmap_sysmem(hdr);
|
|
return res;
|
|
}
|
|
|
|
static int abootimg_get_recovery_dtbo(int argc, char *const argv[])
|
|
{
|
|
ulong addr;
|
|
u32 size;
|
|
|
|
if (argc > 2)
|
|
return CMD_RET_USAGE;
|
|
|
|
if (!android_image_get_dtbo(abootimg_addr(), &addr, &size))
|
|
return CMD_RET_FAILURE;
|
|
|
|
if (argc == 0) {
|
|
printf("%lx\n", addr);
|
|
} else {
|
|
env_set_hex(argv[0], addr);
|
|
if (argc == 2)
|
|
env_set_hex(argv[1], size);
|
|
}
|
|
|
|
return CMD_RET_SUCCESS;
|
|
}
|
|
|
|
static int abootimg_get_dtb_load_addr(int argc, char *const argv[])
|
|
{
|
|
if (argc > 1)
|
|
return CMD_RET_USAGE;
|
|
struct andr_image_data img_data = {0};
|
|
const struct andr_boot_img_hdr_v0 *hdr;
|
|
const struct andr_vnd_boot_img_hdr *vhdr;
|
|
|
|
hdr = map_sysmem(abootimg_addr(), sizeof(*hdr));
|
|
if (get_avendor_bootimg_addr() != -1)
|
|
vhdr = map_sysmem(get_avendor_bootimg_addr(), sizeof(*vhdr));
|
|
|
|
if (!android_image_get_data(hdr, vhdr, &img_data)) {
|
|
if (get_avendor_bootimg_addr() != -1)
|
|
unmap_sysmem(vhdr);
|
|
unmap_sysmem(hdr);
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
if (get_avendor_bootimg_addr() != -1)
|
|
unmap_sysmem(vhdr);
|
|
unmap_sysmem(hdr);
|
|
|
|
if (img_data.header_version < 2) {
|
|
printf("Error: header_version must be >= 2 for this\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
if (!img_data.dtb_load_addr) {
|
|
printf("Error: failed to read dtb_load_addr\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
if (argc == 0)
|
|
printf("%lx\n", (ulong)img_data.dtb_load_addr);
|
|
else
|
|
env_set_hex(argv[0], (ulong)img_data.dtb_load_addr);
|
|
|
|
return CMD_RET_SUCCESS;
|
|
}
|
|
|
|
static int abootimg_get_dtb_by_index(int argc, char *const argv[])
|
|
{
|
|
const char *index_str;
|
|
u32 num;
|
|
char *endp;
|
|
ulong addr;
|
|
u32 size;
|
|
|
|
if (argc < 1 || argc > 3)
|
|
return CMD_RET_USAGE;
|
|
|
|
index_str = argv[0] + strlen("--index=");
|
|
if (index_str[0] == '\0') {
|
|
printf("Error: Wrong index num\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
num = simple_strtoul(index_str, &endp, 0);
|
|
if (*endp != '\0') {
|
|
printf("Error: Wrong index num\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
if (!android_image_get_dtb_by_index(abootimg_addr(),
|
|
get_avendor_bootimg_addr(), num,
|
|
&addr, &size)) {
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
if (argc == 1) {
|
|
printf("%lx\n", addr);
|
|
} else {
|
|
if (env_set_hex(argv[1], addr)) {
|
|
printf("Error: Can't set [addr_var]\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
if (argc == 3) {
|
|
if (env_set_hex(argv[2], size)) {
|
|
printf("Error: Can't set [size_var]\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return CMD_RET_SUCCESS;
|
|
}
|
|
|
|
static int abootimg_get_dtb(int argc, char *const argv[])
|
|
{
|
|
if (argc < 1)
|
|
return CMD_RET_USAGE;
|
|
|
|
if (strstr(argv[0], "--index="))
|
|
return abootimg_get_dtb_by_index(argc, argv);
|
|
|
|
return CMD_RET_USAGE;
|
|
}
|
|
|
|
static int do_abootimg_addr(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char *const argv[])
|
|
{
|
|
char *endp;
|
|
ulong img_addr;
|
|
|
|
if (argc < 2 || argc > 3)
|
|
return CMD_RET_USAGE;
|
|
|
|
img_addr = hextoul(argv[1], &endp);
|
|
if (*endp != '\0') {
|
|
printf("Error: Wrong image address\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
_abootimg_addr = img_addr;
|
|
|
|
if (argc == 3) {
|
|
img_addr = simple_strtoul(argv[2], &endp, 16);
|
|
if (*endp != '\0') {
|
|
printf("Error: Wrong vendor image address\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
_avendor_bootimg_addr = img_addr;
|
|
}
|
|
|
|
return CMD_RET_SUCCESS;
|
|
}
|
|
|
|
static int do_abootimg_get(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char *const argv[])
|
|
{
|
|
const char *param;
|
|
|
|
if (argc < 2)
|
|
return CMD_RET_USAGE;
|
|
|
|
param = argv[1];
|
|
argc -= 2;
|
|
argv += 2;
|
|
if (!strcmp(param, "ver"))
|
|
return abootimg_get_ver(argc, argv);
|
|
else if (!strcmp(param, "recovery_dtbo"))
|
|
return abootimg_get_recovery_dtbo(argc, argv);
|
|
else if (!strcmp(param, "dtb_load_addr"))
|
|
return abootimg_get_dtb_load_addr(argc, argv);
|
|
else if (!strcmp(param, "dtb"))
|
|
return abootimg_get_dtb(argc, argv);
|
|
|
|
return CMD_RET_USAGE;
|
|
}
|
|
|
|
static int do_abootimg_dump(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char *const argv[])
|
|
{
|
|
if (argc != 2)
|
|
return CMD_RET_USAGE;
|
|
|
|
if (!strcmp(argv[1], "dtb")) {
|
|
if (android_image_print_dtb_contents(abootimg_addr()))
|
|
return CMD_RET_FAILURE;
|
|
} else {
|
|
return CMD_RET_USAGE;
|
|
}
|
|
|
|
return CMD_RET_SUCCESS;
|
|
}
|
|
|
|
static struct cmd_tbl cmd_abootimg_sub[] = {
|
|
U_BOOT_CMD_MKENT(addr, 3, 1, do_abootimg_addr, "", ""),
|
|
U_BOOT_CMD_MKENT(dump, 2, 1, do_abootimg_dump, "", ""),
|
|
U_BOOT_CMD_MKENT(get, 5, 1, do_abootimg_get, "", ""),
|
|
};
|
|
|
|
static int do_abootimg(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char *const argv[])
|
|
{
|
|
struct cmd_tbl *cp;
|
|
|
|
cp = find_cmd_tbl(argv[1], cmd_abootimg_sub,
|
|
ARRAY_SIZE(cmd_abootimg_sub));
|
|
|
|
/* Strip off leading 'abootimg' command argument */
|
|
argc--;
|
|
argv++;
|
|
|
|
if (!cp || argc > cp->maxargs)
|
|
return CMD_RET_USAGE;
|
|
if (flag == CMD_FLAG_REPEAT && !cmd_is_repeatable(cp))
|
|
return CMD_RET_SUCCESS;
|
|
|
|
return cp->cmd(cmdtp, flag, argc, argv);
|
|
}
|
|
|
|
U_BOOT_CMD(
|
|
abootimg, CONFIG_SYS_MAXARGS, 0, do_abootimg,
|
|
"manipulate Android Boot Image",
|
|
"addr <boot_img_addr> [<vendor_boot_img_addr>]>\n"
|
|
" - set the address in RAM where boot image is located\n"
|
|
" ($loadaddr is used by default)\n"
|
|
"abootimg dump dtb\n"
|
|
" - print info for all DT blobs in DTB area\n"
|
|
"abootimg get ver [varname]\n"
|
|
" - get header version\n"
|
|
"abootimg get recovery_dtbo [addr_var [size_var]]\n"
|
|
" - get address and size (hex) of recovery DTBO area in the image\n"
|
|
" [addr_var]: variable name to contain DTBO area address\n"
|
|
" [size_var]: variable name to contain DTBO area size\n"
|
|
"abootimg get dtb_load_addr [varname]\n"
|
|
" - get load address (hex) of DTB, from image header\n"
|
|
"abootimg get dtb --index=<num> [addr_var [size_var]]\n"
|
|
" - get address and size (hex) of DT blob in the image by index\n"
|
|
" <num>: index number of desired DT blob in DTB area\n"
|
|
" [addr_var]: variable name to contain DT blob address\n"
|
|
" [size_var]: variable name to contain DT blob size"
|
|
);
|