mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-10-26 09:38:14 +00:00 
			
		
		
		
	It is possible to boot x86-based ChromeOS machines by parsing a table and locating the kernel and command line. Add a bootmeth for this. Signed-off-by: Simon Glass <sjg@chromium.org> Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
		
			
				
	
	
		
			213 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			213 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0+
 | |
| /*
 | |
|  * Bootmethod for ChromiumOS
 | |
|  *
 | |
|  * Copyright 2023 Google LLC
 | |
|  * Written by Simon Glass <sjg@chromium.org>
 | |
|  */
 | |
| 
 | |
| #define LOG_CATEGORY UCLASS_BOOTSTD
 | |
| 
 | |
| #include <common.h>
 | |
| #include <blk.h>
 | |
| #include <bootdev.h>
 | |
| #include <bootflow.h>
 | |
| #include <bootmeth.h>
 | |
| #include <dm.h>
 | |
| #include <malloc.h>
 | |
| #include <mapmem.h>
 | |
| #include <part.h>
 | |
| #ifdef CONFIG_X86
 | |
| #include <asm/zimage.h>
 | |
| #endif
 | |
| #include <linux/sizes.h>
 | |
| 
 | |
| enum {
 | |
| 	/* Offsets in the kernel-partition header */
 | |
| 	KERN_START	= 0x4f0,
 | |
| 	KERN_SIZE	= 0x518,
 | |
| 
 | |
| 	SETUP_OFFSET	= 0x1000,	/* bytes before base */
 | |
| 	CMDLINE_OFFSET	= 0x2000,	/* bytes before base */
 | |
| 	OFFSET_BASE	= 0x100000,	/* assumed kernel load-address */
 | |
| };
 | |
| 
 | |
| static int cros_check(struct udevice *dev, struct bootflow_iter *iter)
 | |
