Hi Simon, On Mon, Nov 25, 2019 at 12:12 PM Simon Glass <[email protected]> wrote: > > The memory and silicon init parts of the FSP need support code to work. > Add this for Apollo Lake. > > Signed-off-by: Simon Glass <[email protected]> > --- > > Changes in v5: > - Allocate the FSP-S data instead of using the stack > - Rename APOLLOLAKE_USB2_PORT_MAX > > Changes in v4: > - Adjust the comment for struct dw_i2c_speed_config > - Rename arch_fsp_s_preinit() to arch_fsps_preinit() > - Switch over to use pinctrl for pad init/config > - Tidy up mixed case in FSP code > - apollolake -> Apollo Lake > > Changes in v3: > - Add bootstage timing for reading vbt > - Add fspm_done() hook to handle FSP-S wierdness (it breaks SPI flash) > - Don't allow BOOT_FROM_FAST_SPI_FLASH with FSP-S > - Set boot_loader_tolum_size to 0 > - Use the IRQ uclass instead of ITSS > > Changes in v2: None > > arch/x86/cpu/apollolake/Makefile | 6 + > arch/x86/cpu/apollolake/fsp_m.c | 210 ++++++++++ > arch/x86/cpu/apollolake/fsp_s.c | 667 +++++++++++++++++++++++++++++++ > 3 files changed, 883 insertions(+) > create mode 100644 arch/x86/cpu/apollolake/fsp_m.c > create mode 100644 arch/x86/cpu/apollolake/fsp_s.c > > diff --git a/arch/x86/cpu/apollolake/Makefile > b/arch/x86/cpu/apollolake/Makefile > index dc6df15dab..1760df54d8 100644 > --- a/arch/x86/cpu/apollolake/Makefile > +++ b/arch/x86/cpu/apollolake/Makefile > @@ -10,6 +10,12 @@ obj-y += cpu_common.o > ifndef CONFIG_TPL_BUILD > obj-y += cpu.o > obj-y += punit.o > +ifdef CONFIG_SPL_BUILD > +obj-y += fsp_m.o > +endif > +endif > +ifndef CONFIG_SPL_BUILD > +obj-y += fsp_s.o > endif > > obj-y += hostbridge.o > diff --git a/arch/x86/cpu/apollolake/fsp_m.c b/arch/x86/cpu/apollolake/fsp_m.c > new file mode 100644 > index 0000000000..39cda7c49b > --- /dev/null > +++ b/arch/x86/cpu/apollolake/fsp_m.c > @@ -0,0 +1,210 @@ > +// SPDX-License-Identifier: GPL-2.0+ > +/* > + * Copyright 2019 Google LLC > + * Written by Simon Glass <[email protected]> > + */ > + > +#include <common.h> > +#include <dm.h> > +#include <asm/arch/iomap.h> > +#include <asm/arch/fsp/fsp_configs.h> > +#include <asm/arch/fsp/fsp_m_upd.h> > +#include <asm/fsp2/fsp_internal.h> > +#include <dm/uclass-internal.h> > + > +/* > + * ODT settings: > + * If ODT PIN to LP4 DRAM is pulled HIGH for ODT_A and HIGH for ODT_B, > + * choose ODT_A_B_HIGH_HIGH. If ODT PIN to LP4 DRAM is pulled HIGH for ODT_A > + * and LOW for ODT_B, choose ODT_A_B_HIGH_LOW. > + * > + * Note that the enum values correspond to the interpreted UPD fields > + * within Ch[3:0]_OdtConfig parameters. > + */ > +enum { > + ODT_A_B_HIGH_LOW = 0 << 1, > + ODT_A_B_HIGH_HIGH = 1 << 1, > + N_WR_24 = 1 << 5, > +}; > + > +/* > + * LPDDR4 helper routines for configuring the memory UPD for LPDDR4 > operation. > + * There are four physical LPDDR4 channels, each 32-bits wide. There are two > + * logical channels using two physical channels together to form a 64-bit > + * interface to memory for each logical channel. > + */ > + > +enum { > + LP4_PHYS_CH0A, > + LP4_PHYS_CH0B, > + LP4_PHYS_CH1A, > + LP4_PHYS_CH1B, > + > + LP4_NUM_PHYS_CHANNELS, > +}; > + > +/* > + * The DQs within a physical channel can be bit-swizzled within each byte. > + * Within a channel the bytes can be swapped, but the DQs need to be routed > + * with the corresponding DQS (strobe). > + */ > +enum { > + LP4_DQS0, > + LP4_DQS1, > + LP4_DQS2, > + LP4_DQS3, > + > + LP4_NUM_BYTE_LANES, > + DQ_BITS_PER_DQS = 8, > +}; > + > +/* Provide bit swizzling per DQS and byte swapping within a channel */ > +struct lpddr4_chan_swizzle_cfg { > + u8 dqs[LP4_NUM_BYTE_LANES][DQ_BITS_PER_DQS]; > +}; > + > +struct lpddr4_swizzle_cfg { > + struct lpddr4_chan_swizzle_cfg phys[LP4_NUM_PHYS_CHANNELS]; > +}; > + > +static void setup_sdram(struct fsp_m_config *cfg, > + const struct lpddr4_swizzle_cfg *swizzle_cfg) > +{ > + const struct lpddr4_chan_swizzle_cfg *sch; > + /* Number of bytes to copy per DQS */ > + const size_t sz = DQ_BITS_PER_DQS; > + int chan; > + > + cfg->memory_down = 1; > + cfg->scrambler_support = 1; > + cfg->channel_hash_mask = 0x36; > + cfg->slice_hash_mask = 9; > + cfg->interleaved_mode = 2; > + cfg->channels_slices_enable = 0; > + cfg->min_ref_rate2x_enable = 0; > + cfg->dual_rank_support_enable = 1; > + > + /* LPDDR4 is memory down so no SPD addresses */ > + cfg->dimm0_spd_address = 0; > + cfg->dimm1_spd_address = 0; > + > + for (chan = 0; chan < 4; chan++) { > + struct fsp_ram_channel *ch = &cfg->chan[chan]; > + > + ch->rank_enable = 1; > + ch->device_width = 1; > + ch->dram_density = 2; > + ch->option = 3; > + ch->odt_config = ODT_A_B_HIGH_HIGH; > + } > + > + /* > + * CH0_DQB byte lanes in the bit swizzle configuration field are > + * not 1:1. The mapping within the swizzling field is: > + * indices [0:7] - byte lane 1 (DQS1) DQ[8:15] > + * indices [8:15] - byte lane 0 (DQS0) DQ[0:7] > + * indices [16:23] - byte lane 3 (DQS3) DQ[24:31] > + * indices [24:31] - byte lane 2 (DQS2) DQ[16:23] > + */ > + sch = &swizzle_cfg->phys[LP4_PHYS_CH0B]; > + memcpy(&cfg->ch_bit_swizzling[0][0], &sch->dqs[LP4_DQS1], sz); > + memcpy(&cfg->ch_bit_swizzling[0][8], &sch->dqs[LP4_DQS0], sz); > + memcpy(&cfg->ch_bit_swizzling[0][16], &sch->dqs[LP4_DQS3], sz); > + memcpy(&cfg->ch_bit_swizzling[0][24], &sch->dqs[LP4_DQS2], sz); > + > + /* > + * CH0_DQA byte lanes in the bit swizzle configuration field are 1:1. > + */ > + sch = &swizzle_cfg->phys[LP4_PHYS_CH0A]; > + memcpy(&cfg->ch_bit_swizzling[1][0], &sch->dqs[LP4_DQS0], sz); > + memcpy(&cfg->ch_bit_swizzling[1][8], &sch->dqs[LP4_DQS1], sz); > + memcpy(&cfg->ch_bit_swizzling[1][16], &sch->dqs[LP4_DQS2], sz); > + memcpy(&cfg->ch_bit_swizzling[1][24], &sch->dqs[LP4_DQS3], sz); > + > + sch = &swizzle_cfg->phys[LP4_PHYS_CH1B]; > + memcpy(&cfg->ch_bit_swizzling[2][0], &sch->dqs[LP4_DQS1], sz); > + memcpy(&cfg->ch_bit_swizzling[2][8], &sch->dqs[LP4_DQS0], sz); > + memcpy(&cfg->ch_bit_swizzling[2][16], &sch->dqs[LP4_DQS3], sz); > + memcpy(&cfg->ch_bit_swizzling[2][24], &sch->dqs[LP4_DQS2], sz); > + > + /* > + * CH0_DQA byte lanes in the bit swizzle configuration field are 1:1. > + */ > + sch = &swizzle_cfg->phys[LP4_PHYS_CH1A]; > + memcpy(&cfg->ch_bit_swizzling[3][0], &sch->dqs[LP4_DQS0], sz); > + memcpy(&cfg->ch_bit_swizzling[3][8], &sch->dqs[LP4_DQS1], sz); > + memcpy(&cfg->ch_bit_swizzling[3][16], &sch->dqs[LP4_DQS2], sz); > + memcpy(&cfg->ch_bit_swizzling[3][24], &sch->dqs[LP4_DQS3], sz); > +} > + > +int fspm_update_config(struct udevice *dev, struct fspm_upd *upd) > +{ > + struct fsp_m_config *cfg = &upd->config; > + struct fspm_arch_upd *arch = &upd->arch; > + > + arch->nvs_buffer_ptr = NULL; > + prepare_mrc_cache(upd); > + arch->stack_base = (void *)0xfef96000; > + arch->boot_loader_tolum_size = 0; > + > + arch->boot_mode = FSP_BOOT_WITH_FULL_CONFIGURATION; > + cfg->serial_debug_port_type = 2; > + cfg->serial_debug_port_device = 2; > + cfg->serial_debug_port_stride_size = 2; > + cfg->serial_debug_port_address = 0; > + > + cfg->package = 1; > + /* Don't enforce a memory size limit */ > + cfg->memory_size_limit = 0; > + cfg->low_memory_max_value = 2048; /* 2 GB */ > + /* No restrictions on memory above 4GiB */ > + cfg->high_memory_max_value = 0; > + > + /* Always default to attempt to use saved training data */ > + cfg->disable_fast_boot = 0; > + > + const u8 *swizzle_data; > + > + swizzle_data = dev_read_u8_array_ptr(dev, "lpddr4-swizzle", > + LP4_NUM_BYTE_LANES * > + DQ_BITS_PER_DQS * > + LP4_NUM_PHYS_CHANNELS); > + if (!swizzle_data) > + return log_msg_ret("Cannot read swizzel data", -EINVAL); > + > + setup_sdram(cfg, (struct lpddr4_swizzle_cfg *)swizzle_data); > + > + cfg->pre_mem_gpio_table_ptr = 0; > + > + cfg->profile = 0xb; > + cfg->msg_level_mask = 0; > + > + /* other */ > + cfg->skip_cse_rbp = 1; > + cfg->periodic_retraining_disable = 0; > + cfg->enable_s3_heci2 = 0; > + > + return 0; > +} > + > +/* > + * The FSP-M binary appears to break the SPI controller. It can be fixed by > + * writing the BAR again, so do that here > + */ > +int fspm_done(struct udevice *dev) > +{ > + struct udevice *spi; > + int ret; > + > + /* Don't probe the device, since that reads the BAR */ > + ret = uclass_find_first_device(UCLASS_SPI, &spi); > + if (ret) > + return log_msg_ret("SPI", ret); > + if (!spi) > + return log_msg_ret("SPI2", -ENODEV);
SPI2? > + > + dm_pci_write_config32(spi, PCI_BASE_ADDRESS_0, > + IOMAP_SPI_BASE | PCI_BASE_ADDRESS_SPACE_MEMORY); > + > + return 0; > +} > diff --git a/arch/x86/cpu/apollolake/fsp_s.c b/arch/x86/cpu/apollolake/fsp_s.c > new file mode 100644 > index 0000000000..477f1089c2 > --- /dev/null > +++ b/arch/x86/cpu/apollolake/fsp_s.c > @@ -0,0 +1,667 @@ > +// SPDX-License-Identifier: GPL-2.0+ > +/* > + * Copyright 2019 Google LLC > + * Written by Simon Glass <[email protected]> > + */ > + > +#include <common.h> > +#include <acpi_s3.h> > +#include <binman.h> > +#include <dm.h> > +#include <irq.h> > +#include <asm/intel_pinctrl.h> > +#include <asm/io.h> > +#include <asm/intel_regs.h> > +#include <asm/msr.h> > +#include <asm/msr-index.h> > +#include <asm/pci.h> > +#include <asm/arch/cpu.h> > +#include <asm/arch/systemagent.h> > +#include <asm/arch/fsp/fsp_configs.h> > +#include <asm/arch/fsp/fsp_s_upd.h> > + > +#define PCH_P2SB_E0 0xe0 > +#define HIDE_BIT BIT(0) > + > +#define INTEL_GSPI_MAX 3 > +#define INTEL_I2C_DEV_MAX 8 > +#define MAX_USB2_PORTS 8 > + > +enum { > + CHIPSET_LOCKDOWN_FSP = 0, /* FSP handles locking per UPDs */ > + CHIPSET_LOCKDOWN_COREBOOT, /* coreboot handles locking */ > +}; > + > +enum i2c_speed { > + I2C_SPEED_STANDARD = 100000, > + I2C_SPEED_FAST = 400000, > + I2C_SPEED_FAST_PLUS = 1000000, > + I2C_SPEED_HIGH = 3400000, > + I2C_SPEED_FAST_ULTRA = 5000000, > +}; > + > +/* > + * Timing values are in units of clock period, with the clock speed > + * provided by the SOC > + * > + * TODO([email protected]): Connect this up to the I2C driver > + */ > +struct dw_i2c_speed_config { > + enum i2c_speed speed; > + /* SCL high and low period count */ > + u16 scl_lcnt; > + u16 scl_hcnt; > + /* > + * SDA hold time should be 300ns in standard and fast modes > + * and long enough for deterministic logic level change in > + * fast-plus and high speed modes. > + * > + * [15:0] SDA TX Hold Time > + * [23:16] SDA RX Hold Time > + */ > + u32 sda_hold; > +}; > + > +/* Serial IRQ control. SERIRQ_QUIET is the default (0) */ > +enum serirq_mode { > + SERIRQ_QUIET, > + SERIRQ_CONTINUOUS, > + SERIRQ_OFF, > +}; > + > +/* > + * This I2C controller has support for 3 independent speed configs but can > + * support both FAST_PLUS and HIGH speeds through the same set of speed > + * config registers. These are treated separately so the speed config values > + * can be provided via ACPI to the OS. > + */ > +#define DW_I2C_SPEED_CONFIG_COUNT 4 > + > +struct dw_i2c_bus_config { > + /* Bus should be enabled prior to ramstage with temporary base */ ramstage sounds like a coreboot stuff. Is this comment relevant with U-Boot? > + int early_init; > + /* Bus speed in Hz, default is I2C_SPEED_FAST (400 KHz) */ > + enum i2c_speed speed; > + /* > + * If rise_time_ns is non-zero the calculations for lcnt and hcnt > + * registers take into account the times of the bus. However, if > + * there is a match in speed_config those register values take > + * precedence > + */ > + int rise_time_ns; > + int fall_time_ns; > + int data_hold_time_ns; > + /* Specific bus speed configuration */ > + struct dw_i2c_speed_config speed_config[DW_I2C_SPEED_CONFIG_COUNT]; > +}; > + > +struct gspi_cfg { > + /* Bus speed in MHz */ > + u32 speed_mhz; > + /* Bus should be enabled prior to ramstage with temporary base */ > + u8 early_init; > +}; > + > +/* > + * This structure will hold data required by common blocks. > + * These are soc specific configurations which will be filled by soc. > + * We'll fill this structure once during init and use the data in common > block. > + */ > +struct soc_intel_common_config { > + int chipset_lockdown; > + struct gspi_cfg gspi[INTEL_GSPI_MAX]; > + struct dw_i2c_bus_config i2c[INTEL_I2C_DEV_MAX]; > +}; > + > +enum pnp_settings { > + PNP_PERF, > + PNP_POWER, > + PNP_PERF_POWER, > +}; > + > +struct usb2_eye_per_port { > + u8 per_port_tx_pe_half; > + u8 per_port_pe_txi_set; > + u8 per_port_txi_set; > + u8 hs_skew_sel; > + u8 usb_tx_emphasis_en; > + u8 per_port_rxi_set; > + u8 hs_npre_drv_sel; > + u8 override_en; > +}; > + > +struct apl_config { > + /* Common structure containing soc config data required by common > code*/ > + struct soc_intel_common_config common_soc_config; > + > + /* > + * Mapping from PCIe root port to CLKREQ input on the SOC. The SOC has > + * four CLKREQ inputs, but six root ports. Root ports without an > + * associated CLKREQ signal must be marked with "CLKREQ_DISABLED" > + */ > + u8 pcie_rp_clkreq_pin[MAX_PCIE_PORTS]; > + > + /* Enable/disable hot-plug for root ports (0 = disable, 1 = enable) */ > + u8 pcie_rp_hotplug_enable[MAX_PCIE_PORTS]; > + > + /* De-emphasis enable configuration for each PCIe root port */ > + u8 pcie_rp_deemphasis_enable[MAX_PCIE_PORTS]; > + > + /* [14:8] DDR mode Number of dealy elements.Each = 125pSec. nits: wrong multi-line comment format > + * [6:0] SDR mode Number of dealy elements.Each = 125pSec. > + */ > + u32 emmc_tx_cmd_cntl; > + > + /* [14:8] HS400 mode Number of dealy elements.Each = 125pSec. ditto > + * [6:0] SDR104/HS200 mode Number of dealy elements.Each = 125pSec. > + */ > + u32 emmc_tx_data_cntl1; > + > + /* [30:24] SDR50 mode Number of dealy elements.Each = 125pSec. ditto > + * [22:16] DDR50 mode Number of dealy elements.Each = 125pSec. > + * [14:8] SDR25/HS50 mode Number of dealy elements.Each = 125pSec. > + * [6:0] SDR12/Compatibility mode Number of dealy elements. > + * Each = 125pSec. > + */ > + u32 emmc_tx_data_cntl2; > + > + /* [30:24] SDR50 mode Number of dealy elements.Each = 125pSec. ditto > + * [22:16] DDR50 mode Number of dealy elements.Each = 125pSec. > + * [14:8] SDR25/HS50 mode Number of dealy elements.Each = 125pSec. > + * [6:0] SDR12/Compatibility mode Number of dealy elements. > + * Each = 125pSec. > + */ > + u32 emmc_rx_cmd_data_cntl1; > + > + /* [14:8] HS400 mode 1 Number of dealy elements.Each = 125pSec. ditto > + * [6:0] HS400 mode 2 Number of dealy elements.Each = 125pSec. > + */ > + u32 emmc_rx_strobe_cntl; > + > + /* [13:8] Auto Tuning mode Number of dealy elements.Each = 125pSec. ditto > + * [6:0] SDR104/HS200 Number of dealy elements.Each = 125pSec. > + */ > + u32 emmc_rx_cmd_data_cntl2; > + > + /* Select the eMMC max speed allowed */ > + u32 emmc_host_max_speed; > + > + /* Specifies on which IRQ the SCI will internally appear */ > + u32 sci_irq; > + > + /* Configure serial IRQ (SERIRQ) line */ > + enum serirq_mode serirq_mode; > + > + /* Configure LPSS S0ix Enable */ > + bool lpss_s0ix_enable; > + > + /* Enable DPTF support */ > + bool dptf_enable; > + > + /* TCC activation offset value in degrees Celsius */ > + int tcc_offset; > + > + /* Configure Audio clk gate and power gate ditto > + * IOSF-SB port ID 92 offset 0x530 [5] and [3] > + */ > + bool hdaudio_clk_gate_enable; > + bool hdaudio_pwr_gate_enable; > + bool hdaudio_bios_config_lockdown; > + > + /* SLP S3 minimum assertion width */ > + int slp_s3_assertion_width_usecs; > + > + /* GPIO pin for PERST_0 */ > + u32 prt0_gpio; > + > + /* USB2 eye diagram settings per port */ > + struct usb2_eye_per_port usb2eye[MAX_USB2_PORTS]; > + > + /* GPIO SD card detect pin */ > + unsigned int sdcard_cd_gpio; > + > + /* > + * PRMRR size setting with three options > + * 0x02000000 - 32MiB > + * 0x04000000 - 64MiB > + * 0x08000000 - 128MiB > + */ > + u32 PrmrrSize; > + > + /* > + * Enable SGX feature. > + * Enabling SGX feature is 2 step process, > + * (1) set sgx_enable = 1 > + * (2) set PrmrrSize to supported size > + */ > + bool sgx_enable; > + > + /* > + * Select PNP Settings. > + * (0) Performance, > + * (1) Power > + * (2) Power & Performance > + */ > + enum pnp_settings pnp_settings; > + > + /* > + * PMIC PCH_PWROK delay configuration - IPC Configuration > + * Upd for changing PCH_PWROK delay configuration : I2C_Slave_Address > + * (31:24) + Register_Offset (23:16) + OR Value (15:8) + AND Value > (7:0) > + */ > + u32 PmicPmcIpcCtrl; nits: wrong coding style > + > + /* > + * Options to disable XHCI Link Compliance Mode. Default is FALSE to > not > + * disable Compliance Mode. Set TRUE to disable Compliance Mode. > + * 0:FALSE(Default), 1:True. > + */ > + bool DisableComplianceMode; ditto > + > + /* > + * Options to change USB3 ModPhy setting for the Integrated Filter > (IF) > + * value. Default is 0 to not changing default IF value (0x12). Set > + * value with the range from 0x01 to 0xff to change IF value. > + */ > + u32 ModPhyIfValue; ditto > + > + /* > + * Options to bump USB3 LDO voltage. Default is FALSE to not > increasing > + * LDO voltage. Set TRUE to increase LDO voltage with 40mV. > + * 0:FALSE (default), 1:True. > + */ > + bool ModPhyVoltageBump; ditto > + > + /* > + * Options to adjust PMIC Vdd2 voltage. Default is 0 to not adjusting > + * the PMIC Vdd2 default voltage 1.20v. Upd for changing Vdd2 Voltage > + * configuration: I2C_Slave_Address (31:23) + Register_Offset (23:16) > + * + OR Value (15:8) + AND Value (7:0) through BUCK5_VID[3:2]: > + * 00=1.10v, 01=1.15v, 10=1.24v, 11=1.20v (default). > + */ > + u32 PmicVdd2Voltage; ditto > + > + /* > + * Option to enable VTD feature. Default is 0 which disables VTD > + * capability in FSP. Setting this option to 1 in devicetree will > enable I suspect the comment is coreboot-specific. ie: devicetree is coreboot devicetree, not the flattened DT. > + * the Upd parameter VtdEnable. > + */ > + bool enable_vtd; > +}; > + > +static int get_config(struct udevice *dev, struct apl_config *apl) > +{ > + const u8 *ptr; > + ofnode node; > + u32 emmc[4]; > + int ret; > + > + memset(apl, '\0', sizeof(*apl)); > + > + node = dev_read_subnode(dev, "fsp-s"); > + if (!ofnode_valid(node)) > + return log_msg_ret("fsp-s settings", -ENOENT); > + > + ptr = ofnode_read_u8_array_ptr(node, "pcie-rp-clkreq-pin", > + MAX_PCIE_PORTS); > + if (!ptr) > + return log_msg_ret("pcie-rp-clkreq-pin", -EINVAL); > + memcpy(apl->pcie_rp_clkreq_pin, ptr, MAX_PCIE_PORTS); > + > + ret = ofnode_read_u32(node, "prt0-gpio", &apl->prt0_gpio); > + if (ret) > + return log_msg_ret("prt0-gpio", ret); > + ret = ofnode_read_u32(node, "sdcard-cd-gpio", &apl->sdcard_cd_gpio); > + if (ret) > + return log_msg_ret("sdcard-cd-gpio", ret); > + > + ret = ofnode_read_u32_array(node, "emmc", emmc, ARRAY_SIZE(emmc)); > + if (ret) > + return log_msg_ret("emmc", ret); > + apl->emmc_tx_data_cntl1 = emmc[0]; > + apl->emmc_tx_data_cntl2 = emmc[1]; > + apl->emmc_rx_cmd_data_cntl1 = emmc[2]; > + apl->emmc_rx_cmd_data_cntl2 = emmc[3]; > + > + apl->dptf_enable = ofnode_read_bool(node, "dptf-enable"); > + > + apl->hdaudio_clk_gate_enable = ofnode_read_bool(node, > + "hdaudio-clk-gate-enable"); > + apl->hdaudio_pwr_gate_enable = ofnode_read_bool(node, > + "hdaudio-pwr-gate-enable"); > + apl->hdaudio_bios_config_lockdown = ofnode_read_bool(node, > + "hdaudio-bios-config-lockdown"); > + apl->lpss_s0ix_enable = ofnode_read_bool(node, "lpss-s0ix-enable"); > + > + /* Santa */ > + apl->usb2eye[1].per_port_pe_txi_set = 7; > + apl->usb2eye[1].per_port_txi_set = 2; > + > + return 0; > +} > + > +static void apl_fsp_silicon_init_params_cb(struct apl_config *apl, > + struct fsp_s_config *cfg) > +{ > + u8 port; > + > + for (port = 0; port < MAX_USB2_PORTS; port++) { > + if (apl->usb2eye[port].per_port_tx_pe_half) > + cfg->port_usb20_per_port_tx_pe_half[port] = > + apl->usb2eye[port].per_port_tx_pe_half; > + > + if (apl->usb2eye[port].per_port_pe_txi_set) > + cfg->port_usb20_per_port_pe_txi_set[port] = > + apl->usb2eye[port].per_port_pe_txi_set; > + > + if (apl->usb2eye[port].per_port_txi_set) > + cfg->port_usb20_per_port_txi_set[port] = > + apl->usb2eye[port].per_port_txi_set; > + > + if (apl->usb2eye[port].hs_skew_sel) > + cfg->port_usb20_hs_skew_sel[port] = > + apl->usb2eye[port].hs_skew_sel; > + > + if (apl->usb2eye[port].usb_tx_emphasis_en) > + cfg->port_usb20_i_usb_tx_emphasis_en[port] = > + apl->usb2eye[port].usb_tx_emphasis_en; > + > + if (apl->usb2eye[port].per_port_rxi_set) > + cfg->port_usb20_per_port_rxi_set[port] = > + apl->usb2eye[port].per_port_rxi_set; > + > + if (apl->usb2eye[port].hs_npre_drv_sel) > + cfg->port_usb20_hs_npre_drv_sel[port] = > + apl->usb2eye[port].hs_npre_drv_sel; > + } > +} > + > +int fsps_update_config(struct udevice *dev, ulong rom_offset, > + struct fsps_upd *upd) > +{ > + struct fsp_s_config *cfg = &upd->config; > + struct apl_config *apl; > + struct binman_entry vbt; > + void *buf; > + int ret; > + > + ret = binman_entry_find("intel-vbt", &vbt); > + if (ret) > + return log_msg_ret("Cannot find VBT", ret); > + vbt.image_pos += rom_offset; > + buf = malloc(vbt.size); > + if (!buf) > + return log_msg_ret("Alloc VBT", -ENOMEM); > + > + /* > + * Load VBT before devicetree-specific config. This only supports > + * memory-mapped SPI at present. > + */ > + bootstage_start(BOOTSTAGE_ID_ACCUM_MMAP_SPI, "mmap_spi"); > + memcpy(buf, (void *)vbt.image_pos, vbt.size); > + bootstage_accum(BOOTSTAGE_ID_ACCUM_MMAP_SPI); > + if (*(u32 *)buf != VBT_SIGNATURE) > + return log_msg_ret("VBT signature", -EINVAL); > + cfg->graphics_config_ptr = (ulong)buf; > + > + apl = malloc(sizeof(*apl)); > + if (!apl) > + return log_msg_ret("config", -ENOMEM); > + get_config(dev, apl); > + > + cfg->ish_enable = 0; > + cfg->enable_sata = 0; > + cfg->pcie_root_port_en[2] = 0; > + cfg->pcie_rp_hot_plug[2] = 0; > + cfg->pcie_root_port_en[3] = 0; > + cfg->pcie_rp_hot_plug[3] = 0; > + cfg->pcie_root_port_en[4] = 0; > + cfg->pcie_rp_hot_plug[4] = 0; > + cfg->pcie_root_port_en[5] = 0; > + cfg->pcie_rp_hot_plug[5] = 0; > + cfg->pcie_root_port_en[1] = 0; > + cfg->pcie_rp_hot_plug[1] = 0; > + cfg->usb_otg = 0; > + cfg->i2c6_enable = 0; > + cfg->i2c7_enable = 0; > + cfg->hsuart3_enable = 0; > + cfg->spi1_enable = 0; > + cfg->spi2_enable = 0; > + cfg->sdio_enabled = 0; > + > + memcpy(cfg->pcie_rp_clk_req_number, apl->pcie_rp_clkreq_pin, > + sizeof(cfg->pcie_rp_clk_req_number)); > + > + memcpy(cfg->pcie_rp_hot_plug, apl->pcie_rp_hotplug_enable, > + sizeof(cfg->pcie_rp_hot_plug)); > + > + switch (apl->serirq_mode) { > + case SERIRQ_QUIET: > + cfg->sirq_enable = 1; > + cfg->sirq_mode = 0; > + break; > + case SERIRQ_CONTINUOUS: > + cfg->sirq_enable = 1; > + cfg->sirq_mode = 1; > + break; > + case SERIRQ_OFF: > + default: > + cfg->sirq_enable = 0; > + break; > + } > + > + if (apl->emmc_tx_cmd_cntl) > + cfg->emmc_tx_cmd_cntl = apl->emmc_tx_cmd_cntl; > + if (apl->emmc_tx_data_cntl1) > + cfg->emmc_tx_data_cntl1 = apl->emmc_tx_data_cntl1; > + if (apl->emmc_tx_data_cntl2) > + cfg->emmc_tx_data_cntl2 = apl->emmc_tx_data_cntl2; > + if (apl->emmc_rx_cmd_data_cntl1) > + cfg->emmc_rx_cmd_data_cntl1 = apl->emmc_rx_cmd_data_cntl1; > + if (apl->emmc_rx_strobe_cntl) > + cfg->emmc_rx_strobe_cntl = apl->emmc_rx_strobe_cntl; > + if (apl->emmc_rx_cmd_data_cntl2) > + cfg->emmc_rx_cmd_data_cntl2 = apl->emmc_rx_cmd_data_cntl2; > + if (apl->emmc_host_max_speed) > + cfg->e_mmc_host_max_speed = apl->emmc_host_max_speed; > + > + cfg->lpss_s0ix_enable = apl->lpss_s0ix_enable; > + > + /* > + * Disable monitor mwait since it is broken due to a hardware bug > + * without a fix. Specific to Apollo Lake. > + */ > + if (!IS_ENABLED(CONFIG_SOC_INTEL_GLK)) see previous comments about this option > + cfg->monitor_mwait_enable = 0; > + > + cfg->skip_mp_init = true; > + > + /* Disable setting of EISS bit in FSP */ > + cfg->spi_eiss = 0; > + > + /* Disable FSP from locking access to the RTC NVRAM */ > + cfg->rtc_lock = 0; > + > + /* Enable Audio clk gate and power gate */ > + cfg->hd_audio_clk_gate = apl->hdaudio_clk_gate_enable; > + cfg->hd_audio_pwr_gate = apl->hdaudio_pwr_gate_enable; > + /* Bios config lockdown Audio clk and power gate */ > + cfg->bios_cfg_lock_down = apl->hdaudio_bios_config_lockdown; > + apl_fsp_silicon_init_params_cb(apl, cfg); > + > + cfg->usb_otg = true; > + > + /* Set VTD feature according to devicetree */ again, devicetree? > + cfg->vtd_enable = apl->enable_vtd; > + > + return 0; > +} > + > +static void p2sb_set_hide_bit(pci_dev_t dev, int hide) > +{ > + pci_x86_clrset_config(dev, PCH_P2SB_E0 + 1, HIDE_BIT, > + hide ? HIDE_BIT : 0, PCI_SIZE_8); > +} > + > +/* Configure package power limits */ > +static int set_power_limits(struct udevice *dev) > +{ > + msr_t rapl_msr_reg, limit; > + u32 power_unit; > + u32 tdp, min_power, max_power; > + u32 pl2_val; > + u32 override_tdp[2]; > + int ret; > + > + /* Get units */ > + rapl_msr_reg = msr_read(MSR_PKG_POWER_SKU_UNIT); > + power_unit = 1 << (rapl_msr_reg.lo & 0xf); > + > + /* Get power defaults for this SKU */ > + rapl_msr_reg = msr_read(MSR_PKG_POWER_SKU); > + tdp = rapl_msr_reg.lo & PKG_POWER_LIMIT_MASK; > + pl2_val = rapl_msr_reg.hi & PKG_POWER_LIMIT_MASK; > + min_power = (rapl_msr_reg.lo >> 16) & PKG_POWER_LIMIT_MASK; > + max_power = rapl_msr_reg.hi & PKG_POWER_LIMIT_MASK; > + > + if (min_power > 0 && tdp < min_power) > + tdp = min_power; > + > + if (max_power > 0 && tdp > max_power) > + tdp = max_power; > + > + ret = dev_read_u32_array(dev, "tdp-pl-override-mw", override_tdp, > + ARRAY_SIZE(override_tdp)); > + if (ret) > + return log_msg_ret("tdp-pl-override-mw", ret); > + > + /* Set PL1 override value */ > + if (override_tdp[0]) > + tdp = override_tdp[0] * power_unit / 1000; > + > + /* Set PL2 override value */ > + if (override_tdp[1]) > + pl2_val = override_tdp[1] * power_unit / 1000; > + > + /* Set long term power limit to TDP */ > + limit.lo = tdp & PKG_POWER_LIMIT_MASK; > + /* Set PL1 Pkg Power clamp bit */ > + limit.lo |= PKG_POWER_LIMIT_CLAMP; > + > + limit.lo |= PKG_POWER_LIMIT_EN; > + limit.lo |= (MB_POWER_LIMIT1_TIME_DEFAULT & > + PKG_POWER_LIMIT_TIME_MASK) << PKG_POWER_LIMIT_TIME_SHIFT; > + > + /* Set short term power limit PL2 */ > + limit.hi = pl2_val & PKG_POWER_LIMIT_MASK; > + limit.hi |= PKG_POWER_LIMIT_EN; > + > + /* Program package power limits in RAPL MSR */ > + msr_write(MSR_PKG_POWER_LIMIT, limit); > + log_info("RAPL PL1 %d.%dW\n", tdp / power_unit, > + 100 * (tdp % power_unit) / power_unit); > + log_info("RAPL PL2 %d.%dW\n", pl2_val / power_unit, > + 100 * (pl2_val % power_unit) / power_unit); > + > + /* > + * Sett RAPL MMIO register for Power limits. RAPL driver is using MSR > + * instead of MMIO, so disable LIMIT_EN bit for MMIO > + */ > + writel(limit.lo & ~PKG_POWER_LIMIT_EN, MCHBAR_REG(MCHBAR_RAPL_PPL)); > + writel(limit.hi & ~PKG_POWER_LIMIT_EN, MCHBAR_REG(MCHBAR_RAPL_PPL + > 4)); > + > + return 0; > +} > + > +int p2sb_unhide(void) > +{ > + pci_dev_t dev = PCI_BDF(0, 0xd, 0); > + ulong val; > + > + p2sb_set_hide_bit(dev, 0); > + > + pci_x86_read_config(dev, PCI_VENDOR_ID, &val, PCI_SIZE_16); > + > + if (val != PCI_VENDOR_ID_INTEL) > + return log_msg_ret("p2sb unhide", -EIO); > + > + return 0; > +} > + > +/* Overwrites the SCI IRQ if another IRQ number is given by device tree */ > +static void set_sci_irq(void) > +{ > + /* Skip this for now */ > +} > + > +int arch_fsps_preinit(void) > +{ > + struct udevice *itss; > + int ret; > + > + ret = uclass_first_device_err(UCLASS_IRQ, &itss); > + if (ret) > + return log_msg_ret("no itss", ret); > + /* > + * Snapshot the current GPIO IRQ polarities. FSP is setting a default > + * policy that doesn't honour boards' requirements > + */ > + irq_snapshot_polarities(itss); > + > + /* > + * Clear the GPI interrupt status and enable registers. These > + * registers do not get reset to default state when booting from S5. > + */ > + ret = pinctrl_gpi_clear_int_cfg(); > + if (ret) > + return log_msg_ret("gpi_clear", ret); > + > + return 0; > +} > + > +int arch_fsp_init_r(void) > +{ > +#ifdef CONFIG_HAVE_ACPI_RESUME > + bool s3wake = gd->arch.prev_sleep_state == ACPI_S3; > +#else > + bool s3wake = false; > +#endif > + struct udevice *dev, *itss; > + int ret; > + > + /* > + * This must be called before any devices are probed. Put any probing > + * into arch_fsps_preinit() above. > + * > + * We don't use BOOT_FROM_FAST_SPI_FLASH here since it will force PCI > + * to be probed. > + */ > + ret = fsp_silicon_init(s3wake, false); > + if (ret) > + return ret; > + > + ret = uclass_first_device_err(UCLASS_IRQ, &itss); > + if (ret) > + return log_msg_ret("no itss", ret); > + /* Restore GPIO IRQ polarities back to previous settings */ > + irq_restore_polarities(itss); > + > + /* soc_init() */ > + ret = p2sb_unhide(); > + if (ret) > + return log_msg_ret("unhide p2sb", ret); > + > + /* Set RAPL MSR for Package power limits*/ > + ret = uclass_first_device_err(UCLASS_NORTHBRIDGE, &dev); > + if (ret) > + return log_msg_ret("Cannot get northbridge", ret); > + set_power_limits(dev); > + > + /* > + * FSP-S routes SCI to IRQ 9. With the help of this function you can > + * select another IRQ for SCI. > + */ > + set_sci_irq(); > + > + return 0; > +} > -- I never imagined even using FSP, the initialization of a SoC platform could still be such complex! Either FSP was designed with something wrong, or we are not doing correctly ... Regards, Bin _______________________________________________ U-Boot mailing list [email protected] https://lists.denx.de/listinfo/u-boot

