mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-03 21:48:15 +00:00 
			
		
		
		
	Universal Payload provides a standard way of handing off control between two firmware phases. Add support for writing the handoff information from a structure. Signed-off-by: Simon Glass <sjg@chromium.org>
		
			
				
	
	
		
			623 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			623 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0+
 | 
						|
/*
 | 
						|
 * UPL handoff generation
 | 
						|
 *
 | 
						|
 * Copyright 2024 Google LLC
 | 
						|
 * Written by Simon Glass <sjg@chromium.org>
 | 
						|
 */
 | 
						|
 | 
						|
#define LOG_CATEGORY UCLASS_BOOTSTD
 | 
						|
 | 
						|
#include <log.h>
 | 
						|
#include <upl.h>
 | 
						|
#include <dm/ofnode.h>
 | 
						|
#include "upl_common.h"
 | 
						|
 | 
						|
/**
 | 
						|
 * write_addr() - Write an address
 | 
						|
 *
 | 
						|
 * Writes an address in the correct format, either 32- or 64-bit
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @node: Node to write to
 | 
						|
 * @prop: Property name to write
 | 
						|
 * @addr: Address to write
 | 
						|
 * Return: 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int write_addr(const struct upl *upl, ofnode node, const char *prop,
 | 
						|
		      ulong addr)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	if (upl->addr_cells == 1)
 | 
						|
		ret = ofnode_write_u32(node, prop, addr);
 | 
						|
	else
 | 
						|
		ret = ofnode_write_u64(node, prop, addr);
 | 
						|
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * write_size() - Write a size
 | 
						|
 *
 | 
						|
 * Writes a size in the correct format, either 32- or 64-bit
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @node: Node to write to
 | 
						|
 * @prop: Property name to write
 | 
						|
 * @size: Size to write
 | 
						|
 * Return: 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int write_size(const struct upl *upl, ofnode node, const char *prop,
 | 
						|
		      ulong size)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	if (upl->size_cells == 1)
 | 
						|
		ret = ofnode_write_u32(node, prop, size);
 | 
						|
	else
 | 
						|
		ret = ofnode_write_u64(node, prop, size);
 | 
						|
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ofnode_write_bitmask() - Write a bit mask as a string list
 | 
						|
 *
 | 
						|
 * @node: Node to write to
 | 
						|
 * @prop: Property name to write
 | 
						|
 * @names: Array of names for each bit
 | 
						|
 * @count: Number of array entries
 | 
						|
 * @value: Bit-mask value to write
 | 
						|
 * Return: 0 if OK, -EINVAL if a bit number is not defined, -ENOSPC if the
 | 
						|
 * string is too long for the (internal) buffer
 | 
						|
 */
 | 
						|
