/* SPDX-License-Identifier: GPL-2.0-only */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include void __weak mainboard_config_gpios(void) {} #if CONFIG(DISPLAY_HOBS) static void display_fsp_smbios_memory_info_hob(void) { const FSP_SMBIOS_MEMORY_INFO *memory_info_hob; /* Get the memory info HOB */ memory_info_hob = soc_get_fsp_smbios_memory_info_hob(); if (memory_info_hob == NULL) return; soc_display_fsp_smbios_memory_info_hob(memory_info_hob); } #endif static void early_pmc_init(void) { /* PMC (B0:D31:F2). */ pci_devfn_t dev = PCH_DEV_PMC; /* Is PMC present */ if (pci_read_config16(dev, 0) == 0xffff) { printk(BIOS_ERR, "PMC controller (B0:D31:F2) does not present!\n"); return; } uint32_t pwrm_base = pci_read_config32(dev, PMC_PWRM_BASE) & MASK_PMC_PWRM_BASE; if (!pwrm_base) { printk(BIOS_ERR, "PWRM base address is not configured!\n"); return; } /* Workaround for sighting report (doc#: 560805) v1.86. 42. System Might Hang In AC Power Loss Problem : When removing and reapplying AC power to the board, the system might hang at serial output 'RESET required : change of frequency' due to PMC ROM change on B0. Implication : 1. This issue is only shown in B0 stepping. 2. This issue does not impact a system without an RTC battery. Alternative workaround : Remove RTC battery on the board if possible. Status : Plan Fix. */ if (silicon_stepping() == SILICON_REV_DENVERTON_B0) { if (!(pci_read_config32(dev, GEN_PMCON_B) & GEN_PMCON_B_RTC_PWR_STS)) { if (read32((void *)(pwrm_base + 0x124)) & ((1 << 11) | (1 << 12))) { /* Performs a global reset */ printk(BIOS_DEBUG, "Requesting Global Reset...\n"); pci_write_config32(dev, ETR3, pci_read_config32(dev, ETR3) | ETR3_CF9GR); full_reset(); } } } } static void early_tco_init(void) { /* SMBUS (B0:D31:F4). */ pci_devfn_t dev = PCI_DEV(0, SMBUS_DEV, SMBUS_FUNC); /* Configure TCO base address */ if (pci_read_config16(dev, TCOBASE) == 0xffff) { printk(BIOS_ERR, "SMBus controller (B0:D31:F4) does not present!\n"); return; } uint16_t tco_ctl = pci_read_config16(dev, TCOCTL); if (tco_ctl & TCOBASE_LOCK) { printk(BIOS_ERR, "TCO base register already has been locked!\n"); } else { pci_write_config16(dev, TCOCTL, tco_ctl & (~TCOBASE_EN)); pci_write_config16(dev, TCOBASE, DEFAULT_TCO_BASE | 0x1); pci_write_config16(dev, TCOCTL, tco_ctl | TCOBASE_EN); } uint16_t tco_base = pci_read_config16(dev, TCOBASE) & MASK_TCOBASE; printk(BIOS_DEBUG, "TCO base address set to 0x%x!\n", tco_base); /* Disable the TCO timer expiration from causing a system reset */ MMIO32_OR(PCH_PCR_ADDRESS(PID_SMB, PCR_SMBUS_GC), (uint32_t)PCR_SMBUS_GC_NR); /* Halt the TCO timer */ uint16_t reg16 = inw(tco_base + TCO1_CNT); reg16 |= TCO_TMR_HLT; outw(reg16, tco_base + TCO1_CNT); /* Clear the Second TCO status bit */ reg16 = inw(tco_base + TCO2_STS); reg16 |= TCO2_STS_SECOND_TO; outw(reg16, tco_base + TCO2_STS); } void mainboard_romstage_entry(void) { printk(BIOS_DEBUG, "FSP TempRamInit was successful...\n"); mainboard_config_gpios(); early_tco_init(); early_pmc_init(); fsp_memory_init(false); #if CONFIG(DISPLAY_HOBS) display_fsp_smbios_memory_info_hob(); #endif } static void soc_memory_init_params(FSP_M_CONFIG *m_cfg) { FSPM_UPD *mupd = container_of(m_cfg, FSPM_UPD, FspmConfig); size_t num; uint16_t supported_hsio_lanes; BL_HSIO_INFORMATION *hsio_config; /* Set the parameters for MemoryInit */ m_cfg->PcdEnableIQAT = CONFIG(IQAT_ENABLE); /* if ME HECI communication is disabled, apply default one*/ if (mupd->FspmConfig.PcdMeHeciCommunication == 0) { /* Configure FIA MUX PCD */ /* Assume the validating silicon has max lanes. */ supported_hsio_lanes = BL_ME_FIA_MUX_LANE_NUM_MAX; num = mainboard_get_hsio_config(&hsio_config); if (get_fiamux_hsio_info(supported_hsio_lanes, num, &hsio_config)) die("HSIO Configuration is invalid, please correct " "it!"); mupd->FspmConfig.PcdHsioLanesNumber = (uint32_t)hsio_config->NumLanesSupported; mupd->FspmConfig.PcdFiaMuxConfigPtr = (uint32_t)&hsio_config->FiaConfig; } } __weak void mainboard_memory_init_params(FSPM_UPD *mupd) { /* Do nothing */ } void platform_fsp_memory_init_params_cb(FSPM_UPD *mupd, uint32_t version) { FSP_M_CONFIG *m_cfg = &mupd->FspmConfig; soc_memory_init_params(m_cfg); mainboard_memory_init_params(mupd); }