Hi Simon, On Mon, Oct 21, 2019 at 11:40 AM Simon Glass <[email protected]> wrote: > > Add support for some important configuration options and FSP memory init. > The memory init uses swizzle tables from the device tree. > > Support for the FSP_S binary is also included. > > Bootstage timing is used for both FSP_M and FSP_M and memory-mapped SPI
FSP_T and FSP_M ? > reads. > > Signed-off-by: Simon Glass <[email protected]> > --- > > Changes in v3: > - Add a proper implementation of fsp_notify > - Add an fsp: tag > - Add bootstage timing for memory-mapped reads > - Add fsp_locate_fsp to locate an fsp component > - Add fspm_done() hook > - Add support for FSP-S component and VBT > - Simplify types for fsp_locate_fsp() > - Switch mmap to use SPI instead of SPI flash > > Changes in v2: None > > arch/x86/Kconfig | 61 ++++++++- > arch/x86/include/asm/fsp2/fsp_api.h | 60 +++++++++ > arch/x86/include/asm/fsp2/fsp_internal.h | 97 ++++++++++++++ > arch/x86/lib/fsp2/Makefile | 10 ++ > arch/x86/lib/fsp2/fsp_common.c | 13 ++ > arch/x86/lib/fsp2/fsp_dram.c | 77 +++++++++++ > arch/x86/lib/fsp2/fsp_init.c | 157 +++++++++++++++++++++++ > arch/x86/lib/fsp2/fsp_meminit.c | 97 ++++++++++++++ > arch/x86/lib/fsp2/fsp_silicon_init.c | 52 ++++++++ > arch/x86/lib/fsp2/fsp_support.c | 129 +++++++++++++++++++ > include/bootstage.h | 3 + > 11 files changed, 754 insertions(+), 2 deletions(-) > create mode 100644 arch/x86/include/asm/fsp2/fsp_api.h > create mode 100644 arch/x86/include/asm/fsp2/fsp_internal.h > create mode 100644 arch/x86/lib/fsp2/Makefile > create mode 100644 arch/x86/lib/fsp2/fsp_common.c > create mode 100644 arch/x86/lib/fsp2/fsp_dram.c > create mode 100644 arch/x86/lib/fsp2/fsp_init.c > create mode 100644 arch/x86/lib/fsp2/fsp_meminit.c > create mode 100644 arch/x86/lib/fsp2/fsp_silicon_init.c > create mode 100644 arch/x86/lib/fsp2/fsp_support.c > > diff --git a/arch/x86/Kconfig b/arch/x86/Kconfig > index 17a6fe6d3d9..cbd3fc4f581 100644 > --- a/arch/x86/Kconfig > +++ b/arch/x86/Kconfig > @@ -326,7 +326,7 @@ config X86_RAMTEST > > config FLASH_DESCRIPTOR_FILE > string "Flash descriptor binary filename" > - depends on HAVE_INTEL_ME > + depends on HAVE_INTEL_ME || FSP_VERSION2 > default "descriptor.bin" > help > The filename of the file to use as flash descriptor in the > @@ -411,6 +411,63 @@ config FSP_ADDR > The default base address of 0xfffc0000 indicates that the binary > must > be located at offset 0xc0000 from the beginning of a 1MB flash > device. > > +if FSP_VERSION2 > + > +config FSP_FILE_T > + string "Firmware-Support-Package binary filename (Temp RAM)" > + default "fsp_t.bin" > + help > + The filename of the file to use for the temporary-RAM init phase > from > + the Firmware-Support-Package binary. Put this in the board > directory. > + It is used to set up an initial area of RAM which can be used for > the > + stack and other purposes, while bringing up the main system DRAM. > + > +config FSP_ADDR_T > + hex "Firmware-Support-Package binary location (Temp RAM)" > + default 0xffff8000 > + help > + FSP is not Position-Independent Code (PIC) and FSP components have > to > + be rebased if placed at a location which is different from the > + perferred base address specified during the FSP build. Use Intel's > + Binary Configuration Tool (BCT) to do the rebase. > + > +config FSP_FILE_M > + string "Firmware-Support-Package binary filename (Memory Init)" > + default "fsp_m.bin" > + help > + The filename of the file to use for the RAM init phase from the > + Firmware Support Package binary. Put this in the board directory. > + It is used to set up the main system DRAM and runs in SPL, once > + temporary RAM (CAR) is working. > + > +config FSP_FILE_S > + string "Firmware-Support-Package binary filename (Silicon Init)" > + default "fsp_s.bin" > + help > + The filename of the file to use for the Silicon init phase from the > + Firmware Support Package binary. Put this in the board directory. > + It is used to set up the silicon to work correctly and must be > + executed after DRAM is running. > + > +config FSP_FILE_VBT There is already a VBT_FILE config option for this. > + string "Firmware-Support-Package Video BIOS Table (VBT)" > + default "vbt.bin" > + help > + The filename of the file to use for the video data needd by the > + Silicon init phase from the Firmware Support Package binary. Put > this > + in the board directory. It is used to set up the video parameters so > + that the display can be used. > + > +config IFWI_INPUT_FILE > + string "Filename containing FIT (Firmware Interface Table) with IFWI" > + default "fitimage.bin" > + help > + The IFWI is obtained by running a tool on this file to extract the > + IFWI. Put this in the board directory. The IFWI contains U-Boot TPL, > + microcode and other internal items. > + > +endif > + > config FSP_TEMP_RAM_ADDR > hex > depends on FSP_VERSION1 > @@ -629,7 +686,7 @@ config VBT_ADDR > > config VIDEO_FSP > bool "Enable FSP framebuffer driver support" > - depends on HAVE_VBT && DM_VIDEO > + depends on (HAVE_VBT || FSP_VERSION2) && DM_VIDEO > help > Turn on this option to enable a framebuffer driver when U-Boot is > using Video BIOS Table (VBT) image for FSP firmware to initialize > diff --git a/arch/x86/include/asm/fsp2/fsp_api.h > b/arch/x86/include/asm/fsp2/fsp_api.h > new file mode 100644 > index 00000000000..93b6472dce0 > --- /dev/null > +++ b/arch/x86/include/asm/fsp2/fsp_api.h > @@ -0,0 +1,60 @@ > +/* SPDX-License-Identifier: Intel */ > +/* > + * Copyright (C) 2015-2016 Intel Corp. > + * (Written by Andrey Petrov <[email protected]> for Intel Corp.) > + * (Written by Alexandru Gagniuc <[email protected]> for Intel > Corp.) > + * Mostly taken from coreboot fsp2_0/memory_init.c > + */ > + > +#ifndef __ASM_FSP2_API_H > +#define __ASM_FSP2_API_H > + > +#include <asm/fsp/fsp_api.h> > + > +struct fspm_upd; > +struct fsps_upd; > +struct hob_header; > + > +enum fsp_boot_mode { > + FSP_BOOT_WITH_FULL_CONFIGURATION = 0x00, > + FSP_BOOT_WITH_MINIMAL_CONFIGURATION = 0x01, > + FSP_BOOT_ASSUMING_NO_CONFIGURATION_CHANGES = 0x02, > + FSP_BOOT_ON_S4_RESUME = 0x05, > + FSP_BOOT_ON_S3_RESUME = 0x11, > + FSP_BOOT_ON_FLASH_UPDATE = 0x12, > + FSP_BOOT_IN_RECOVERY_MODE = 0x20 > +}; > + > +struct __packed fsp_upd_header { > + u64 signature; > + u8 revision; > + u8 reserved[23]; > +}; > + > +/** > + * fsp_memory_init() - Init the SDRAM > + * > + * @s3wake: true if we are booting from resume, so cannot reinit the mememory > + * from scatch since we will lose its contents > + * @use_spi_flash: true to use the fast SPI driver to read FSP, otherwise use > + * mapped SPI > + * @return 0 if OK, -ve on error > + */ > +int fsp_memory_init(bool s3wake, bool use_spi_flash); > + > +typedef asmlinkage int (*fsp_memory_init_func)(struct fspm_upd *params, > + struct hob_header **hobp); > + > +/** > + * fsp_silicon_init() - Init the silicon > + * > + * This calls the FSP's 'silicon init' entry point > + * > + * @s3wake: true if we are booting from resume Should have the same descriptions as fsp_memory_init() > + * @return 0 if OK, -ve on error > + */ > +int fsp_silicon_init(bool s3wake, bool use_spi_flash); > + > +typedef asmlinkage int (*fsp_silicon_init_func)(struct fsps_upd *params); > + > +#endif > diff --git a/arch/x86/include/asm/fsp2/fsp_internal.h > b/arch/x86/include/asm/fsp2/fsp_internal.h > new file mode 100644 > index 00000000000..5893f1ffcc7 > --- /dev/null > +++ b/arch/x86/include/asm/fsp2/fsp_internal.h > @@ -0,0 +1,97 @@ > +/* SPDX-License-Identifier: Intel */ > +/* > + * Copyright (C) 2015-2016 Intel Corp. > + * (Written by Alexandru Gagniuc <[email protected]> for Intel > Corp.) > + * Mostly taken from coreboot > + */ > + > +#ifndef __ASM_FSP_INTERNAL_H > +#define __ASM_FSP_INTERNAL_H > + > +struct binman_entry; > +struct fsp_header; > +struct fspm_upd; > +struct fsps_upd; > + > +enum fsp_type_t { > + FSP_M, > + FSP_S, > +}; > + > +int fsp_get_header(ulong offset, ulong size, bool use_spi_flash, > + struct fsp_header **fspp); > + > +/** > + * fsp_locate_fsp() - Locate an FSP component > + * > + * This finds an FSP component by various methods. It is not as > general-purpose > + * as it looks, since it expects FSP-M to be requested in SPL (only), and > FSP-S > + * to be requested in U-Boot proper. > + * > + * @type: Component to locate > + * @entry: Returns location of component > + * @use_spi_flash: true to read using the Fast SPI driver, false to use > + * memory-mapped SPI flash > + * @devp: Returns northbridge device > + * @hdrp: Returns FSP header > + * @rom_offsetp: If non-NULL, returns the offset to add to any image > position to > + * find the memory-mapped location of that position. For example, for ROM > + * position 0x1000, it will be mapped into 0x1000 + *rom_offsetp. > + */ > +int fsp_locate_fsp(enum fsp_type_t type, struct binman_entry *entry, > + bool use_spi_flash, struct udevice **devp, > + struct fsp_header **hdrp, ulong *rom_offsetp); > + > +/** > + * arch_fsp_s_preinit() - Perform init needed before calling FSP-S Considering other routines in this file are having fspm_ as the prefix, should we name this function to arch_fsps_preinit()? Or rename others to fsp_m_ as the prefix? > + * > + * This allows use of probed drivers and PCI so is a convenient place to do > any > + * init that is needed before FSP-S is called. After this, U-Boot relocates > and > + * calls arch_fsp_init_r() before PCI is probed, and that function is not > + * allowed to probe PCI before calling FSP-S. > + */ > +int arch_fsp_s_preinit(void); > + > +/** > + * fspm_update_config() - Set up the config structure for FSP-M > + * > + * @dev: Hostbridge device containing config > + * @upd: Config data to fill in > + * @return 0 if OK, -ve on error > + */ > +int fspm_update_config(struct udevice *dev, struct fspm_upd *upd); > + > +/** > + * fspm_done() - Indicate that memory init is complete > + * > + * This allows the board to do whatever post-init it needs before things > + * continue. > + * > + * @dev: Hostbridge device > + * @return 0 if OK, -ve on error > + */ > +int fspm_done(struct udevice *dev); > + > +/** > + * fsps_update_config() - Set up the config structure for FSP-S > + * > + * @dev: Hostbridge device containing config > + * @rom_offset: Value to add to convert from ROM offset to memory-mapped > address > + * @upd: Config data to fill in > + * @return 0 if OK, -ve on error > + */ > +int fsps_update_config(struct udevice *dev, ulong rom_offset, > + struct fsps_upd *upd); > + > +/** > + * prepare_mrc_cache() - Read the MRC cache into the product-data struct > + * > + * This looks for cached Memory-reference code (MRC) data and stores it into > + * @upd for use by the FSP-M binary. > + * > + * @return 0 if OK, -ENOENT if no data (whereupon the caller can continue and > + * expect a slower boot), other -ve value on other error > + */ > +int prepare_mrc_cache(struct fspm_upd *upd); > + > +#endif > diff --git a/arch/x86/lib/fsp2/Makefile b/arch/x86/lib/fsp2/Makefile > new file mode 100644 > index 00000000000..ddbe2d0db26 > --- /dev/null > +++ b/arch/x86/lib/fsp2/Makefile > @@ -0,0 +1,10 @@ > +# SPDX-License-Identifier: GPL-2.0+ > +# > +# Copyright 2019 Google LLC > + > +obj-y += fsp_common.o > +obj-y += fsp_dram.o > +obj-y += fsp_init.o > +obj-y += fsp_meminit.o > +obj-y += fsp_silicon_init.o > +obj-y += fsp_support.o > diff --git a/arch/x86/lib/fsp2/fsp_common.c b/arch/x86/lib/fsp2/fsp_common.c > new file mode 100644 > index 00000000000..f69456e43a2 > --- /dev/null > +++ b/arch/x86/lib/fsp2/fsp_common.c > @@ -0,0 +1,13 @@ > +// SPDX-License-Identifier: GPL-2.0+ > +/* > + * Copyright 2019 Google LLC > + * Written by Simon Glass <[email protected]> > + */ > + > +#include <common.h> > +#include <init.h> > + > +int arch_fsp_init(void) > +{ > + return 0; > +} > diff --git a/arch/x86/lib/fsp2/fsp_dram.c b/arch/x86/lib/fsp2/fsp_dram.c > new file mode 100644 > index 00000000000..a3f95d0cf96 > --- /dev/null > +++ b/arch/x86/lib/fsp2/fsp_dram.c > @@ -0,0 +1,77 @@ > +// SPDX-License-Identifier: GPL-2.0+ > +/* > + * Copyright 2019 Google LLC > + * Written by Simon Glass <[email protected]> > + */ > + > +#include <common.h> > +#include <acpi_s3.h> > +#include <handoff.h> > +#include <spl.h> > +#include <asm/arch/cpu.h> > +#include <asm/fsp/fsp_support.h> > +#include <asm/fsp2/fsp_api.h> > +#include <asm/fsp2/fsp_internal.h> > + > +int dram_init(void) > +{ > + int ret; > + > + if (spl_phase() == PHASE_SPL) { > +#ifdef CONFIG_HAVE_ACPI_RESUME > + bool s3wake = gd->arch.prev_sleep_state == ACPI_S3; > +#else > + bool s3wake = false; > +#endif > + > + ret = fsp_memory_init(s3wake, BOOT_FROM_FAST_SPI_FLASH); > + if (ret) { > + debug("Memory init failed (err=%x)\n", ret); > + return ret; > + } > + > + /* The FSP has already set up DRAM, so grab the info we need > */ > + ret = fsp_scan_for_ram_size(); > + if (ret) > + return ret; > + > +#ifdef CONFIG_ENABLE_MRC_CACHE > + gd->arch.mrc[MRC_TYPE_NORMAL].buf = > + fsp_get_nvs_data(gd->arch.hob_list, > + &gd->arch.mrc[MRC_TYPE_NORMAL].len); > + gd->arch.mrc[MRC_TYPE_VAR].buf = > + fsp_get_var_nvs_data(gd->arch.hob_list, > + &gd->arch.mrc[MRC_TYPE_VAR].len); > + log_debug("normal %x, var %x\n", > + gd->arch.mrc[MRC_TYPE_NORMAL].len, > + gd->arch.mrc[MRC_TYPE_VAR].len); > +#endif > + } else { > +#if CONFIG_IS_ENABLED(HANDOFF) > + struct spl_handoff *ho = gd->spl_handoff; > + > + if (!ho) { > + debug("No SPL handoff found\n"); > + return -ESTRPIPE; > + } > + gd->ram_size = ho->ram_size; > + handoff_load_dram_banks(ho); > +#endif > + ret = arch_fsp_s_preinit(); > + if (ret) > + return log_msg_ret("fsp_s_preinit", ret); > + } > + > + return 0; > +} > + > +ulong board_get_usable_ram_top(ulong total_size) > +{ > +#if CONFIG_IS_ENABLED(HANDOFF) > + struct spl_handoff *ho = gd->spl_handoff; > + > + return ho->arch.usable_ram_top; > +#endif > + > + return gd->ram_top; > +} > diff --git a/arch/x86/lib/fsp2/fsp_init.c b/arch/x86/lib/fsp2/fsp_init.c > new file mode 100644 > index 00000000000..111eb51d34b > --- /dev/null > +++ b/arch/x86/lib/fsp2/fsp_init.c > @@ -0,0 +1,157 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * Copyright 2019 Google LLC > + */ > + > +#include <common.h> > +#include <binman.h> > +#include <binman_sym.h> > +#include <cbfs.h> > +#include <dm.h> > +#include <init.h> > +#include <spi.h> > +#include <spl.h> > +#include <spi_flash.h> > +#include <asm/arch/gpio.h> > +#include <dm/uclass-internal.h> > +#include <asm/fsp2/fsp_internal.h> > + > +int arch_cpu_init_dm(void) > +{ > + struct udevice *dev; > + ofnode node; > + int ret; > + > + if (spl_phase() != PHASE_BOARD_F) > + return 0; > + > + /* Probe all GPIO devices to set up the pads */ > + ret = uclass_first_device_err(UCLASS_GPIO, &dev); Why is this GPIO probe needed in the generic FSP support? > + if (ret) > + return log_msg_ret("no fsp GPIO", ret); > + node = ofnode_path("fsp"); > + if (!ofnode_valid(node)) > + return log_msg_ret("no fsp params", -EINVAL); > + ret = hostbridge_config_pads_for_node(dev, node); > + if (ret) > + return log_msg_ret("pad config", ret); > + > + return ret; > +} > + > +#if !defined(CONFIG_TPL_BUILD) > +binman_sym_declare(ulong, intel_fsp_m, image_pos); > +binman_sym_declare(ulong, intel_fsp_m, size); > + > +static int get_coreboot_fsp(enum fsp_type_t type, ulong map_base, get_uboot_fsp? > + struct binman_entry *entry) > +{ > + /* Hard-coded position of CBFS in ROM */ Why hard-coded? > + ulong cbfs_base = 0x205000; > + ulong cbfs_size = 0x1bb000; > + struct cbfs_priv *cbfs; > + int ret; > + > + ret = cbfs_init_mem(map_base + cbfs_base, cbfs_size, &cbfs); > + if (ret) > + return ret; > + if (!ret) { > + const struct cbfs_cachenode *node; > + > + node = cbfs_find_file(cbfs, "fspm.bin"); > + if (!node) > + return log_msg_ret("fspm node", -ENOENT); > + > + entry->image_pos = (ulong)node->data; > + entry->size = node->data_length; > + } > + > + return 0; > +} > + > +int fsp_locate_fsp(enum fsp_type_t type, struct binman_entry *entry, > + bool use_spi_flash, struct udevice **devp, > + struct fsp_header **hdrp, ulong *rom_offsetp) > +{ > + ulong mask = CONFIG_ROM_SIZE - 1; > + struct udevice *dev; > + ulong rom_offset = 0; > + uint map_size; > + ulong map_base; > + uint offset; > + int ret; > + > + /* > + * Find the devices but don't probe them, since we don't want to > + * auto-config PCI before silicon init runs > + */ > + ret = uclass_find_first_device(UCLASS_NORTHBRIDGE, &dev); > + if (ret) > + return log_msg_ret("Cannot get northbridge", ret); > + if (!use_spi_flash) { > + struct udevice *sf; > + > + /* Just use the SPI driver to get the memory map */ > + ret = uclass_find_first_device(UCLASS_SPI_FLASH, &sf); > + if (ret) > + return log_msg_ret("Cannot get SPI flash", ret); > + ret = dm_spi_get_mmap(sf, &map_base, &map_size, &offset); > + if (ret) > + return log_msg_ret("Could not get flash mmap", ret); > + } > + > + if (spl_phase() >= PHASE_BOARD_F) { > + if (type != FSP_S) > + return -EPROTONOSUPPORT; > + ret = binman_entry_find("intel-fsp-s", entry); > + if (ret) > + return log_msg_ret("binman entry", ret); > + if (!use_spi_flash) > + rom_offset = (map_base & mask) - CONFIG_ROM_SIZE; > + } else { > + ret = -ENOENT; > + if (false) > + /* Support using a hybrid image build by coreboot */ U-Boot? > + ret = get_coreboot_fsp(type, map_base, entry); > + if (ret) { > + ulong mask = CONFIG_ROM_SIZE - 1; > + > + if (type != FSP_M) > + return -EPROTONOSUPPORT; > + entry->image_pos = binman_sym(ulong, intel_fsp_m, > + image_pos); > + entry->size = binman_sym(ulong, intel_fsp_m, size); > + if (entry->image_pos != BINMAN_SYM_MISSING) { > + ret = 0; > + if (use_spi_flash) > + entry->image_pos &= mask; > + else > + entry->image_pos += (map_base & mask); > + } else { > + ret = -ENOENT; > + } > + } > + } > + if (ret) > + return log_msg_ret("Cannot find FSP", ret); > + entry->image_pos += rom_offset; > + > + /* > + * Account for the time taken to read memory-mapped SPI flash since in > + * this case we don't use the SPI driver and BOOTSTAGE_ID_ACCUM_SPI. > + */ > + if (!use_spi_flash) > + bootstage_start(BOOTSTAGE_ID_ACCUM_MMAP_SPI, "mmap_spi"); > + ret = fsp_get_header(entry->image_pos, entry->size, use_spi_flash, > + hdrp); > + if (!use_spi_flash) > + bootstage_accum(BOOTSTAGE_ID_ACCUM_MMAP_SPI); > + if (ret) > + return log_msg_ret("fsp_get_header", ret); > + *devp = dev; > + if (rom_offsetp) > + *rom_offsetp = rom_offset; > + > + return 0; > +} > +#endif > diff --git a/arch/x86/lib/fsp2/fsp_meminit.c b/arch/x86/lib/fsp2/fsp_meminit.c > new file mode 100644 > index 00000000000..bf30c479899 > --- /dev/null > +++ b/arch/x86/lib/fsp2/fsp_meminit.c > @@ -0,0 +1,97 @@ > +// SPDX-License-Identifier: Intel > +/* > + * Copyright (C) 2015-2016 Intel Corp. > + * (Written by Andrey Petrov <[email protected]> for Intel Corp.) > + * (Written by Alexandru Gagniuc <[email protected]> for Intel > Corp.) > + * Mostly taken from coreboot fsp2_0/memory_init.c > + */ > + > +#include <common.h> > +#include <binman.h> > +#include <asm/mrccache.h> > +#include <asm/fsp/fsp_infoheader.h> > +#include <asm/fsp2/fsp_api.h> > +#include <asm/fsp2/fsp_internal.h> > +#include <asm/arch/fsp/fsp_configs.h> > +#include <asm/arch/fsp/fsp_m_upd.h> > + > +static int prepare_mrc_cache_type(enum mrc_type_t type, > + struct mrc_data_container **cachep) > +{ > + struct mrc_data_container *cache; > + struct mrc_region entry; > + int ret; > + > + ret = mrccache_get_region(type, NULL, &entry); > + if (ret) > + return ret; > + cache = mrccache_find_current(&entry); > + if (!cache) > + return -ENOENT; > + > + log_debug("MRC at %x, size %x\n", (uint)cache->data, > cache->data_size); > + *cachep = cache; > + > + return 0; > +} > + > +int prepare_mrc_cache(struct fspm_upd *upd) > +{ > + struct mrc_data_container *cache; > + int ret; > + > + ret = prepare_mrc_cache_type(MRC_TYPE_NORMAL, &cache); > + if (ret) > + return log_msg_ret("Cannot get normal cache", ret); > + upd->arch.nvs_buffer_ptr = cache->data; > + > + ret = prepare_mrc_cache_type(MRC_TYPE_VAR, &cache); > + if (ret) > + return log_msg_ret("Cannot get var cache", ret); > + upd->config.variable_nvs_buffer_ptr = cache->data; > + > + return 0; > +} > + > +int fsp_memory_init(bool s3wake, bool use_spi_flash) > +{ > + struct fspm_upd upd, *fsp_upd; > + fsp_memory_init_func func; > + struct binman_entry entry; > + struct fsp_header *hdr; > + struct hob_header *hob; > + struct udevice *dev; > + int ret; > + > + ret = fsp_locate_fsp(FSP_M, &entry, use_spi_flash, &dev, &hdr, NULL); > + if (ret) > + return log_msg_ret("locate FSP", ret); > + debug("Found FSP_M at %x, size %x\n", hdr->img_base, hdr->img_size); > + > + /* Copy over the default config */ > + fsp_upd = (struct fspm_upd *)(hdr->img_base + hdr->cfg_region_off); > + if (fsp_upd->header.signature != FSPM_UPD_SIGNATURE) > + return log_msg_ret("Bad UPD signature", -EPERM); > + memcpy(&upd, fsp_upd, sizeof(upd)); > + > + ret = fspm_update_config(dev, &upd); > + if (ret) > + return log_msg_ret("Could not setup config", ret); > + > + debug("SDRAM init..."); > + bootstage_start(BOOTSTATE_ID_ACCUM_FSP_M, "fsp-m"); > + func = (fsp_memory_init_func)(hdr->img_base + hdr->fsp_mem_init); > + ret = func(&upd, &hob); > + bootstage_accum(BOOTSTATE_ID_ACCUM_FSP_M); > + if (ret) > + return log_msg_ret("SDRAM init fail\n", ret); > + > + gd->arch.hob_list = hob; > + debug("done\n"); > + > + ret = fspm_done(dev); > + if (ret) > + return log_msg_ret("fsm_done\n", ret); > + > + return 0; > +} > diff --git a/arch/x86/lib/fsp2/fsp_silicon_init.c > b/arch/x86/lib/fsp2/fsp_silicon_init.c > new file mode 100644 > index 00000000000..c758e80c383 > --- /dev/null > +++ b/arch/x86/lib/fsp2/fsp_silicon_init.c > @@ -0,0 +1,52 @@ > +// SPDX-License-Identifier: Intel > +/* > + * Copyright (C) 2015-2016 Intel Corp. > + * (Written by Andrey Petrov <[email protected]> for Intel Corp.) > + * > + * Mostly taken from coreboot fsp2_0/silicon_init.c > + */ > + > +#include <common.h> > +#include <binman.h> > +#include <dm.h> > +#include <asm/arch/fsp/fsp_configs.h> > +#include <asm/arch/fsp/fsp_s_upd.h> > +#include <asm/fsp/fsp_infoheader.h> > +#include <asm/fsp2/fsp_internal.h> > + > +int fsp_silicon_init(bool s3wake, bool use_spi_flash) > +{ > + struct fsps_upd upd, *fsp_upd; > + fsp_silicon_init_func func; > + struct fsp_header *hdr; > + struct binman_entry entry; > + struct udevice *dev; > + ulong rom_offset = 0; > + int ret; > + > + ret = fsp_locate_fsp(FSP_S, &entry, use_spi_flash, &dev, &hdr, > + &rom_offset); > + if (ret) > + return log_msg_ret("locate FSP", ret); > + gd->arch.fsp_s_hdr = hdr; > + > + /* Copy over the default config */ > + fsp_upd = (struct fsps_upd *)(hdr->img_base + hdr->cfg_region_off); > + if (fsp_upd->header.signature != FSPS_UPD_SIGNATURE) > + return log_msg_ret("Bad UPD signature", -EPERM); > + memcpy(&upd, fsp_upd, sizeof(upd)); > + > + ret = fsps_update_config(dev, rom_offset, &upd); > + if (ret) > + return log_msg_ret("Could not setup config", ret); > + log_debug("Silicon init..."); > + bootstage_start(BOOTSTATE_ID_ACCUM_FSP_S, "fsp-s"); > + func = (fsp_silicon_init_func)(hdr->img_base + hdr->fsp_silicon_init); > + ret = func(&upd); > + bootstage_accum(BOOTSTATE_ID_ACCUM_FSP_S); > + if (ret) > + return log_msg_ret("Silicon init fail\n", ret); > + log_debug("done\n"); > + > + return 0; > +} > diff --git a/arch/x86/lib/fsp2/fsp_support.c b/arch/x86/lib/fsp2/fsp_support.c > new file mode 100644 > index 00000000000..477eaa268db > --- /dev/null > +++ b/arch/x86/lib/fsp2/fsp_support.c > @@ -0,0 +1,129 @@ > +// SPDX-License-Identifier: Intel > +/* > + * Copyright 2019 Google LLC > + * Written by Simon Glass <[email protected]> > + */ > + > +#include <common.h> > +#include <dm.h> > +#include <spi_flash.h> > +#include <asm/fsp/fsp_support.h> > +#include <asm/fsp2/fsp_internal.h> > + > +/* The amount of the FSP header to probe to obtain what we need */ > +#define PROBE_BUF_SIZE 0x180 > + > +/* Not needed in SPL apparently */ > +#define SAFETY_MARGIN 0 > + > +int fsp_get_header(ulong offset, ulong size, bool use_spi_flash, > + struct fsp_header **fspp) > +{ > + static efi_guid_t guid = FSP_HEADER_GUID; > + struct fv_ext_header *exhdr; > + struct fsp_header *fsp; > + struct ffs_file_header *file_hdr; > + struct fv_header *fv; > + struct raw_section *raw; > + void *ptr, *base; > + u8 buf[PROBE_BUF_SIZE]; > + struct udevice *dev; > + int ret; > + > + /* You are in a maze of twisty headers all alike */ What does this comment mean? > + debug("offset=%x buf=%x\n", (uint)offset, (uint)buf); > + if (use_spi_flash) { > + ret = uclass_first_device_err(UCLASS_SPI_FLASH, &dev); > + if (ret) > + return log_msg_ret("Cannot find flash device", ret); > + ret = spi_flash_read_dm(dev, offset, PROBE_BUF_SIZE, buf); > + if (ret) > + return log_msg_ret("Cannot read flash", ret); > + } else { > + memcpy(buf, (void *)offset, PROBE_BUF_SIZE); > + } > + > + /* Initalise the FSP base */ > + ptr = buf; > + fv = ptr; > + > + /* Check the FV signature, _FVH */ > + debug("offset=%x sign=%x\n", (uint)offset, (uint)fv->sign); > + if (fv->sign != EFI_FVH_SIGNATURE) > + return log_msg_ret("Base FV signature", -EINVAL); > + > + /* Go to the end of the FV header and align the address */ > + debug("fv->ext_hdr_off = %x\n", fv->ext_hdr_off); > + ptr += fv->ext_hdr_off; > + exhdr = ptr; > + ptr += ALIGN(exhdr->ext_hdr_size, 8); > + debug("ptr=%x\n", ptr - (void *)buf); > + > + /* Check the FFS GUID */ > + file_hdr = ptr; > + if (memcmp(&file_hdr->name, &guid, sizeof(guid))) > + return log_msg_ret("Base FFS GUID", -ENXIO); > + /* Add the FFS header size to find the raw section header */ > + ptr = file_hdr + 1; > + > + raw = ptr; > + debug("raw->type = %x\n", raw->type); > + if (raw->type != EFI_SECTION_RAW) > + return log_msg_ret("Section type not RAW", -ENOEXEC); > + > + /* Add the raw section header size to find the FSP header */ > + ptr = raw + 1; > + fsp = ptr; > + > + /* Check the FSPH header */ > + debug("fsp %x\n", (uint)fsp); > + if (fsp->sign != EFI_FSPH_SIGNATURE) > + return log_msg_ret("Base FSPH signature", -EACCES); > + > + base = (void *)fsp->img_base; > + debug("Image base %x\n", (uint)base); > + debug("Image addr %x\n", (uint)fsp->fsp_mem_init); > + if (use_spi_flash) { > + ret = spi_flash_read_dm(dev, offset, size + SAFETY_MARGIN, > + base); > + if (ret) > + return log_msg_ret("Could not read FPS-M", ret); > + } else { > + memcpy(base, (void *)offset, size + SAFETY_MARGIN); > + } > + ptr = base + (ptr - (void *)buf); > + *fspp = ptr; > + > + return 0; > +} > + > +u32 fsp_notify(struct fsp_header *fsp_hdr, u32 phase) > +{ > + fsp_notify_f notify; > + struct fsp_notify_params params; > + struct fsp_notify_params *params_ptr; > + u32 status; > + > + if (!fsp_hdr) > + fsp_hdr = gd->arch.fsp_s_hdr; > + > + if (!fsp_hdr) > + return log_msg_ret("no FSP", -ENOENT); > + > + notify = (fsp_notify_f)(fsp_hdr->img_base + fsp_hdr->fsp_notify); > + params.phase = phase; > + params_ptr = ¶ms; > + > + /* > + * Use ASM code to ensure correct parameter is on the stack for > + * FspNotify as U-Boot is using different ABI from FSP > + */ > + asm volatile ( > + "pushl %1;" /* push notify phase */ > + "call *%%eax;" /* call FspNotify */ > + "addl $4, %%esp;" /* clean up the stack */ > + : "=a"(status) : "m"(params_ptr), "a"(notify), > "m"(*params_ptr) > + ); > + > + return status; > +} > diff --git a/include/bootstage.h b/include/bootstage.h > index 5e7e242b834..c271d3a6cc8 100644 > --- a/include/bootstage.h > +++ b/include/bootstage.h > @@ -200,6 +200,9 @@ enum bootstage_id { > BOOTSTATE_ID_ACCUM_DM_SPL, > BOOTSTATE_ID_ACCUM_DM_F, > BOOTSTATE_ID_ACCUM_DM_R, > + BOOTSTATE_ID_ACCUM_FSP_M, > + BOOTSTATE_ID_ACCUM_FSP_S, > + BOOTSTAGE_ID_ACCUM_MMAP_SPI, > > /* a few spare for the user, from here */ > BOOTSTAGE_ID_USER, > -- Regards, Bin _______________________________________________ U-Boot mailing list [email protected] https://lists.denx.de/listinfo/u-boot