static int ofnode_write_bitmask(ofnode node, const char *prop,
 | 
						|
				const char *const names[], uint count,
 | 
						|
				uint value)
 | 
						|
{
 | 
						|
	char buf[128];
 | 
						|
	char *ptr, *end = buf + sizeof(buf);
 | 
						|
	uint bit;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	ptr = buf;
 | 
						|
	for (bit = 0; bit < count; bit++) {
 | 
						|
		if (value & BIT(bit)) {
 | 
						|
			const char *str = names[bit];
 | 
						|
			uint len;
 | 
						|
 | 
						|
			if (!str) {
 | 
						|
				log_debug("Unnamed bit number %d\n", bit);
 | 
						|
				return log_msg_ret("bit", -EINVAL);
 | 
						|
			}
 | 
						|
			len = strlen(str) + 1;
 | 
						|
			if (ptr + len > end) {
 | 
						|
				log_debug("String array too long\n");
 | 
						|
				return log_msg_ret("bit", -ENOSPC);
 | 
						|
			}
 | 
						|
 | 
						|
			memcpy(ptr, str, len);
 | 
						|
			ptr += len;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	ret = ofnode_write_prop(node, prop, buf, ptr - buf, true);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("wri", ret);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ofnode_write_value() - Write an int as a string value using a lookup
 | 
						|
 *
 | 
						|
 * @node: Node to write to
 | 
						|
 * @prop: Property name to write
 | 
						|
 * @names: Array of names for each int value
 | 
						|
 * @count: Number of array entries
 | 
						|
 * @value: Int value to write
 | 
						|
 * Return: 0 if OK, -EINVAL if a bit number is not defined, -ENOSPC if the
 | 
						|
 * string is too long for the (internal) buffer
 | 
						|
 */
 | 
						|
static int ofnode_write_value(ofnode node, const char *prop,
 | 
						|
			      const char *const names[], uint count,
 | 
						|
			      uint value)
 | 
						|
{
 | 
						|
	const char *str;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	if (value >= count) {
 | 
						|
		log_debug("Value of range %d\n", value);
 | 
						|
		return log_msg_ret("val", -ERANGE);
 | 
						|
	}
 | 
						|
	str = names[value];
 | 
						|
	if (!str) {
 | 
						|
		log_debug("Unnamed value %d\n", value);
 | 
						|
		return log_msg_ret("val", -EINVAL);
 | 
						|
	}
 | 
						|
	ret = ofnode_write_string(node, prop, str);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("wri", ret);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * add_root_props() - Add root properties to the tree
 | 
						|
 *
 | 
						|
 * @node: Node to add to
 | 
						|
 * Return 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int add_root_props(const struct upl *upl, ofnode node)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	ret = ofnode_write_u32(node, UPLP_ADDRESS_CELLS, upl->addr_cells);
 | 
						|
	if (!ret)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_SIZE_CELLS, upl->size_cells);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("cel", ret);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * add_upl_params() - Add UPL parameters node
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @options: /options node to add to
 | 
						|
 * Return 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int add_upl_params(const struct upl *upl, ofnode options)
 | 
						|
{
 | 
						|
	ofnode node;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	ret = ofnode_add_subnode(options, UPLN_UPL_PARAMS, &node);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("img", ret);
 | 
						|
 | 
						|
	ret = write_addr(upl, node, UPLP_SMBIOS, upl->smbios);
 | 
						|
	if (!ret)
 | 
						|
		ret = write_addr(upl, node, UPLP_ACPI, upl->acpi);
 | 
						|
	if (!ret && upl->bootmode)
 | 
						|
		ret = ofnode_write_bitmask(node, UPLP_BOOTMODE, bootmode_names,
 | 
						|
					   UPLBM_COUNT, upl->bootmode);
 | 
						|
	if (!ret)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_ADDR_WIDTH, upl->addr_width);
 | 
						|
	if (!ret)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_ACPI_NVS_SIZE,
 | 
						|
				       upl->acpi_nvs_size);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("cnf", ret);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * add_upl_image() - Add /options/upl-image nodes and properties to the tree
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @node: /options node to add to
 | 
						|
 * Return 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int add_upl_image(const struct upl *upl, ofnode options)
 | 
						|
{
 | 
						|
	ofnode node;
 | 
						|
	int ret, i;
 | 
						|
 | 
						|
	ret = ofnode_add_subnode(options, UPLN_UPL_IMAGE, &node);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("img", ret);
 | 
						|
 | 
						|
	if (upl->fit)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_FIT, upl->fit);
 | 
						|
	if (!ret && upl->conf_offset)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_CONF_OFFSET,
 | 
						|
				       upl->conf_offset);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("cnf", ret);
 | 
						|
 | 
						|
	for (i = 0; i < upl->image.count; i++) {
 | 
						|
		const struct upl_image *img = alist_get(&upl->image, i,
 | 
						|
							struct upl_image);
 | 
						|
		ofnode subnode;
 | 
						|
		char name[10];
 | 
						|
 | 
						|
		snprintf(name, sizeof(name), UPLN_IMAGE "-%d", i + 1);
 | 
						|
		ret = ofnode_add_subnode(node, name, &subnode);
 | 
						|
		if (ret)
 | 
						|
			return log_msg_ret("sub", ret);
 | 
						|
 | 
						|
		ret = write_addr(upl, subnode, UPLP_LOAD, img->load);
 | 
						|
		if (!ret)
 | 
						|
			ret = write_size(upl, subnode, UPLP_SIZE, img->size);
 | 
						|
		if (!ret && img->offset)
 | 
						|
			ret = ofnode_write_u32(subnode, UPLP_OFFSET,
 | 
						|
					       img->offset);
 | 
						|
		ret = ofnode_write_string(subnode, UPLP_DESCRIPTION,
 | 
						|
					  img->description);
 | 
						|
		if (ret)
 | 
						|
			return log_msg_ret("sim", ret);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * buffer_addr_size() - Generate a set of addr/size pairs
 | 
						|
 *
 | 
						|
 * Each base/size value from each region is written to the buffer in a suitable
 | 
						|
 * format to be written to the devicetree
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @buf: Buffer to write to
 | 
						|
 * @size: Buffer size
 | 
						|
 * @num_regions: Number of regions to process
 | 
						|
 * @region: List of regions to process (struct memregion)
 | 
						|
 * Returns: Number of bytes written, or -ENOSPC if the buffer is too small
 | 
						|
 */
 | 
						|
static int buffer_addr_size(const struct upl *upl, char *buf, int size,
 | 
						|
			    uint num_regions, const struct alist *region)
 | 
						|
{
 | 
						|
	char *ptr, *end = buf + size;
 | 
						|
	int i;
 | 
						|
 | 
						|
	ptr = buf;
 | 
						|
	for (i = 0; i < num_regions; i++) {
 | 
						|
		const struct memregion *reg = alist_get(region, i,
 | 
						|
							struct memregion);
 | 
						|
 | 
						|
		if (upl->addr_cells == 1)
 | 
						|
			*(u32 *)ptr = cpu_to_fdt32(reg->base);
 | 
						|
		else
 | 
						|
			*(u64 *)ptr = cpu_to_fdt64(reg->base);
 | 
						|
		ptr += upl->addr_cells * sizeof(u32);
 | 
						|
 | 
						|
		if (upl->size_cells == 1)
 | 
						|
			*(u32 *)ptr = cpu_to_fdt32(reg->size);
 | 
						|
		else
 | 
						|
			*(u64 *)ptr = cpu_to_fdt64(reg->size);
 | 
						|
		ptr += upl->size_cells * sizeof(u32);
 | 
						|
		if (ptr > end)
 | 
						|
			return -ENOSPC;
 | 
						|
	}
 | 
						|
 | 
						|
	return ptr - buf;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * add_upl_memory() - Add /memory nodes to the tree
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @root: Parent node to contain the new /memory nodes
 | 
						|
 * Return 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int add_upl_memory(const struct upl *upl, ofnode root)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < upl->mem.count; i++) {
 | 
						|
		const struct upl_mem *mem = alist_get(&upl->mem, i,
 | 
						|
						      struct upl_mem);
 | 
						|
		char buf[mem->region.count * sizeof(64) * 2];
 | 
						|
		const struct memregion *first;
 | 
						|
		char name[26];
 | 
						|
		int ret, len;
 | 
						|
		ofnode node;
 | 
						|
 | 
						|
		if (!mem->region.count) {
 | 
						|
			log_debug("Memory %d has no regions\n", i);
 | 
						|
			return log_msg_ret("reg", -EINVAL);
 | 
						|
		}
 | 
						|
		first = alist_get(&mem->region, 0, struct memregion);
 | 
						|
		sprintf(name, UPLN_MEMORY "@0x%lx", first->base);
 | 
						|
		ret = ofnode_add_subnode(root, name, &node);
 | 
						|
		if (ret)
 | 
						|
			return log_msg_ret("mem", ret);
 | 
						|
 | 
						|
		len = buffer_addr_size(upl, buf, sizeof(buf), mem->region.count,
 | 
						|
				       &mem->region);
 | 
						|
		if (len < 0)
 | 
						|
			return log_msg_ret("buf", len);
 | 
						|
 | 
						|
		ret = ofnode_write_prop(node, UPLP_REG, buf, len, true);
 | 
						|
		if (!ret && mem->hotpluggable)
 | 
						|
			ret = ofnode_write_bool(node, UPLP_HOTPLUGGABLE,
 | 
						|
						mem->hotpluggable);
 | 
						|
		if (ret)
 | 
						|
			return log_msg_ret("lst", ret);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * add_upl_memmap() - Add memory-map nodes to the tree
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @root: Parent node to contain the new /memory-map node and its subnodes
 | 
						|
 * Return 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int add_upl_memmap(const struct upl *upl, ofnode root)
 | 
						|
{
 | 
						|
	ofnode mem_node;
 | 
						|
	int i, ret;
 | 
						|
 | 
						|
	if (!upl->memmap.count)
 | 
						|
		return 0;
 | 
						|
	ret = ofnode_add_subnode(root, UPLN_MEMORY_MAP, &mem_node);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("img", ret);
 | 
						|
 | 
						|
	for (i = 0; i < upl->memmap.count; i++) {
 | 
						|
		const struct upl_memmap *memmap = alist_get(&upl->memmap, i,
 | 
						|
							struct upl_memmap);
 | 
						|
		char buf[memmap->region.count * sizeof(64) * 2];
 | 
						|
		const struct memregion *first;
 | 
						|
		char name[26];
 | 
						|
		int ret, len;
 | 
						|
		ofnode node;
 | 
						|
 | 
						|
		if (!memmap->region.count) {
 | 
						|
			log_debug("Memory %d has no regions\n", i);
 | 
						|
			return log_msg_ret("reg", -EINVAL);
 | 
						|
		}
 | 
						|
		first = alist_get(&memmap->region, 0, struct memregion);
 | 
						|
		sprintf(name, "%s@0x%lx", memmap->name, first->base);
 | 
						|
		ret = ofnode_add_subnode(mem_node, name, &node);
 | 
						|
		if (ret)
 | 
						|
			return log_msg_ret("memmap", ret);
 | 
						|
 | 
						|
		len = buffer_addr_size(upl, buf, sizeof(buf),
 | 
						|
				       memmap->region.count, &memmap->region);
 | 
						|
		if (len < 0)
 | 
						|
			return log_msg_ret("buf", len);
 | 
						|
		ret = ofnode_write_prop(node, UPLP_REG, buf, len, true);
 | 
						|
		if (!ret && memmap->usage)
 | 
						|
			ret = ofnode_write_bitmask(node, UPLP_USAGE,
 | 
						|
						   usage_names,
 | 
						|
						   UPLUS_COUNT, memmap->usage);
 | 
						|
		if (ret)
 | 
						|
			return log_msg_ret("lst", ret);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * add_upl_memres() - Add /memory-reserved nodes to the tree
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @root: Parent node to contain the new node
 | 
						|
 * Return 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int add_upl_memres(const struct upl *upl, ofnode root,
 | 
						|
			  bool skip_existing)
 | 
						|
{
 | 
						|
	ofnode mem_node;
 | 
						|
	int i, ret;
 | 
						|
 | 
						|
	if (!upl->memmap.count)
 | 
						|
		return 0;
 | 
						|
	ret = ofnode_add_subnode(root, UPLN_MEMORY_RESERVED, &mem_node);
 | 
						|
	if (ret) {
 | 
						|
		if (skip_existing && ret == -EEXIST)
 | 
						|
			return 0;
 | 
						|
		return log_msg_ret("img", ret);
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; i < upl->memres.count; i++) {
 | 
						|
		const struct upl_memres *memres = alist_get(&upl->memres, i,
 | 
						|
							struct upl_memres);
 | 
						|
		char buf[memres->region.count * sizeof(64) * 2];
 | 
						|
		const struct memregion *first;
 | 
						|
		char name[26];
 | 
						|
		int ret, len;
 | 
						|
		ofnode node;
 | 
						|
 | 
						|
		if (!memres->region.count) {
 | 
						|
			log_debug("Memory %d has no regions\n", i);
 | 
						|
			return log_msg_ret("reg", -EINVAL);
 | 
						|
		}
 | 
						|
		first = alist_get(&memres->region, 0, struct memregion);
 | 
						|
		sprintf(name, "%s@0x%lx", memres->name, first->base);
 | 
						|
		ret = ofnode_add_subnode(mem_node, name, &node);
 | 
						|
		if (ret)
 | 
						|
			return log_msg_ret("memres", ret);
 | 
						|
 | 
						|
		len = buffer_addr_size(upl, buf, sizeof(buf),
 | 
						|
				       memres->region.count, &memres->region);
 | 
						|
		ret = ofnode_write_prop(node, UPLP_REG, buf, len, true);
 | 
						|
		if (!ret && memres->no_map)
 | 
						|
			ret = ofnode_write_bool(node, UPLP_NO_MAP,
 | 
						|
						memres->no_map);
 | 
						|
		if (ret)
 | 
						|
			return log_msg_ret("lst", ret);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * add_upl_serial() - Add serial node
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @root: Parent node to contain the new node
 | 
						|
 * Return 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int add_upl_serial(const struct upl *upl, ofnode root,
 | 
						|
			  bool skip_existing)
 | 
						|
{
 | 
						|
	const struct upl_serial *ser = &upl->serial;
 | 
						|
	const struct memregion *first;
 | 
						|
	char name[26];
 | 
						|
	ofnode node;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	if (!ser->compatible || skip_existing)
 | 
						|
		return 0;
 | 
						|
	if (!ser->reg.count)
 | 
						|
		return log_msg_ret("ser", -EINVAL);
 | 
						|
	first = alist_get(&ser->reg, 0, struct memregion);
 | 
						|
	sprintf(name, UPLN_SERIAL "@0x%lx", first->base);
 | 
						|
	ret = ofnode_add_subnode(root, name, &node);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("img", ret);
 | 
						|
	ret = ofnode_write_string(node, UPLP_COMPATIBLE, ser->compatible);
 | 
						|
	if (!ret)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_CLOCK_FREQUENCY,
 | 
						|
				       ser->clock_frequency);
 | 
						|
	if (!ret)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_CURRENT_SPEED,
 | 
						|
				       ser->current_speed);
 | 
						|
	if (!ret) {
 | 
						|
		char buf[16];
 | 
						|
		int len;
 | 
						|
 | 
						|
		len = buffer_addr_size(upl, buf, sizeof(buf), 1, &ser->reg);
 | 
						|
		if (len < 0)
 | 
						|
			return log_msg_ret("buf", len);
 | 
						|
 | 
						|
		ret = ofnode_write_prop(node, UPLP_REG, buf, len, true);
 | 
						|
	}
 | 
						|
	if (!ret && ser->reg_io_shift != UPLD_REG_IO_SHIFT)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_REG_IO_SHIFT,
 | 
						|
				       ser->reg_io_shift);
 | 
						|
	if (!ret && ser->reg_offset != UPLD_REG_OFFSET)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_REG_OFFSET, ser->reg_offset);
 | 
						|
	if (!ret && ser->reg_io_width != UPLD_REG_IO_WIDTH)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_REG_IO_WIDTH,
 | 
						|
				       ser->reg_io_width);
 | 
						|
	if (!ret && ser->virtual_reg)
 | 
						|
		ret = write_addr(upl, node, UPLP_VIRTUAL_REG, ser->virtual_reg);
 | 
						|
	if (!ret) {
 | 
						|
		ret = ofnode_write_value(node, UPLP_ACCESS_TYPE, access_types,
 | 
						|
					 ARRAY_SIZE(access_types),
 | 
						|
					 ser->access_type);
 | 
						|
	}
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("ser", ret);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * add_upl_graphics() - Add graphics node
 | 
						|
 *
 | 
						|
 * @upl: UPL state
 | 
						|
 * @root: Parent node to contain the new node
 | 
						|
 * Return 0 if OK, -ve on error
 | 
						|
 */
 | 
						|
static int add_upl_graphics(const struct upl *upl, ofnode root)
 | 
						|
{
 | 
						|
	const struct upl_graphics *gra = &upl->graphics;
 | 
						|
	const struct memregion *first;
 | 
						|
	char name[36];
 | 
						|
	ofnode node;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	if (!gra->reg.count)
 | 
						|
		return log_msg_ret("gra", -ENOENT);
 | 
						|
	first = alist_get(&gra->reg, 0, struct memregion);
 | 
						|
	sprintf(name, UPLN_GRAPHICS "@0x%lx", first->base);
 | 
						|
	ret = ofnode_add_subnode(root, name, &node);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("gra", ret);
 | 
						|
 | 
						|
	ret = ofnode_write_string(node, UPLP_COMPATIBLE, UPLC_GRAPHICS);
 | 
						|
	if (!ret) {
 | 
						|
		char buf[16];
 | 
						|
		int len;
 | 
						|
 | 
						|
		len = buffer_addr_size(upl, buf, sizeof(buf), 1, &gra->reg);
 | 
						|
		if (len < 0)
 | 
						|
			return log_msg_ret("buf", len);
 | 
						|
 | 
						|
		ret = ofnode_write_prop(node, UPLP_REG, buf, len, true);
 | 
						|
	}
 | 
						|
	if (!ret)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_WIDTH, gra->width);
 | 
						|
	if (!ret)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_HEIGHT, gra->height);
 | 
						|
	if (!ret)
 | 
						|
		ret = ofnode_write_u32(node, UPLP_STRIDE, gra->stride);
 | 
						|
	if (!ret) {
 | 
						|
		ret = ofnode_write_value(node, UPLP_GRAPHICS_FORMAT,
 | 
						|
					 graphics_formats,
 | 
						|
					 ARRAY_SIZE(graphics_formats),
 | 
						|
					 gra->format);
 | 
						|
	}
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("pro", ret);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int upl_write_handoff(const struct upl *upl, ofnode root, bool skip_existing)
 | 
						|
{
 | 
						|
	ofnode options;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	ret = add_root_props(upl, root);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("ad1", ret);
 | 
						|
	ret = ofnode_add_subnode(root, UPLN_OPTIONS, &options);
 | 
						|
	if (ret && ret != -EEXIST)
 | 
						|
		return log_msg_ret("opt", -EINVAL);
 | 
						|
 | 
						|
	ret = add_upl_params(upl, options);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("ad1", ret);
 | 
						|
 | 
						|
	ret = add_upl_image(upl, options);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("ad2", ret);
 | 
						|
 | 
						|
	ret = add_upl_memory(upl, root);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("ad3", ret);
 | 
						|
 | 
						|
	ret = add_upl_memmap(upl, root);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("ad4", ret);
 | 
						|
 | 
						|
	ret = add_upl_memres(upl, root, skip_existing);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("ad5", ret);
 | 
						|
 | 
						|
	ret = add_upl_serial(upl, root, skip_existing);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("ad6", ret);
 | 
						|
 | 
						|
	ret = add_upl_graphics(upl, root);
 | 
						|
	if (ret && ret != -ENOENT)
 | 
						|
		return log_msg_ret("ad6", ret);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int upl_create_handoff_tree(const struct upl *upl, oftree *treep)
 | 
						|
{
 | 
						|
	ofnode root;
 | 
						|
	oftree tree;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	ret = oftree_new(&tree);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("new", ret);
 | 
						|
 | 
						|
	root = oftree_root(tree);
 | 
						|
	if (!ofnode_valid(root))
 | 
						|
		return log_msg_ret("roo", -EINVAL);
 | 
						|
 | 
						|
	ret = upl_write_handoff(upl, root, false);
 | 
						|
	if (ret)
 | 
						|
		return log_msg_ret("wr", ret);
 | 
						|
 | 
						|
	*treep = tree;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 |