mirror of
https://github.com/u-boot/u-boot.git
synced 2025-04-11 15:34:55 +00:00

Use the new symbol to refer to any 'SPL' build, including TPL and VPL Signed-off-by: Simon Glass <sjg@chromium.org>
382 lines
9.1 KiB
C
382 lines
9.1 KiB
C
/* SPDX-License-Identifier: GPL-2.0+ */
|
|
/*
|
|
* UPL handoff generation
|
|
*
|
|
* Copyright 2024 Google LLC
|
|
* Written by Simon Glass <sjg@chromium.org>
|
|
*/
|
|
|
|
#ifndef __UPL_WRITE_H
|
|
#define __UPL_WRITE_H
|
|
|
|
#ifndef USE_HOSTCC
|
|
|
|
#include <alist.h>
|
|
#include <image.h>
|
|
#include <dm/ofnode_decl.h>
|
|
|
|
struct unit_test_state;
|
|
|
|
#define UPLP_ADDRESS_CELLS "#address-cells"
|
|
#define UPLP_SIZE_CELLS "#size-cells"
|
|
|
|
#define UPLN_OPTIONS "options"
|
|
#define UPLN_UPL_PARAMS "upl-params"
|
|
#define UPLP_SMBIOS "smbios"
|
|
#define UPLP_ACPI "acpi"
|
|
#define UPLP_BOOTMODE "bootmode"
|
|
#define UPLP_ADDR_WIDTH "addr-width"
|
|
#define UPLP_ACPI_NVS_SIZE "acpi-nvs-size"
|
|
|
|
#define UPLPATH_UPL_IMAGE "/options/upl-image"
|
|
#define UPLN_UPL_IMAGE "upl-image"
|
|
#define UPLN_IMAGE "image"
|
|
#define UPLP_FIT "fit"
|
|
#define UPLP_CONF_OFFSET "conf-offset"
|
|
#define UPLP_LOAD "load"
|
|
#define UPLP_SIZE "size"
|
|
#define UPLP_OFFSET "offset"
|
|
#define UPLP_DESCRIPTION "description"
|
|
|
|
#define UPLN_MEMORY "memory"
|
|
#define UPLP_HOTPLUGGABLE "hotpluggable"
|
|
|
|
#define UPLPATH_MEMORY_MAP "/memory-map"
|
|
#define UPLN_MEMORY_MAP "memory-map"
|
|
#define UPLP_USAGE "usage"
|
|
|
|
#define UPLN_MEMORY_RESERVED "reserved-memory"
|
|
#define UPLPATH_MEMORY_RESERVED "/reserved-memory"
|
|
#define UPLP_NO_MAP "no-map"
|
|
|
|
#define UPLN_SERIAL "serial"
|
|
#define UPLP_REG "reg"
|
|
#define UPLP_COMPATIBLE "compatible"
|
|
#define UPLP_CLOCK_FREQUENCY "clock-frequency"
|
|
#define UPLP_CURRENT_SPEED "current-speed"
|
|
#define UPLP_REG_IO_SHIFT "reg-io-shift"
|
|
#define UPLP_REG_OFFSET "reg-offset"
|
|
#define UPLP_REG_IO_WIDTH "reg-io-width"
|
|
#define UPLP_VIRTUAL_REG "virtual-reg"
|
|
#define UPLP_ACCESS_TYPE "access-type"
|
|
|
|
#define UPLN_GRAPHICS "framebuffer"
|
|
#define UPLC_GRAPHICS "simple-framebuffer"
|
|
#define UPLP_WIDTH "width"
|
|
#define UPLP_HEIGHT "height"
|
|
#define UPLP_STRIDE "stride"
|
|
#define UPLP_GRAPHICS_FORMAT "format"
|
|
|
|
/**
|
|
* enum upl_boot_mode - Encodes the boot mode
|
|
*
|
|
* Each is a bit number from the boot_mode mask
|
|
*/
|
|
enum upl_boot_mode {
|
|
UPLBM_FULL,
|
|
UPLBM_MINIMAL,
|
|
UPLBM_FAST,
|
|
UPLBM_DIAG,
|
|
UPLBM_DEFAULT,
|
|
UPLBM_S2,
|
|
UPLBM_S3,
|
|
UPLBM_S4,
|
|
UPLBM_S5,
|
|
UPLBM_FACTORY,
|
|
UPLBM_FLASH,
|
|
UPLBM_RECOVERY,
|
|
|
|
UPLBM_COUNT,
|
|
};
|
|
|
|
/**
|
|
* struct upl_image - UPL image informaiton
|
|
*
|
|
* @load: Address image was loaded to
|
|
* @size: Size of image in bytes
|
|
* @offset: Offset of the image in the FIT (0=none)
|
|
* @desc: Description of the iamge (taken from the FIT)
|
|
*/
|
|
struct upl_image {
|
|
ulong load;
|
|
ulong size;
|
|
uint offset;
|
|
const char *description;
|
|
};
|
|
|
|
/**
|
|
* struct memregion - Information about a region of memory
|
|
*
|
|
* @base: Base address
|
|
* @size: Size in bytes
|
|
*/
|
|
struct memregion {
|
|
ulong base;
|
|
ulong size;
|
|
};
|
|
|
|
/**
|
|
* struct upl_mem - Information about physical-memory layout
|
|
*
|
|
* TODO: Figure out initial-mapped-area
|
|
*
|
|
* @region: Memory region list (struct memregion)
|
|
* @hotpluggable: true if hotpluggable
|
|
*/
|
|
struct upl_mem {
|
|
struct alist region;
|
|
bool hotpluggable;
|
|
};
|
|
|
|
/**
|
|
* enum upl_usage - Encodes the usage
|
|
*
|
|
* Each is a bit number from the usage mask
|
|
*/
|
|
enum upl_usage {
|
|
UPLUS_ACPI_RECLAIM,
|
|
UPLUS_ACPI_NVS,
|
|
UPLUS_BOOT_CODE,
|
|
UPLUS_BOOT_DATA,
|
|
UPLUS_RUNTIME_CODE,
|
|
UPLUS_RUNTIME_DATA,
|
|
UPLUS_COUNT
|
|
};
|
|
|
|
/**
|
|
* struct upl_memmap - Information about logical-memory layout
|
|
*
|
|
* @name: Node name to use
|
|
* @region: Memory region list (struct memregion)
|
|
* @usage: Memory-usage mask (enum upl_usage)
|
|
*/
|
|
struct upl_memmap {
|
|
const char *name;
|
|
struct alist region;
|
|
uint usage;
|
|
};
|
|
|
|
/**
|
|
* struct upl_memres - Reserved memory
|
|
*
|
|
* @name: Node name to use
|
|
* @region: Reserved memory region list (struct memregion)
|
|
* @no_map: true to indicate that a virtual mapping must not be created
|
|
*/
|
|
struct upl_memres {
|
|
const char *name;
|
|
struct alist region;
|
|
bool no_map;
|
|
};
|
|
|
|
enum upl_serial_access_type {
|
|
UPLSAT_MMIO,
|
|
UPLSAT_IO,
|
|
};
|
|
|
|
/* serial defaults */
|
|
enum {
|
|
UPLD_REG_IO_SHIFT = 0,
|
|
UPLD_REG_OFFSET = 0,
|
|
UPLD_REG_IO_WIDTH = 1,
|
|
};
|
|
|
|
/**
|
|
* enum upl_access_type - Access types
|
|
*
|
|
* @UPLAT_MMIO: Memory-mapped I/O
|
|
* @UPLAT_IO: Separate I/O
|
|
*/
|
|
enum upl_access_type {
|
|
UPLAT_MMIO,
|
|
UPLAT_IO,
|
|
};
|
|
|
|
/**
|
|
* struct upl_serial - Serial console
|
|
*
|
|
* @compatible: Compatible string (NULL if there is no serial console)
|
|
* @clock_frequency: Input clock frequency of UART
|
|
* @current_speed: Current baud rate of UART
|
|
* @reg: List of base address and size of registers (struct memregion)
|
|
* @reg_shift_log2: log2 of distance between each register
|
|
* @reg_offset: Offset of registers from the base address
|
|
* @reg_width: Register width in bytes
|
|
* @virtual_reg: Virtual register access (0 for none)
|
|
* @access_type: Register access type to use
|
|
*/
|
|
struct upl_serial {
|
|
const char *compatible;
|
|
uint clock_frequency;
|
|
uint current_speed;
|
|
struct alist reg;
|
|
uint reg_io_shift;
|
|
uint reg_offset;
|
|
uint reg_io_width;
|
|
ulong virtual_reg;
|
|
enum upl_serial_access_type access_type;
|
|
};
|
|
|
|
/**
|
|
* enum upl_graphics_format - Graphics formats
|
|
*
|
|
* @UPLGF_ARGB32: 32bpp format using 0xaarrggbb
|
|
* @UPLGF_ABGR32: 32bpp format using 0xaabbggrr
|
|
* @UPLGF_ARGB64: 64bpp format using 0xaaaabbbbggggrrrr
|
|
*/
|
|
enum upl_graphics_format {
|
|
UPLGF_ARGB32,
|
|
UPLGF_ABGR32,
|
|
UPLGF_ABGR64,
|
|
};
|
|
|
|
/**
|
|
* @reg: List of base address and size of registers (struct memregion)
|
|
* @width: Width of display in pixels
|
|
* @height: Height of display in pixels
|
|
* @stride: Number of bytes from one line to the next
|
|
* @format: Pixel format
|
|
*/
|
|
struct upl_graphics {
|
|
struct alist reg;
|
|
uint width;
|
|
uint height;
|
|
uint stride;
|
|
enum upl_graphics_format format;
|
|
};
|
|
|
|
/*
|
|
* Information about the UPL state
|
|
*
|
|
* @addr_cells: Number of address cells used in the handoff
|
|
* @size_cells: Number of size cells used in the handoff
|
|
* @bootmode: Boot-mode mask (enum upl_boot_mode)
|
|
* @fit: Address of FIT image that was loaded
|
|
* @conf_offset: Offset in FIT of the configuration that was selected
|
|
* @addr_width: Adress-bus width of machine, e.g. 46 for 46 bits
|
|
* @acpi_nvs_size: Size of the ACPI non-volatile-storage area in bytes
|
|
* @image: Information about each image (struct upl_image)
|
|
* @mem: Information about physical-memory regions (struct upl_mem)
|
|
* @nennap: Information about logical-memory regions (struct upl_memmap)
|
|
* @nennap: Information about reserved-memory regions (struct upl_memres)
|
|
*/
|
|
struct upl {
|
|
int addr_cells;
|
|
int size_cells;
|
|
|
|
ulong smbios;
|
|
ulong acpi;
|
|
uint bootmode;
|
|
ulong fit;
|
|
uint conf_offset;
|
|
uint addr_width;
|
|
uint acpi_nvs_size;
|
|
|
|
struct alist image;
|
|
struct alist mem;
|
|
struct alist memmap;
|
|
struct alist memres;
|
|
struct upl_serial serial;
|
|
struct upl_graphics graphics;
|
|
};
|
|
|
|
/**
|
|
* upl_write_handoff() - Write a Unversal Payload handoff structure
|
|
*
|
|
* upl: UPL state to write
|
|
* @root: root node to write it to
|
|
* @skip_existing: Avoid recreating any nodes which already exist in the
|
|
* devicetree. For example, if there is a serial node, just leave it alone,
|
|
* since don't need to create a new one
|
|
* Return: 0 on success, -ve on error
|
|
*/
|
|
int upl_write_handoff(const struct upl *upl, ofnode root, bool skip_existing);
|
|
|
|
/**
|
|
* upl_create_handoff_tree() - Write a Unversal Payload handoff structure
|
|
*
|
|
* upl: UPL state to write
|
|
* @treep: Returns a new tree containing the handoff
|
|
* Return: 0 on success, -ve on error
|
|
*/
|
|
int upl_create_handoff_tree(const struct upl *upl, oftree *treep);
|
|
|
|
/**
|
|
* upl_read_handoff() - Read a Unversal Payload handoff structure
|
|
*
|
|
* upl: UPL state to read into
|
|
* @tree: Devicetree containing the data to read
|
|
* Return: 0 on success, -ve on error
|
|
*/
|
|
int upl_read_handoff(struct upl *upl, oftree tree);
|
|
|
|
/**
|
|
* upl_get_test_data() - Fill a UPL with some test data
|
|
*
|
|
* @uts: Test state (can be uninited)
|
|
* @upl: Returns test data
|
|
* Return: 0 on success, 1 on error
|
|
*/
|
|
int upl_get_test_data(struct unit_test_state *uts, struct upl *upl);
|
|
#endif /* USE_HOSTCC */
|
|
|
|
#if CONFIG_IS_ENABLED(UPL) && defined(CONFIG_XPL_BUILD)
|
|
|
|
/**
|
|
* upl_set_fit_info() - Set up basic info about the FIT
|
|
*
|
|
* @fit: Address of FIT
|
|
* @conf_offset: Configuration node being used
|
|
* @entry_addr: Entry address for next phase
|
|
*/
|
|
void upl_set_fit_info(ulong fit, int conf_offset, ulong entry_addr);
|
|
|
|
/**
|
|
* upl_set_fit_addr() - Set up the address of the FIT
|
|
*
|
|
* @fit: Address of FIT
|
|
*/
|
|
void upl_set_fit_addr(ulong fit);
|
|
|
|
#else
|
|
static inline void upl_set_fit_addr(ulong fit) {}
|
|
static inline void upl_set_fit_info(ulong fit, int conf_offset,
|
|
ulong entry_addr) {}
|
|
#endif /* UPL && SPL */
|
|
|
|
/**
|
|
* _upl_add_image() - Internal function to add a new image to the UPL
|
|
*
|
|
* @node: Image node offset in FIT
|
|
* @load_addr: Address to which images was loaded
|
|
* @size: Image size in bytes
|
|
* @desc: Description of image
|
|
* Return: 0 if OK, -ENOMEM if out of memory
|
|
*/
|
|
int _upl_add_image(int node, ulong load_addr, ulong size, const char *desc);
|
|
|
|
/**
|
|
* upl_add_image() - Add a new image to the UPL
|
|
*
|
|
* @fit: Pointer to FIT
|
|
* @node: Image node offset in FIT
|
|
* @load_addr: Address to which images was loaded
|
|
* @size: Image size in bytes
|
|
* Return: 0 if OK, -ENOMEM if out of memory
|
|
*/
|
|
static inline int upl_add_image(const void *fit, int node, ulong load_addr,
|
|
ulong size)
|
|
{
|
|
if (CONFIG_IS_ENABLED(UPL) && IS_ENABLED(CONFIG_XPL_BUILD)) {
|
|
const char *desc = fdt_getprop(fit, node, FIT_DESC_PROP, NULL);
|
|
|
|
return _upl_add_image(node, load_addr, size, desc);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/** upl_init() - Set up a UPL struct */
|
|
void upl_init(struct upl *upl);
|
|
|
|
#endif /* __UPL_WRITE_H */
|