Loading...
// 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");