Provide a simple command to display the memory map described by this table. Signed-off-by: Simon Glass <sjg@chromium.org>
297 lines
7.8 KiB
C
297 lines
7.8 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* (C) Copyright 2015 Miao Yan <yanmiaobest@gmail.com>
|
|
* Copyright 2025 Simon Glass <sjg@chromium.org>
|
|
*/
|
|
|
|
#include <abuf.h>
|
|
#include <command.h>
|
|
#include <display_options.h>
|
|
#include <env.h>
|
|
#include <errno.h>
|
|
#include <qfw.h>
|
|
#include <dm.h>
|
|
#include <u-boot/uuid.h>
|
|
|
|
static struct udevice *qfw_dev;
|
|
|
|
static int qemu_fwcfg_cmd_list_firmware(void)
|
|
{
|
|
int ret;
|
|
struct fw_cfg_file_iter iter;
|
|
struct fw_file *file;
|
|
|
|
/* make sure fw_list is loaded */
|
|
ret = qfw_read_firmware_list(qfw_dev);
|
|
if (ret)
|
|
return ret;
|
|
|
|
printf(" Addr Size Sel Name\n");
|
|
printf("-------- -------- --- ------------\n");
|
|
for (file = qfw_file_iter_init(qfw_dev, &iter);
|
|
!qfw_file_iter_end(&iter);
|
|
file = qfw_file_iter_next(&iter)) {
|
|
printf("%8lx %8x %3x %-56s\n", file->addr,
|
|
be32_to_cpu(file->cfg.size),
|
|
be16_to_cpu(file->cfg.select), file->cfg.name);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int qemu_fwcfg_do_list(struct cmd_tbl *cmdtp, int flag,
|
|
int argc, char *const argv[])
|
|
{
|
|
if (qemu_fwcfg_cmd_list_firmware() < 0)
|
|
return CMD_RET_FAILURE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int qemu_fwcfg_do_cpus(struct cmd_tbl *cmdtp, int flag,
|
|
int argc, char *const argv[])
|
|
{
|
|
printf("%d cpu(s) online\n", qfw_online_cpus(qfw_dev));
|
|
return 0;
|
|
}
|
|
|
|
static int qemu_fwcfg_do_load(struct cmd_tbl *cmdtp, int flag,
|
|
int argc, char *const argv[])
|
|
{
|
|
char *env;
|
|
ulong load_addr;
|
|
ulong initrd_addr;
|
|
|
|
env = env_get("loadaddr");
|
|
load_addr = env ?
|
|
hextoul(env, NULL) :
|
|
CONFIG_SYS_LOAD_ADDR;
|
|
|
|
env = env_get("ramdiskaddr");
|
|
initrd_addr = env ?
|
|
hextoul(env, NULL) :
|
|
#ifdef CFG_RAMDISK_ADDR
|
|
CFG_RAMDISK_ADDR;
|
|
#else
|
|
0;
|
|
#endif
|
|
|
|
if (argc == 2) {
|
|
load_addr = hextoul(argv[0], NULL);
|
|
initrd_addr = hextoul(argv[1], NULL);
|
|
} else if (argc == 1) {
|
|
load_addr = hextoul(argv[0], NULL);
|
|
}
|
|
|
|
if (!load_addr || !initrd_addr) {
|
|
printf("missing load or initrd address\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
return qemu_fwcfg_setup_kernel(qfw_dev, load_addr, initrd_addr);
|
|
}
|
|
|
|
static uint get_val(enum fw_cfg_selector sel)
|
|
{
|
|
u32 val;
|
|
|
|
qfw_read_entry(qfw_dev, sel, sizeof(val), &val);
|
|
|
|
return val;
|
|
}
|
|
|
|
static int do_dump(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char * const argv[])
|
|
{
|
|
char uuid_str[UUID_STR_LEN + 1];
|
|
struct abuf buf;
|
|
struct uuid uuid;
|
|
u64 ramsize;
|
|
char sig[5];
|
|
|
|
qfw_read_entry(qfw_dev, FW_CFG_SIGNATURE, 4, sig);
|
|
sig[4] = '\0';
|
|
lprint_str("signature", sig);
|
|
lprint_num_32("id", get_val(FW_CFG_ID));
|
|
|
|
qfw_read_entry(qfw_dev, FW_CFG_UUID, sizeof(uuid), &uuid);
|
|
uuid_bin_to_str((u8 *)&uuid, uuid_str, 0);
|
|
lprint_str("uuid", uuid_str);
|
|
|
|
qfw_read_entry(qfw_dev, FW_CFG_RAM_SIZE, sizeof(ramsize), &ramsize);
|
|
lprint_num_ll("ram_size", ramsize);
|
|
|
|
lprint_num_32("nographic", get_val(FW_CFG_NOGRAPHIC));
|
|
lprint_num_32("nb cpus", get_val(FW_CFG_NB_CPUS));
|
|
lprint_num_32("machine id", get_val(FW_CFG_MACHINE_ID));
|
|
lprint_num_32("kernel addr", get_val(FW_CFG_KERNEL_ADDR));
|
|
lprint_num_32("kernel size", get_val(FW_CFG_KERNEL_SIZE));
|
|
|
|
lprint_num_32("kernel cmdl", get_val(FW_CFG_KERNEL_CMDLINE));
|
|
lprint_num_32("initrd addr", get_val(FW_CFG_INITRD_ADDR));
|
|
lprint_num_32("initrd size", get_val(FW_CFG_INITRD_SIZE));
|
|
lprint_num_32("boot device", get_val(FW_CFG_BOOT_DEVICE));
|
|
lprint_num_32("numa", get_val(FW_CFG_NUMA));
|
|
lprint_num_32("boot menu", get_val(FW_CFG_BOOT_MENU));
|
|
lprint_num_32("max cpus", get_val(FW_CFG_MAX_CPUS));
|
|
lprint_num_32("kernel entry", get_val(FW_CFG_KERNEL_ENTRY));
|
|
|
|
if (!abuf_init_size(&buf, get_val(FW_CFG_CMDLINE_SIZE)))
|
|
goto nomem;
|
|
qfw_read_entry(qfw_dev, FW_CFG_CMDLINE_DATA, buf.size, buf.data);
|
|
|
|
lprint_num_32("cmdline addr", get_val(FW_CFG_CMDLINE_ADDR));
|
|
lprint_num_32("cmdline size", get_val(FW_CFG_CMDLINE_SIZE));
|
|
lprint_str("cmdline data", (const char *)buf.data);
|
|
lprint_num_32("setup addr", get_val(FW_CFG_SETUP_ADDR));
|
|
lprint_num_32("setup size", get_val(FW_CFG_SETUP_SIZE));
|
|
|
|
/* convert the number of files to little-endian */
|
|
lprint_num_32("file dir le",
|
|
be32_to_cpu(get_val(FW_CFG_FILE_DIR)));
|
|
abuf_uninit(&buf);
|
|
|
|
return 0;
|
|
|
|
nomem:
|
|
printf("Out of memory\n");
|
|
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
static int do_table(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char * const argv[])
|
|
{
|
|
struct bios_linker_entry *entry, *end;
|
|
struct abuf loader;
|
|
int ret, i;
|
|
|
|
ret = qfw_get_table_loader(qfw_dev, &loader);
|
|
if (ret) {
|
|
printf("Error %dE\n", ret);
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
for (entry = loader.data, end = loader.data + loader.size, i = 0;
|
|
entry < end; entry++, i++) {
|
|
int cmd = le32_to_cpu(entry->command);
|
|
const char *zone_name;
|
|
|
|
if (!cmd)
|
|
continue;
|
|
printf("%3d ", i);
|
|
switch (cmd) {
|
|
case BIOS_LINKER_LOADER_COMMAND_ALLOCATE:
|
|
zone_name = entry->alloc.zone == 2 ? "fseg" :
|
|
entry->alloc.zone == 1 ? "high" : "?";
|
|
printf("alloc: align %x zone %s name '%s'\n",
|
|
entry->alloc.align, zone_name,
|
|
entry->alloc.file);
|
|
break;
|
|
case BIOS_LINKER_LOADER_COMMAND_ADD_POINTER:
|
|
printf("add-ptr offset %x size %x dest '%s' src '%s'\n",
|
|
entry->pointer.offset, entry->pointer.size,
|
|
entry->pointer.dest_file,
|
|
entry->pointer.src_file);
|
|
break;
|
|
case BIOS_LINKER_LOADER_COMMAND_ADD_CHECKSUM:
|
|
printf("add-chksum offset %x start %x length %x name '%s'\n",
|
|
entry->cksum.offset, entry->cksum.start,
|
|
entry->cksum.length, entry->cksum.file);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int do_arch(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char * const argv[])
|
|
{
|
|
if (!IS_ENABLED(CONFIG_X86))
|
|
return 0;
|
|
|
|
lprint_num_32("acpi tables", get_val(FW_CFG_ACPI_TABLES));
|
|
lprint_num_32("smbios entrs", get_val(FW_CFG_SMBIOS_ENTRIES));
|
|
lprint_num_32("irq0 overr", get_val(FW_CFG_IRQ0_OVERRIDE));
|
|
lprint_num_32("hpet", get_val(FW_CFG_HPET));
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int do_read(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char * const argv[])
|
|
{
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
return CMD_RET_USAGE;
|
|
|
|
ret = qfw_load_file(qfw_dev, argv[1], hextoul(argv[0], NULL));
|
|
if (ret)
|
|
return CMD_RET_FAILURE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int do_e820(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char * const argv[])
|
|
{
|
|
if (!IS_ENABLED(CONFIG_X86)) {
|
|
printf("Not supported on this architecture\n");
|
|
return CMD_RET_FAILURE;
|
|
}
|
|
|
|
cmd_qfw_e820(qfw_dev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct cmd_tbl fwcfg_commands[] = {
|
|
U_BOOT_CMD_MKENT(list, 0, 1, qemu_fwcfg_do_list, "", ""),
|
|
U_BOOT_CMD_MKENT(cpus, 0, 1, qemu_fwcfg_do_cpus, "", ""),
|
|
U_BOOT_CMD_MKENT(load, 2, 1, qemu_fwcfg_do_load, "", ""),
|
|
U_BOOT_CMD_MKENT(dump, 0, 1, do_dump, "", ""),
|
|
U_BOOT_CMD_MKENT(table, 0, 1, do_table, "", ""),
|
|
U_BOOT_CMD_MKENT(arch, 0, 1, do_arch, "", ""),
|
|
U_BOOT_CMD_MKENT(read, 2, 1, do_read, "", ""),
|
|
U_BOOT_CMD_MKENT(e820, 0, 1, do_e820, "", ""),
|
|
};
|
|
|
|
static int do_qemu_fw(struct cmd_tbl *cmdtp, int flag, int argc,
|
|
char *const argv[])
|
|
{
|
|
int ret;
|
|
struct cmd_tbl *fwcfg_cmd;
|
|
|
|
ret = qfw_get_dev(&qfw_dev);
|
|
if (ret) {
|
|
printf("QEMU fw_cfg interface not found\n");
|
|
return CMD_RET_USAGE;
|
|
}
|
|
|
|
fwcfg_cmd = find_cmd_tbl(argv[1], fwcfg_commands,
|
|
ARRAY_SIZE(fwcfg_commands));
|
|
argc -= 2;
|
|
argv += 2;
|
|
if (!fwcfg_cmd || argc > fwcfg_cmd->maxargs)
|
|
return CMD_RET_USAGE;
|
|
|
|
ret = fwcfg_cmd->cmd(fwcfg_cmd, flag, argc, argv);
|
|
|
|
return cmd_process_error(fwcfg_cmd, ret);
|
|
}
|
|
|
|
U_BOOT_CMD(
|
|
qfw, 4, 1, do_qemu_fw,
|
|
"QEMU firmware interface",
|
|
"<command>\n"
|
|
" - dump : dump out all values\n"
|
|
" - list : print firmware(s) currently loaded\n"
|
|
" - cpus : print online cpu number\n"
|
|
" - load <kernel addr> <initrd addr> : load kernel and initrd (if any), and setup for zboot\n"
|
|
" - table : show /etc/table-loader\n"
|
|
" - arch : show arch-specific data\n"
|
|
" - read <addr> <filename> : read a flle into memory\n"
|
|
" - e820 : show QEMU e820 table");
|