/** @file File to contain all the hardware specific stuff for the Smm Sx dispatch protocol. Copyright (c) 2017, Intel Corporation. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent **/ #include "PchSmmHelpers.h" #define PROGRESS_CODE_S3_SUSPEND_END PcdGet32 (PcdProgressCodeS3SuspendEnd) GLOBAL_REMOVE_IF_UNREFERENCED CONST PCH_SMM_SOURCE_DESC SX_SOURCE_DESC = { PCH_SMM_NO_FLAGS, { { { ACPI_ADDR_TYPE, {R_PCH_SMI_EN} }, S_PCH_SMI_EN, N_PCH_SMI_EN_ON_SLP_EN }, NULL_BIT_DESC_INITIALIZER }, { { { ACPI_ADDR_TYPE, {R_PCH_SMI_STS} }, S_PCH_SMI_STS, N_PCH_SMI_STS_ON_SLP_EN } }, { { ACPI_ADDR_TYPE, {R_PCH_SMI_STS} }, S_PCH_SMI_STS, N_PCH_SMI_STS_ON_SLP_EN } }; /** Get the Sleep type @param[in] Record No use @param[out] Context The context that includes SLP_TYP bits to be filled **/ VOID EFIAPI SxGetContext ( IN DATABASE_RECORD *Record, OUT PCH_SMM_CONTEXT *Context ) { UINT32 Pm1Cnt; Pm1Cnt = IoRead32 ((UINTN) (mAcpiBaseAddr + R_PCH_ACPI_PM1_CNT)); /// /// By design, the context phase will always be ENTRY /// Context->Sx.Phase = SxEntry; /// /// Map the PM1_CNT register's SLP_TYP bits to the context type /// switch (Pm1Cnt & B_PCH_ACPI_PM1_CNT_SLP_TYP) { case V_PCH_ACPI_PM1_CNT_S0: Context->Sx.Type = SxS0; break; case V_PCH_ACPI_PM1_CNT_S1: Context->Sx.Type = SxS1; break; case V_PCH_ACPI_PM1_CNT_S3: Context->Sx.Type = SxS3; break; case V_PCH_ACPI_PM1_CNT_S4: Context->Sx.Type = SxS4; break; case V_PCH_ACPI_PM1_CNT_S5: Context->Sx.Type = SxS5; break; default: ASSERT (FALSE); break; } } /** Check whether sleep type of two contexts match @param[in] Context1 Context 1 that includes sleep type 1 @param[in] Context2 Context 2 that includes sleep type 2 @retval FALSE Sleep types match @retval TRUE Sleep types don't match **/ BOOLEAN EFIAPI SxCmpContext ( IN PCH_SMM_CONTEXT *Context1, IN PCH_SMM_CONTEXT *Context2 ) { return (BOOLEAN) (Context1->Sx.Type == Context2->Sx.Type); } /** For each PCIE RP clear PME SCI status and disable SCI, then PCIEXP_WAKE_STS from PMC. This prevents platform from waking more than one time due to a single PCIE wake event. Normally it's up to OS to clear SCI statuses. But in a scenario where platform wakes and goes to S5 instead of booting to OS, the SCI status would remain set and would trigger another wake. **/ VOID ClearPcieSci ( VOID ) { UINT32 MaxPorts; UINT32 RpIndex; UINTN RpBase; MaxPorts = GetPchMaxPciePortNum (); for (RpIndex = 0; RpIndex < MaxPorts; RpIndex++) { RpBase = PchPcieBase (RpIndex); if (MmioRead16 (RpBase + PCI_VENDOR_ID_OFFSET) != 0xFFFF) { MmioAnd8 ((RpBase + R_PCH_PCIE_MPC + 3), (UINT8)~((UINT8)(B_PCH_PCIE_MPC_PMCE >> 24))); MmioWrite32 (RpBase + R_PCH_PCIE_SMSCS, B_PCH_PCIE_SMSCS_PMCS); } } IoWrite16 (mAcpiBaseAddr + R_PCH_ACPI_PM1_STS, B_PCH_ACPI_PM1_STS_PCIEXP_WAKE_STS); } /** When we get an SMI that indicates that we are transitioning to a sleep state, we need to actually transition to that state. We do this by disabling the "SMI on sleep enable" feature, which generates an SMI when the operating system tries to put the system to sleep, and then physically putting the system to sleep. **/ VOID PchSmmSxGoToSleep ( VOID ) { UINT32 Pm1Cnt; UINT32 PchPwrmBase; ClearPcieSci (); PchPwrmBaseGet (&PchPwrmBase); /// /// Flush cache into memory before we go to sleep. It is necessary for S3 sleep /// because we may update memory in SMM Sx sleep handlers -- the updates are in cache now /// AsmWbinvd (); /// /// Disable SMIs /// PchSmmClearSource (&SX_SOURCE_DESC); PchSmmDisableSource (&SX_SOURCE_DESC); /// /// Get Power Management 1 Control Register Value /// Pm1Cnt = IoRead32 ((UINTN) (mAcpiBaseAddr + R_PCH_ACPI_PM1_CNT)); /// /// Record S3 suspend performance data /// if ((Pm1Cnt & B_PCH_ACPI_PM1_CNT_SLP_TYP) == V_PCH_ACPI_PM1_CNT_S3) { /// /// Report status code before goto S3 sleep /// REPORT_STATUS_CODE (EFI_PROGRESS_CODE, PROGRESS_CODE_S3_SUSPEND_END); /// /// Flush cache into memory before we go to sleep. /// AsmWbinvd (); } /// /// Now that SMIs are disabled, write to the SLP_EN bit again to trigger the sleep /// Pm1Cnt |= B_PCH_ACPI_PM1_CNT_SLP_EN; IoWrite32 ((UINTN) (mAcpiBaseAddr + R_PCH_ACPI_PM1_CNT), Pm1Cnt); /// /// Should only proceed if wake event is generated. /// if ((Pm1Cnt & B_PCH_ACPI_PM1_CNT_SLP_TYP) == V_PCH_ACPI_PM1_CNT_S1) { while (((IoRead16 ((UINTN) (mAcpiBaseAddr + R_PCH_ACPI_PM1_STS))) & B_PCH_ACPI_PM1_STS_WAK) == 0x0); } else { CpuDeadLoop (); } /// /// The system just went to sleep. If the sleep state was S1, then code execution will resume /// here when the system wakes up. /// Pm1Cnt = IoRead32 ((UINTN) (mAcpiBaseAddr + R_PCH_ACPI_PM1_CNT)); if ((Pm1Cnt & B_PCH_ACPI_PM1_CNT_SCI_EN) == 0) { /// /// An ACPI OS isn't present, clear the sleep information /// Pm1Cnt &= ~B_PCH_ACPI_PM1_CNT_SLP_TYP; Pm1Cnt |= V_PCH_ACPI_PM1_CNT_S0; IoWrite32 ((UINTN) (mAcpiBaseAddr + R_PCH_ACPI_PM1_CNT), Pm1Cnt); } PchSmmClearSource (&SX_SOURCE_DESC); PchSmmEnableSource (&SX_SOURCE_DESC); }