| {
 | |
| 	/* This only works on block and network devices */
 | |
| 	if (bootflow_iter_check_blk(iter))
 | |
| 		return log_msg_ret("blk", -ENOTSUPP);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int copy_cmdline(const char *from, const char *uuid, char **bufp)
 | |
| {
 | |
| 	const int maxlen = 2048;
 | |
| 	char buf[maxlen];
 | |
| 	char *cmd, *to, *end;
 | |
| 	int len;
 | |
| 
 | |
| 	/* Allow space for cmdline + UUID */
 | |
| 	len = strnlen(from, sizeof(buf));
 | |
| 	if (len >= maxlen)
 | |
| 		return -E2BIG;
 | |
| 
 | |
| 	log_debug("uuid %d %s\n", uuid ? (int)strlen(uuid) : 0, uuid);
 | |
| 	for (to = buf, end = buf + maxlen - UUID_STR_LEN - 1; *from; from++) {
 | |
| 		if (to >= end)
 | |
| 			return -E2BIG;
 | |
| 		if (from[0] == '%' && from[1] == 'U' && uuid &&
 | |
| 		    strlen(uuid) == UUID_STR_LEN) {
 | |
| 			strcpy(to, uuid);
 | |
| 			to += UUID_STR_LEN;
 | |
| 			from++;
 | |
| 		} else {
 | |
| 			*to++ = *from;
 | |
| 		}
 | |
| 	}
 | |
| 	*to = '\0';
 | |
| 	len = to - buf;
 | |
| 	cmd = strdup(buf);
 | |
| 	if (!cmd)
 | |
| 		return -ENOMEM;
 | |
| 	free(*bufp);
 | |
| 	*bufp = cmd;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cros_read_bootflow(struct udevice *dev, struct bootflow *bflow)
 | |
| {
 | |
| 	struct blk_desc *desc = dev_get_uclass_plat(bflow->blk);
 | |
| 	ulong base, start, size, setup, cmdline, num_blks, kern_base;
 | |
| 	struct disk_partition info;
 | |
| 	const char *uuid = NULL;
 | |
| 	void *buf, *hdr;
 | |
| 	int ret;
 | |
| 
 | |
| 	log_debug("starting, part=%d\n", bflow->part);
 | |
| 
 | |
| 	/* We consider the whole disk, not any one partition */
 | |
| 	if (bflow->part)
 | |
| 		return log_msg_ret("max", -ENOENT);
 | |
| 
 | |
| 	/* Check partition 2 */
 | |
| 	ret = part_get_info(desc, 2, &info);
 | |
| 	if (ret)
 | |
| 		return log_msg_ret("part", ret);
 | |
| 
 | |
| 	/* Make a buffer for the header information */
 | |
| 	num_blks = SZ_4K >> desc->log2blksz;
 | |
| 	log_debug("Reading header, blk=%s, start=%lx, blocks=%lx\n",
 | |
| 		  bflow->blk->name, (ulong)info.start, num_blks);
 | |
| 	hdr = memalign(SZ_1K, SZ_4K);
 | |
| 	if (!hdr)
 | |
| 		return log_msg_ret("hdr", -ENOMEM);
 | |
| 	ret = blk_read(bflow->blk, info.start, num_blks, hdr);
 | |
| 	if (ret != num_blks)
 | |
| 		return log_msg_ret("inf", ret);
 | |
| 
 | |
| 	if (memcmp("CHROMEOS", hdr, 8))
 | |
| 		return -ENOENT;
 | |
| 
 | |
| 	log_info("Header at %lx\n", (ulong)map_to_sysmem(hdr));
 | |
| 	start = *(u32 *)(hdr + KERN_START);
 | |
| 	size = ALIGN(*(u32 *)(hdr + KERN_SIZE), desc->blksz);
 | |
| 	log_debug("Reading start %lx size %lx\n", start, size);
 | |
| 	bflow->size = size;
 | |
| 
 | |
| 	buf = memalign(SZ_1K, size);
 | |
| 	if (!buf)
 | |
| 		return log_msg_ret("buf", -ENOMEM);
 | |
| 	num_blks = size >> desc->log2blksz;
 | |
| 	log_debug("Reading data, blk=%s, start=%lx, blocks=%lx\n",
 | |
| 		  bflow->blk->name, (ulong)info.start, num_blks);
 | |
| 	ret = blk_read(bflow->blk, (ulong)info.start + 0x80, num_blks, buf);
 | |
| 	if (ret != num_blks)
 | |
| 		return log_msg_ret("inf", ret);
 | |
| 	base = map_to_sysmem(buf);
 | |
| 
 | |
| 	setup = base + start - OFFSET_BASE - SETUP_OFFSET;
 | |
| 	cmdline = base + start - OFFSET_BASE - CMDLINE_OFFSET;
 | |
| 	kern_base = base + start - OFFSET_BASE + SZ_16K;
 | |
| 	log_debug("base %lx setup %lx, cmdline %lx, kern_base %lx\n", base,
 | |
| 		  setup, cmdline, kern_base);
 | |
| 
 | |
| #ifdef CONFIG_X86
 | |
| 	const char *version;
 | |
| 
 | |
| 	version = zimage_get_kernel_version(map_sysmem(setup, 0),
 | |
| 					    map_sysmem(kern_base, 0));
 | |
| 	log_debug("version %s\n", version);
 | |
| 	if (version)
 | |
| 		bflow->name = strdup(version);
 | |
| #endif
 | |
| 	if (!bflow->name)
 | |
| 		bflow->name = strdup("ChromeOS");
 | |
| 	if (!bflow->name)
 | |
| 		return log_msg_ret("nam", -ENOMEM);
 | |
| 	bflow->os_name = strdup("ChromeOS");
 | |
| 	if (!bflow->os_name)
 | |
| 		return log_msg_ret("os", -ENOMEM);
 | |
| 
 | |
| #if CONFIG_IS_ENABLED(PARTITION_UUIDS)
 | |
| 	uuid = info.uuid;
 | |
| #endif
 | |
| 	ret = copy_cmdline(map_sysmem(cmdline, 0), uuid, &bflow->cmdline);
 | |
| 	if (ret)
 | |
| 		return log_msg_ret("cmd", ret);
 | |
| 
 | |
| 	bflow->state = BOOTFLOWST_READY;
 | |
| 	bflow->buf = buf;
 | |
| 	bflow->x86_setup = map_sysmem(setup, 0);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cros_read_file(struct udevice *dev, struct bootflow *bflow,
 | |
| 			 const char *file_path, ulong addr, ulong *sizep)
 | |
| {
 | |
| 	return -ENOSYS;
 | |
| }
 | |
| 
 | |
| static int cros_boot(struct udevice *dev, struct bootflow *bflow)
 | |
| {
 | |
| #ifdef CONFIG_X86
 | |
| 	zboot_start(map_to_sysmem(bflow->buf), bflow->size, 0, 0,
 | |
| 		    map_to_sysmem(bflow->x86_setup),
 | |
| 		    bflow->cmdline);
 | |
| #endif
 | |
| 
 | |
| 	return log_msg_ret("go", -EFAULT);
 | |
| }
 | |
| 
 | |
| static int cros_bootmeth_bind(struct udevice *dev)
 | |
| {
 | |
| 	struct bootmeth_uc_plat *plat = dev_get_uclass_plat(dev);
 | |
| 
 | |
| 	plat->desc = "ChromiumOS boot";
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct bootmeth_ops cros_bootmeth_ops = {
 | |
| 	.check		= cros_check,
 | |
| 	.read_bootflow	= cros_read_bootflow,
 | |
| 	.read_file	= cros_read_file,
 | |
| 	.boot		= cros_boot,
 | |
| };
 | |
| 
 | |
| static const struct udevice_id cros_bootmeth_ids[] = {
 | |
| 	{ .compatible = "u-boot,cros" },
 | |
| 	{ }
 | |
| };
 | |
| 
 | |
| U_BOOT_DRIVER(bootmeth_cros) = {
 | |
| 	.name		= "bootmeth_cros",
 | |
| 	.id		= UCLASS_BOOTMETH,
 | |
| 	.of_match	= cros_bootmeth_ids,
 | |
| 	.ops		= &cros_bootmeth_ops,
 | |
| 	.bind		= cros_bootmeth_bind,
 | |
| };
 |