/** @file
Copyright (c) 2004 - 2014, Intel Corporation. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent
Module Name:
PchInitPeim.c
Abstract:
Do Early PCH platform initialization.
--*/
#include "PlatformEarlyInit.h"
#include "Ppi/PchPlatformPolicy.h"
#include "PchRegs.h"
#include
#include "Ppi/PchInit.h"
#include
EFI_GUID gPchPlatformPolicyPpiGuid = PCH_PLATFORM_POLICY_PPI_GUID;
#define MC_PMSTS_OFFSET 0xC
#define DEFAULT_BUS_INFO 0x2020
#define PCI_LPC_BASE (0x8000F800)
#define PCI_LPC_REG(x) (PCI_LPC_BASE + (x))
#define PCIEX_BASE_ADDRESS 0xE0000000
#define PciD31F0RegBase PCIEX_BASE_ADDRESS + (UINT32) (31 << 15)
VOID
PchPolicySetupInit (
IN CONST EFI_PEI_SERVICES **PeiServices,
IN SYSTEM_CONFIGURATION *SystemConfiguration
);
VOID
PchInitInterrupt (
IN SYSTEM_CONFIGURATION *SystemConfiguration
);
EFI_STATUS
InstallPeiPchUsbPolicy (
IN CONST EFI_PEI_SERVICES **PeiServices
);
#ifndef __GNUC__
#pragma warning (push)
#pragma warning (disable : 4245)
#pragma warning (pop)
#endif
UINT8
ReadCmosBank1Byte (
IN UINT8 Address
)
{
UINT8 Data;
IoWrite8(R_PCH_RTC_EXT_INDEX, Address);
Data = IoRead8 (R_PCH_RTC_EXT_TARGET);
return Data;
}
VOID
WriteCmosBank1Byte (
IN UINT8 Address,
IN UINT8 Data
)
{
IoWrite8(R_PCH_RTC_EXT_INDEX, Address);
IoWrite8(R_PCH_RTC_EXT_TARGET, Data);
}
/**
Turn off system if needed.
@param PeiServices Pointer to PEI Services
@param CpuIo Pointer to CPU I/O Protocol
@retval None.
**/
VOID
CheckPowerOffNow (
VOID
)
{
UINT16 Pm1Sts;
//
// Read and check the ACPI registers
//
Pm1Sts = IoRead16 (ACPI_BASE_ADDRESS + R_PCH_ACPI_PM1_STS);
if ((Pm1Sts & B_PCH_ACPI_PM1_STS_PWRBTN) == B_PCH_ACPI_PM1_STS_PWRBTN) {
IoWrite16 (ACPI_BASE_ADDRESS + R_PCH_ACPI_PM1_STS, B_PCH_ACPI_PM1_STS_PWRBTN);
IoWrite16 (ACPI_BASE_ADDRESS + R_PCH_ACPI_PM1_CNT, V_PCH_ACPI_PM1_CNT_S5);
IoWrite16 (ACPI_BASE_ADDRESS + R_PCH_ACPI_PM1_CNT, V_PCH_ACPI_PM1_CNT_S5 + B_PCH_ACPI_PM1_CNT_SLP_EN);
//
// Should not return
//
CpuDeadLoop();
}
}
VOID
ClearPowerState (
IN SYSTEM_CONFIGURATION *SystemConfiguration
)
{
UINT8 Data8;
UINT16 Data16;
UINT32 Data32;
//
// Check for PowerState option for AC power loss and program the chipset
//
//
// Clear PWROK (Set to Clear)
//
MmioOr32 (PMC_BASE_ADDRESS + R_PCH_PMC_GEN_PMCON_1, B_PCH_PMC_GEN_PMCON_PWROK_FLR);
//
// Clear Power Failure Bit (Set to Clear)
//
// TODO: Check if it is OK to clear here
//
MmioOr32 (PMC_BASE_ADDRESS + R_PCH_PMC_GEN_PMCON_1, B_PCH_PMC_GEN_PMCON_SUS_PWR_FLR);
//
// Clear the GPE and PM enable
//
IoWrite16 (ACPI_BASE_ADDRESS + R_PCH_ACPI_PM1_EN, (UINT16) 0x00);
IoWrite32 (ACPI_BASE_ADDRESS + R_PCH_ACPI_GPE0a_EN, (UINT32) 0x00);
//
// Halt the TCO timer
//
Data16 = IoRead16 (ACPI_BASE_ADDRESS + R_PCH_TCO_CNT);
Data16 |= B_PCH_TCO_CNT_TMR_HLT;
IoWrite16 (ACPI_BASE_ADDRESS + R_PCH_TCO_CNT, Data16);
//
// if NMI_NOW_STS is set
// NMI NOW bit is "Write '1' to clear"
//
Data8 = MmioRead8(ILB_BASE_ADDRESS + R_PCH_ILB_GNMI);
if ((Data8 & B_PCH_ILB_GNMI_NMINS) == B_PCH_ILB_GNMI_NMINS) {
MmioOr8 (ILB_BASE_ADDRESS + R_PCH_ILB_GNMI, B_PCH_ILB_GNMI_NMIN);
}
//
// Before we clear the TO status bit here we need to save the results in a CMOS bit for later use.
//
Data32 = IoRead32 (ACPI_BASE_ADDRESS + R_PCH_TCO_STS);
if ((Data32 & B_PCH_TCO_STS_SECOND_TO) == B_PCH_TCO_STS_SECOND_TO)
{
#if (defined(HW_WATCHDOG_TIMER_SUPPORT) && (HW_WATCHDOG_TIMER_SUPPORT != 0))
WriteCmosBank1Byte (
EFI_CMOS_PERFORMANCE_FLAGS,
ReadCmosBank1Byte (EFI_CMOS_PERFORMANCE_FLAGS) | B_CMOS_TCO_WDT_RESET
);
#endif
}
}
/*++
Clear any SMI status or wake status left over from boot.
**/
VOID
ClearSmiAndWake (
VOID
)
{
UINT16 Pm1Sts;
UINT32 Gpe0Sts;
UINT32 SmiSts;
//
// Read the ACPI registers
//
Pm1Sts = IoRead16 (ACPI_BASE_ADDRESS + R_PCH_ACPI_PM1_STS);
Gpe0Sts = IoRead32 (ACPI_BASE_ADDRESS + R_PCH_ACPI_GPE0a_STS);
SmiSts = IoRead32 (ACPI_BASE_ADDRESS + R_PCH_SMI_STS);
//
// Register Wake up reason for S4. This information is used to notify
// WinXp of wake up reason because S4 wake up path doesn't keep SCI.
// This is important for Viiv(Quick resume) platform.
//
//
// First Clear CMOS S4 Wake up flag.
//
WriteCmosBank1Byte(CMOS_S4_WAKEUP_FLAG_ADDRESS, 0);
//
// Check wake up reason and set CMOS accordingly. Currently checks
// Power button, USB, PS/2.
// Note : PS/2 wake up is using GPI13 (IO_PME). This must be changed depending
// on board design.
//
if ((Pm1Sts & B_PCH_ACPI_PM1_STS_PWRBTN) || (Gpe0Sts & (B_PCH_ACPI_GPE0a_STS_CORE_GPIO | B_PCH_ACPI_GPE0a_STS_SUS_GPIO))) {
WriteCmosBank1Byte(CMOS_S4_WAKEUP_FLAG_ADDRESS, 1);
}
//
// Clear any SMI or wake state from the boot
//
Pm1Sts = (B_PCH_ACPI_PM1_STS_PRBTNOR | B_PCH_ACPI_PM1_STS_PWRBTN);
Gpe0Sts |=
(
B_PCH_ACPI_GPE0a_STS_CORE_GPIO |
B_PCH_ACPI_GPE0a_STS_SUS_GPIO |
B_PCH_ACPI_GPE0a_STS_PME_B0 |
B_PCH_ACPI_GPE0a_STS_BATLOW |
B_PCH_ACPI_GPE0a_STS_PCI_EXP |
B_PCH_ACPI_GPE0a_STS_GUNIT_SCI |
B_PCH_ACPI_GPE0a_STS_PUNIT_SCI |
B_PCH_ACPI_GPE0a_STS_SWGPE |
B_PCH_ACPI_GPE0a_STS_HOT_PLUG
);
SmiSts |=
(
B_PCH_SMI_STS_SMBUS |
B_PCH_SMI_STS_PERIODIC |
B_PCH_SMI_STS_TCO |
B_PCH_SMI_STS_SWSMI_TMR |
B_PCH_SMI_STS_APM |
B_PCH_SMI_STS_ON_SLP_EN |
B_PCH_SMI_STS_BIOS
);
//
// Write them back
//
IoWrite16 (ACPI_BASE_ADDRESS + R_PCH_ACPI_PM1_STS, Pm1Sts);
IoWrite32 (ACPI_BASE_ADDRESS + R_PCH_ACPI_GPE0a_STS, Gpe0Sts);
IoWrite32 (ACPI_BASE_ADDRESS + R_PCH_SMI_STS, SmiSts);
}
/**
Issue PCI-E Secondary Bus Reset
@param Bus Bus number of the bridge
@param Dev Devices number of the bridge
@param Fun Function number of the bridge
@retval EFI_SUCCESS
**/
EFI_STATUS
PcieSecondaryBusReset (
IN CONST EFI_PEI_SERVICES **PeiServices,
IN UINT8 Bus,
IN UINT8 Dev,
IN UINT8 Fun
)
{
EFI_PEI_STALL_PPI *PeiStall;
EFI_STATUS Status;
Status = (**PeiServices).LocatePpi (
PeiServices,
&gEfiPeiStallPpiGuid,
0,
NULL,
(void **)&PeiStall
);
ASSERT_EFI_ERROR (Status);
//
// Issue secondary bus reset
//
MmPci16Or(0, Bus, Dev, Fun, PCI_BRIDGE_CONTROL_REGISTER_OFFSET, EFI_PCI_BRIDGE_CONTROL_RESET_SECONDARY_BUS);
//
// Wait 1ms
//
PeiStall->Stall (PeiServices, PeiStall, 1000);
//
// Clear the reset bit
// Note: The PCIe spec suggests 100ms delay between clearing this bit and accessing
// the device's config space. Since we will not access the config space until we enter DXE
// we don't put delay expressly here.
//
MmPci16And(0, Bus, Dev, Fun, PCI_BRIDGE_CONTROL_REGISTER_OFFSET, ~(EFI_PCI_BRIDGE_CONTROL_RESET_SECONDARY_BUS));
return EFI_SUCCESS;
}
/**
Provide hard reset PPI service.
To generate full hard reset, write 0x0E to ICH RESET_GENERATOR_PORT (0xCF9).
@param PeiServices General purpose services available to every PEIM.
@retval Not return System reset occured.
@retval EFI_DEVICE_ERROR Device error, could not reset the system.
**/
EFI_STATUS
EFIAPI
IchReset (
IN CONST EFI_PEI_SERVICES **PeiServices
)
{
IoWrite8 (
R_PCH_RST_CNT,
V_PCH_RST_CNT_HARDSTARTSTATE
);
IoWrite8 (
R_PCH_RST_CNT,
V_PCH_RST_CNT_HARDRESET
);
//
// System reset occured, should never reach at this line.
//
ASSERT_EFI_ERROR (EFI_DEVICE_ERROR);
CpuDeadLoop();
return EFI_DEVICE_ERROR;
}
VOID
PchPlatformLpcInit (
IN CONST EFI_PEI_SERVICES **PeiServices,
IN SYSTEM_CONFIGURATION *SystemConfiguration
)
{
EFI_BOOT_MODE BootMode;
UINT8 Data8;
UINT16 Data16;
(*PeiServices)->GetBootMode(PeiServices, &BootMode);
if ((BootMode != BOOT_ON_S3_RESUME)) {
//
// Clear all pending SMI. On S3 clear power button enable so it wll not generate an SMI
//
ClearSmiAndWake ();
}
ClearPowerState (SystemConfiguration);
//
// Need to set and clear SET bit (RTC_REGB Bit 7) as requested by the ICH EDS
// early in POST after each power up directly after coin-cell battery insertion.
// This is to avoid the UIP bit (RTC_REGA Bit 7) from stuck at "1".
// The UIP bit status may be polled by software (i.e ME FW) during POST.
//
if (MmioRead8 (PMC_BASE_ADDRESS + R_PCH_PMC_GEN_PMCON_1) & B_PCH_PMC_GEN_PMCON_RTC_PWR_STS) {
//
// Set and clear SET bit in RTC_REGB
//
IoWrite8(R_PCH_RTC_INDEX, R_PCH_RTC_REGISTERB);
Data8 = IoRead8(R_PCH_RTC_TARGET);
Data8 |= B_PCH_RTC_REGISTERB_SET;
IoWrite8(R_PCH_RTC_TARGET, Data8);
IoWrite8(R_PCH_RTC_INDEX, R_PCH_RTC_REGISTERB);
Data8 &= (~B_PCH_RTC_REGISTERB_SET);
IoWrite8(R_PCH_RTC_TARGET, Data8);
//
// Clear the UIP bit in RTC_REGA
//
IoWrite8(R_PCH_RTC_INDEX, R_PCH_RTC_REGISTERA);
IoWrite8(R_PCH_RTC_TARGET, 0x00);
}
//
// Disable SERR NMI and IOCHK# NMI in port 61
//
Data8 = IoRead8 (R_PCH_NMI_SC);
IoWrite8(R_PCH_NMI_SC, (UINT8) (Data8 | B_PCH_NMI_SC_PCI_SERR_EN | B_PCH_NMI_SC_IOCHK_NMI_EN));
//
// Enable Bus Master, I/O, Mem, and SERR on LPC bridge
//
Data16 = PchLpcPciCfg16 (R_PCH_LPC_COMMAND);
MmioWrite16 (
MmPciAddress (0,
DEFAULT_PCI_BUS_NUMBER_PCH,
PCI_DEVICE_NUMBER_PCH_LPC,
PCI_FUNCTION_NUMBER_PCH_LPC,
R_PCH_LPC_COMMAND
),
(Data16 |
B_PCH_LPC_COMMAND_IOSE |
B_PCH_LPC_COMMAND_MSE |
B_PCH_LPC_COMMAND_BME |
B_PCH_LPC_COMMAND_SERR_EN)
);
//
// Set Stretch S4 to 1-2s per marketing request.
// Note: This register is powered by RTC well.
//
MmioAndThenOr8 (
PMC_BASE_ADDRESS + R_PCH_PMC_GEN_PMCON_1 ,
(UINT8) (~B_PCH_PMC_GEN_PMCON_SLP_S4_MAW),
(UINT8) (B_PCH_PMC_GEN_PMCON_SLP_S4_ASE | V_PCH_PMC_GEN_PMCON_SLP_S4_MAW_4S)
);
}
#define V_PCH_ILB_IRQE_UARTIRQEN_IRQ3 BIT3 // UART IRQ3 Enable
VOID
UARTInit (
IN SYSTEM_CONFIGURATION *SystemConfiguration
)
{
if (0) { // for fix cr4 issue
//
// Program and enable PMC Base.
//
IoWrite32 (0xCF8, PCI_LPC_REG(R_PCH_LPC_PMC_BASE));
IoWrite32 (0xCFC, (PMC_BASE_ADDRESS | B_PCH_LPC_PMC_BASE_EN));
if( (SystemConfiguration->PcuUart1 == 1) &&
(SystemConfiguration->LpssHsuart0Enabled == 0)){
//
// Enable COM1 for debug message output.
//
MmioOr32 (PMC_BASE_ADDRESS + R_PCH_PMC_GEN_PMCON_1, BIT24);
//
//Enable internal UART3 port(COM1)
//
MmioOr8 (ILB_BASE_ADDRESS + R_PCH_ILB_IRQE, (UINT8) V_PCH_ILB_IRQE_UARTIRQEN_IRQ3);
MmioOr32 (IO_BASE_ADDRESS + 0x0520, 0x01); // UART3_RXD-L
MmioOr32 (IO_BASE_ADDRESS + 0x0530, 0x01); // UART3_TXD-0
MmioOr8 (PciD31F0RegBase + R_PCH_LPC_UART_CTRL, (UINT8) B_PCH_LPC_UART_CTRL_COM1_EN);
} else {
//
//Disable UART3(COM1)
//
MmioAnd8 (ILB_BASE_ADDRESS + R_PCH_ILB_IRQE, (UINT8) ~V_PCH_ILB_IRQE_UARTIRQEN_IRQ3);
MmioAnd32 (IO_BASE_ADDRESS + 0x0520, ~(UINT32)0x07);
MmioAnd32 (IO_BASE_ADDRESS + 0x0530, ~(UINT32)0x07);
MmioAnd8 (PciD31F0RegBase + R_PCH_LPC_UART_CTRL, (UINT8) ~B_PCH_LPC_UART_CTRL_COM1_EN);
if (SystemConfiguration->LpssHsuart0Enabled == 1){
//
//Valleyview BIOS Specification Vol2,17.2
//LPSS_UART1 ¨C set each pad PAD_CONF0.Func_Pin_Mux to function 1:
//
MmioAnd8 (IO_BASE_ADDRESS + 0x0090, (UINT8)~0x07);
MmioOr8 (IO_BASE_ADDRESS + 0x0090, 0x01);
MmioAnd8 (IO_BASE_ADDRESS + 0x00D0, (UINT8)~0x07);
MmioOr8 (IO_BASE_ADDRESS + 0x00D0, 0x01);
}
}
DEBUG ((EFI_D_ERROR, "EnableInternalUart\n"));
} else {
//
// If SIO UART interface selected
//Disable internal UART port(COM1)
//
if (0) {; // For fix CR4 issue
MmioAnd8 (ILB_BASE_ADDRESS + R_PCH_ILB_IRQE, (UINT8) ~V_PCH_ILB_IRQE_UARTIRQEN_IRQ3);
MmioAnd8 (IO_BASE_ADDRESS + 0x0090, (UINT8)~0x07);
MmioAnd8 (IO_BASE_ADDRESS + 0x00D0, (UINT8)~0x07);
MmioAnd8 (PciD31F0RegBase + R_PCH_LPC_UART_CTRL, (UINT8) ~B_PCH_LPC_UART_CTRL_COM1_EN);
}
}
}
VOID
IchRcrbInit (
IN CONST EFI_PEI_SERVICES **PeiServices,
IN SYSTEM_CONFIGURATION *SystemConfiguration
)
{
EFI_BOOT_MODE BootMode;
(*PeiServices)->GetBootMode(PeiServices, &BootMode);
//
// If not recovery or flash update boot path. set the BIOS interface lock down bit.
// It locks the top swap bit and BIOS boot strap bits from being changed.
//
if ((BootMode != BOOT_IN_RECOVERY_MODE) && (BootMode != BOOT_ON_FLASH_UPDATE)) {
MmioOr8 (RCBA_BASE_ADDRESS + R_PCH_RCRB_GCS, B_PCH_RCRB_GCS_BILD);
}
//
// Disable the Watchdog timer expiration from causing a system reset
//
MmioOr8 (PMC_BASE_ADDRESS + R_PCH_PMC_PM_CFG, B_PCH_PMC_PM_CFG_NO_REBOOT);
//
// Initial RCBA according to the PeiRCBA table
//
if ((BootMode == BOOT_ON_S3_RESUME)) {
//
// We are resuming from S3
// Enable HPET if enabled in Setup
// ICH Config register Offset 0x3404 bit 7 (Enable) = 1,
// Bit 1:0 (Mem I/O address) = 0 (0xFED00000)
//
MmioOr8 (R_PCH_PCH_HPET + R_PCH_PCH_HPET_GCFG, B_PCH_PCH_HPET_GCFG_EN);
}
}
EFI_STATUS
PlatformPchInit (
IN SYSTEM_CONFIGURATION *SystemConfiguration,
IN CONST EFI_PEI_SERVICES **PeiServices,
IN UINT16 PlatformType
)
{
EFI_STATUS Status;
EFI_BOOT_MODE BootMode;
Status = PeiServicesGetBootMode (&BootMode);
ASSERT_EFI_ERROR (Status);
IchRcrbInit (PeiServices, SystemConfiguration);
if (BootMode == BOOT_IN_RECOVERY_MODE) {
InstallPeiPchUsbPolicy(PeiServices);
}
//
// PCH Policy Initialization based on Setup variable.
//
PchPolicySetupInit (PeiServices, SystemConfiguration);
UARTInit(SystemConfiguration);
PchPlatformLpcInit (PeiServices, SystemConfiguration);
return EFI_SUCCESS;
}
/**
Returns the state of A16 inversion
@retval TRUE A16 is inverted
@retval FALSE A16 is not inverted
**/
BOOLEAN
IsA16Inverted (
)
{
UINT8 Data;
Data = MmioRead8 (RCBA_BASE_ADDRESS + R_PCH_RCRB_GCS);
return (Data & B_PCH_RCRB_GCS_TS) ? TRUE : FALSE;
}
VOID
PchPolicySetupInit (
IN CONST EFI_PEI_SERVICES **PeiServices,
IN SYSTEM_CONFIGURATION *SystemConfiguration
)
{
EFI_STATUS Status;
EFI_PEI_PPI_DESCRIPTOR *PchPlatformPolicyPpiDesc;
PCH_PLATFORM_POLICY_PPI *PchPlatformPolicyPpi;
PCH_HPET_CONFIG *HpetConfig;
PCH_PCIE_CONFIG *PcieConfig;
UINT8 Index;
PCH_IOAPIC_CONFIG *IoApicConfig;
PCH_LPSS_CONFIG *LpssConfig;
UINT32 SpiHsfsReg;
UINT32 SpiFdodReg;
//
// Disable codec ALC-262
//
UINT32 IoBase;
//
// Install Pch Platform Policy PPI. As we depend on Pch Init PPI so we are executed after
// PchInit PEIM. Thus we can insure PCH Initialization is performed when we install the Pch Platform Policy PPI,
// as PchInit PEIM registered a notification function on our policy PPI.
//
// --cr-- For better code structure / modularity, we should use a notification function on Pch Init PPI to perform
// actions that depend on PchInit PEIM's initialization.
//
//Todo: confirm if we need update to PCH_PLATFORM_POLICY_PPI_REVISION_5
//
DEBUG ((EFI_D_ERROR, "PchPolicySetupInit() - Start\n"));
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (EFI_PEI_PPI_DESCRIPTOR), (void **)&PchPlatformPolicyPpiDesc);
ASSERT_EFI_ERROR (Status);
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (PCH_PLATFORM_POLICY_PPI), (void **)&PchPlatformPolicyPpi);
ASSERT_EFI_ERROR (Status);
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (PCH_HPET_CONFIG), (void **)&HpetConfig);
ASSERT_EFI_ERROR (Status);
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (PCH_PCIE_CONFIG), (void **)&PcieConfig);
ASSERT_EFI_ERROR (Status);
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (PCH_IOAPIC_CONFIG), (void **)&IoApicConfig);
ASSERT_EFI_ERROR (Status);
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (PCH_LPSS_CONFIG), (void **)&LpssConfig);
ASSERT_EFI_ERROR (Status);
PchPlatformPolicyPpi->Revision = PCH_PLATFORM_POLICY_PPI_REVISION_1;
PchPlatformPolicyPpi->BusNumber = DEFAULT_PCI_BUS_NUMBER_PCH;
PchPlatformPolicyPpi->SpiBase = SPI_BASE_ADDRESS;
PchPlatformPolicyPpi->PmcBase = PMC_BASE_ADDRESS;
PchPlatformPolicyPpi->IoBase = IO_BASE_ADDRESS;
PchPlatformPolicyPpi->IlbBase = ILB_BASE_ADDRESS;
PchPlatformPolicyPpi->PUnitBase = PUNIT_BASE_ADDRESS;
PchPlatformPolicyPpi->MphyBase = MPHY_BASE_ADDRESS;
PchPlatformPolicyPpi->Rcba = RCBA_BASE_ADDRESS;
PchPlatformPolicyPpi->AcpiBase = ACPI_BASE_ADDRESS;
PchPlatformPolicyPpi->GpioBase = GPIO_BASE_ADDRESS;
PchPlatformPolicyPpi->SataMode = SystemConfiguration->SataType;
PchPlatformPolicyPpi->EnableRmh = SystemConfiguration->PchUsbRmh;
PchPlatformPolicyPpi->EhciPllCfgEnable = SystemConfiguration->EhciPllCfgEnable;
PchPlatformPolicyPpi->HpetConfig = HpetConfig;
PchPlatformPolicyPpi->PcieConfig = PcieConfig;
PchPlatformPolicyPpi->IoApicConfig = IoApicConfig;
PchPlatformPolicyPpi->HpetConfig->Enable = SystemConfiguration->Hpet;
PchPlatformPolicyPpi->HpetConfig->Base = HPET_BASE_ADDRESS;
PchPlatformPolicyPpi->IoApicConfig->IoApicId = 0x01;
//
// Set LPSS configuration according to setup value.
//
PchPlatformPolicyPpi->LpssConfig->LpssPciModeEnabled = SystemConfiguration->LpssPciModeEnabled;
PchPlatformPolicyPpi->LpssConfig->Dma1Enabled = SystemConfiguration->LpssDma1Enabled;
PchPlatformPolicyPpi->LpssConfig->I2C0Enabled = SystemConfiguration->LpssI2C0Enabled;
PchPlatformPolicyPpi->LpssConfig->I2C1Enabled = SystemConfiguration->LpssI2C1Enabled;
PchPlatformPolicyPpi->LpssConfig->I2C2Enabled = SystemConfiguration->LpssI2C2Enabled;
PchPlatformPolicyPpi->LpssConfig->I2C3Enabled = SystemConfiguration->LpssI2C3Enabled;
PchPlatformPolicyPpi->LpssConfig->I2C4Enabled = SystemConfiguration->LpssI2C4Enabled;
PchPlatformPolicyPpi->LpssConfig->I2C5Enabled = SystemConfiguration->LpssI2C5Enabled;
PchPlatformPolicyPpi->LpssConfig->I2C6Enabled = SystemConfiguration->LpssI2C6Enabled;
PchPlatformPolicyPpi->LpssConfig->Dma0Enabled = SystemConfiguration->LpssDma0Enabled;;
PchPlatformPolicyPpi->LpssConfig->Pwm0Enabled = SystemConfiguration->LpssPwm0Enabled;
PchPlatformPolicyPpi->LpssConfig->Pwm1Enabled = SystemConfiguration->LpssPwm1Enabled;
PchPlatformPolicyPpi->LpssConfig->Hsuart0Enabled = SystemConfiguration->LpssHsuart0Enabled;
PchPlatformPolicyPpi->LpssConfig->Hsuart1Enabled = SystemConfiguration->LpssHsuart1Enabled;
PchPlatformPolicyPpi->LpssConfig->SpiEnabled = SystemConfiguration->LpssSpiEnabled;
for (Index = 0; Index < PCH_PCIE_MAX_ROOT_PORTS; Index++) {
PchPlatformPolicyPpi->PcieConfig->PcieSpeed[Index] = SystemConfiguration->PcieRootPortSpeed[Index];
}
SpiHsfsReg = MmioRead32 (SPI_BASE_ADDRESS + R_PCH_SPI_HSFS);
if ((SpiHsfsReg & B_PCH_SPI_HSFS_FDV) == B_PCH_SPI_HSFS_FDV) {
MmioWrite32 (SPI_BASE_ADDRESS + R_PCH_SPI_FDOC, V_PCH_SPI_FDOC_FDSS_FSDM);
SpiFdodReg = MmioRead32 (SPI_BASE_ADDRESS + R_PCH_SPI_FDOD);
if (SpiFdodReg == V_PCH_SPI_FDBAR_FLVALSIG) {
}
//
// Disable codec ALC-262
//
if (SystemConfiguration->DisableCodec262 == 1) {
IoBase = MmioRead32 (MmPciAddress (0,
PchPlatformPolicyPpi->BusNumber,
PCI_DEVICE_NUMBER_PCH_LPC,
PCI_FUNCTION_NUMBER_PCH_LPC,
0
) + R_PCH_LPC_IO_BASE) & B_PCH_LPC_IO_BASE_BAR;
MmioAnd32 ((UINTN) (IoBase + 0x270), (UINT32) (~0x07));
}
}
PchPlatformPolicyPpiDesc->Flags = EFI_PEI_PPI_DESCRIPTOR_PPI | EFI_PEI_PPI_DESCRIPTOR_TERMINATE_LIST;
PchPlatformPolicyPpiDesc->Guid = &gPchPlatformPolicyPpiGuid;
PchPlatformPolicyPpiDesc->Ppi = PchPlatformPolicyPpi;
//
// Install PCH Platform Policy PPI
//
Status = (**PeiServices).InstallPpi (
PeiServices,
PchPlatformPolicyPpiDesc
);
ASSERT_EFI_ERROR (Status);
DEBUG ((EFI_D_ERROR, "PchPolicySetupInit() - End\n"));
}
EFI_STATUS
InstallPeiPchUsbPolicy (
IN CONST EFI_PEI_SERVICES **PeiServices
)
{
EFI_STATUS Status = EFI_SUCCESS;
EFI_PEI_PPI_DESCRIPTOR *PeiPchUsbPolicyPpiDesc;
PCH_USB_POLICY_PPI *PeiPchUsbPolicyPpi;
PCH_USB_CONFIG *UsbConfig;
DEBUG ((EFI_D_INFO, "InstallPeiPchUsbPolicy...\n"));
//
// Allocate descriptor and PPI structures. Since these are dynamically updated
// we cannot do a global variable PPI.
//
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (EFI_PEI_PPI_DESCRIPTOR), (void **)&PeiPchUsbPolicyPpiDesc);
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (PCH_USB_POLICY_PPI), (void **)&PeiPchUsbPolicyPpi);
Status = (*PeiServices)->AllocatePool (PeiServices, sizeof (PCH_USB_CONFIG), (void **)&UsbConfig);
//
// Initiate PCH USB policy.
//
PeiPchUsbPolicyPpi->Revision = PCH_USB_POLICY_PPI_REVISION_1;
UsbConfig->Usb20Settings[0].Enable = PCH_DEVICE_ENABLE;
UsbConfig->UsbPerPortCtl = PCH_DEVICE_DISABLE;
UsbConfig->Ehci1Usbr = PCH_DEVICE_DISABLE;
UsbConfig->Usb20OverCurrentPins[0] = PchUsbOverCurrentPin0;
UsbConfig->Usb20OverCurrentPins[1] = PchUsbOverCurrentPin0;
UsbConfig->Usb20OverCurrentPins[2] = PchUsbOverCurrentPin1;
UsbConfig->Usb20OverCurrentPins[3] = PchUsbOverCurrentPin1;
//
// Enable USB Topology control and program the topology setting for every USB port
// See Platform Design Guide for description of topologies
//
//
// Port 0: ~10.9", Port 1: ~10.1", Port 2: ~11.2", Port 3: ~11.5", Port 4: ~3.7", Port 5: ~2.7", Port 6: ~4.1"
// Port 7: ~4.5", Port 8: ~10.7", Port 9: ~10.5", Port 10: ~4.2", Port 11: ~4.3", Port 12: ~3.1", Port 13: ~2.9"
//
//
// Port 0: ~3.5", Port 1: ~4.1", Port 2: ~4.6", Port 3: ~4.6", Port 4: ~12.5", Port 5: ~12", Port 6: ~5.1"
// Port 7: ~5.1", Port 8: ~4.1", Port 9: ~4.1", Port 10: ~14.5", Port 11: ~12.8", Port 12: ~12.9", Port 13: ~14.6"
//
UsbConfig->Usb20PortLength[0] = 0x53;
UsbConfig->Usb20PortLength[1] = 0x49;
UsbConfig->Usb20PortLength[2] = 0x47;
UsbConfig->Usb20PortLength[3] = 0x80;
PeiPchUsbPolicyPpi->Mode = EHCI_MODE;
PeiPchUsbPolicyPpi->EhciMemBaseAddr = PcdGet32(PcdPeiIchEhciControllerMemoryBaseAddress);
PeiPchUsbPolicyPpi->EhciMemLength = (UINT32) 0x400 * PchEhciControllerMax;
PeiPchUsbPolicyPpi->XhciMemBaseAddr = 0;
PeiPchUsbPolicyPpi->UsbConfig = UsbConfig;
PeiPchUsbPolicyPpiDesc->Flags = EFI_PEI_PPI_DESCRIPTOR_PPI | EFI_PEI_PPI_DESCRIPTOR_TERMINATE_LIST;
PeiPchUsbPolicyPpiDesc->Guid = &gPchUsbPolicyPpiGuid;
PeiPchUsbPolicyPpiDesc->Ppi = PeiPchUsbPolicyPpi;
//
// Install PCH USB Policy PPI
//
Status = (**PeiServices).InstallPpi (PeiServices, PeiPchUsbPolicyPpiDesc);
return Status;
}