/** @file PCH PCR library. All function in this library is available for PEI, DXE, and SMM, But do not support UEFI RUNTIME environment call. Copyright (c) 2021, Intel Corporation. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent **/ #include #include #include #include #include #include #include #include #ifndef MDEPKG_NDEBUG /** Checks if the offset is valid for a given memory access width. Offset must align to width size. @param[in] Offset Offset of a register @param[in] Size Size of memory access in bytes @retval FALSE Offset is not valid for a given memory access @retval TRUE Offset is valid **/ STATIC BOOLEAN PchIsPcrOffsetValid ( IN UINT32 Offset, IN UINTN Size ) { if (!IsP2sb20bPcrSupported ()) { if (((Offset & (Size - 1)) != 0) || (Offset > 0xFFFF)) { DEBUG ((DEBUG_ERROR, "PCR offset error. Invalid Offset: %x Size: %x", Offset, Size)); return FALSE; } else { return TRUE; } } else { if (((Offset & (Size - 1)) != 0) || (Offset > 0xFFFFF)) { DEBUG ((DEBUG_ERROR, "PCR offset error. Invalid Offset: %x Size: %x", Offset, Size)); return FALSE; } else { return TRUE; } } } #endif /** Read PCR register. It returns PCR register and size in 4bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of this Port ID @retval UINT32 PCR register value. **/ UINT32 PchPcrRead32 ( IN PCH_SBI_PID Pid, IN UINT32 Offset ) { #ifndef MDEPKG_NDEBUG ASSERT (PchIsPcrOffsetValid (Offset, 4)); #endif return MmioRead32 (PCH_PCR_ADDRESS (Pid, Offset)); } /** Read PCR register. It returns PCR register and size in 2bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of this Port ID @retval UINT16 PCR register value. **/ UINT16 PchPcrRead16 ( IN PCH_SBI_PID Pid, IN UINT32 Offset ) { #ifndef MDEPKG_NDEBUG ASSERT (PchIsPcrOffsetValid (Offset, 2)); #endif return MmioRead16 (PCH_PCR_ADDRESS (Pid, Offset)); } /** Read PCR register. It returns PCR register and size in 1bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of this Port ID @retval UINT8 PCR register value **/ UINT8 PchPcrRead8 ( IN PCH_SBI_PID Pid, IN UINT32 Offset ) { return MmioRead8 (PCH_PCR_ADDRESS (Pid, Offset)); } /** Write PCR register. It programs PCR register and size in 4bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of Port ID. @param[in] Data Input Data. Must be the same size as Size parameter. @retval UINT32 Value written to register **/ UINT32 PchPcrWrite32 ( IN PCH_SBI_PID Pid, IN UINT32 Offset, IN UINT32 Data ) { #ifndef MDEPKG_NDEBUG ASSERT (PchIsPcrOffsetValid (Offset, 4)); #endif MmioWrite32 (PCH_PCR_ADDRESS (Pid, Offset), Data); return Data; } /** Write PCR register. It programs PCR register and size in 2bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of Port ID. @param[in] Data Input Data. Must be the same size as Size parameter. @retval UINT16 Value written to register **/ UINT16 PchPcrWrite16 ( IN PCH_SBI_PID Pid, IN UINT32 Offset, IN UINT16 Data ) { #ifndef MDEPKG_NDEBUG ASSERT (PchIsPcrOffsetValid (Offset, 2)); #endif MmioWrite16 (PCH_PCR_ADDRESS (Pid, Offset), Data); return Data; } /** Write PCR register. It programs PCR register and size in 1bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of Port ID. @param[in] Data Input Data. Must be the same size as Size parameter. @retval UINT8 Value written to register **/ UINT8 PchPcrWrite8 ( IN PCH_SBI_PID Pid, IN UINT32 Offset, IN UINT8 Data ) { MmioWrite8 (PCH_PCR_ADDRESS (Pid, Offset), Data); return Data; } /** Write PCR register. It programs PCR register and size in 4bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of Port ID. @param[in] AndData AND Data. Must be the same size as Size parameter. @param[in] OrData OR Data. Must be the same size as Size parameter. @retval UINT32 Value written to register **/ UINT32 PchPcrAndThenOr32 ( IN PCH_SBI_PID Pid, IN UINT32 Offset, IN UINT32 AndData, IN UINT32 OrData ) { return PchPcrWrite32 (Pid, Offset, (PchPcrRead32 (Pid, Offset) & AndData) | OrData); } /** Write PCR register and read back. The read back ensures the PCR cycle is completed before next operation. It programs PCR register and size in 4bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of Port ID. @param[in] AndData AND Data. Must be the same size as Size parameter. @param[in] OrData OR Data. Must be the same size as Size parameter. @retval UINT32 Value read back from the register **/ UINT32 PchPcrAndThenOr32WithReadback ( IN PCH_SBI_PID Pid, IN UINT32 Offset, IN UINT32 AndData, IN UINT32 OrData ) { PchPcrWrite32 (Pid, Offset, (PchPcrRead32 (Pid, Offset) & AndData) | OrData); return PchPcrRead32 (Pid, Offset); } /** Write PCR register. It programs PCR register and size in 2bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of Port ID. @param[in] AndData AND Data. Must be the same size as Size parameter. @param[in] OrData OR Data. Must be the same size as Size parameter. @retval UINT16 Value written to register **/ UINT16 PchPcrAndThenOr16 ( IN PCH_SBI_PID Pid, IN UINT32 Offset, IN UINT16 AndData, IN UINT16 OrData ) { return PchPcrWrite16 (Pid, Offset, (PchPcrRead16 (Pid, Offset) & AndData) | OrData); } /** Write PCR register. It programs PCR register and size in 1bytes. The Offset should not exceed 0xFFFF and must be aligned with size. @param[in] Pid Port ID @param[in] Offset Register offset of Port ID. @param[in] AndData AND Data. Must be the same size as Size parameter. @param[in] OrData OR Data. Must be the same size as Size parameter. @retval UINT8 Value written to register **/ UINT8 PchPcrAndThenOr8 ( IN PCH_SBI_PID Pid, IN UINT32 Offset, IN UINT8 AndData, IN UINT8 OrData ) { return PchPcrWrite8 (Pid, Offset, (PchPcrRead8 (Pid, Offset) & AndData) | OrData); } /** Get PCH IP PID number @param[in] IpEnum PCH IP in PCH_IP_PID_ENUM @retval 0 PID of this IP is not supported !0 PID of the IP. **/ PCH_SBI_PID PchPcrGetPid ( PCH_IP_PID_ENUM IpEnum ) { switch (IpEnum) { case PchIpDmi: return PID_DMI; case PchIpIclk: return PID_ICLK; default: ASSERT (FALSE); return PCH_INVALID_PID; } }