/** @file
PCH DMI library.
Copyright (c) 2019 Intel Corporation. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent
**/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "PchDmi14.h"
#include "PchDmi15.h"
/**
This function checks if DMI Secured Register Lock (SRL) is set
@retval SRL state
**/
BOOLEAN
IsPchDmiLocked (
VOID
)
{
if (IsPchWithPdmi ()) {
return IsPchDmi15Locked ();
} else {
return IsPchDmi14Locked ();
}
}
/**
Backward DMI library API compatibility
ACPI base address programming is done in PSF
@param[in] Address Address for ACPI base.
@retval EFI_UNSUPPORTED NOT supported programming.
**/
EFI_STATUS
PchDmiSetAcpiBase (
IN UINT16 Address
)
{
return EFI_UNSUPPORTED;
}
/**
Backward DMI library API compatibility
PWRMBASE is a standard BAR and doesn't require
additional DMI base decoding programming
@param[in] Address Address for PWRM base.
@retval EFI_UNSUPPORTED NOT supported programming.
**/
EFI_STATUS
PchDmiSetPwrmBase (
IN UINT32 Address
)
{
return EFI_UNSUPPORTED;
}
/**
Set PCH TCO base address decoding in DMI
@param[in] Address Address for TCO base address.
@retval EFI_SUCCESS Successfully completed.
@retval EFI_INVALID_PARAMETER Invalid base address passed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiSetTcoBase (
IN UINT16 Address
)
{
if (IsPchDmiLocked ()) {
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
//
// Program "TCO Base Address" PCR[DMI] + 2778h[15:5, 1] to [SMBUS PCI offset 50h[15:5], 1].
//
PchPcrWrite16 (
PID_DMI, R_PCH_DMI_PCR_TCOBASE,
(Address | BIT1)
);
return EFI_SUCCESS;
}
/**
Get PCH TCO base address.
@retval Address Address of TCO base address.
**/
UINT16
PchDmiGetTcoBase (
VOID
)
{
//
// Read "TCO Base Address" PCR[DMI] + 2778h[15:5]
//
return (PchPcrRead16 (PID_DMI, R_PCH_DMI_PCR_TCOBASE) & B_PCH_DMI_PCR_TCOBASE_TCOBA);
}
/**
Set PCH LPC/eSPI generic IO range decoding in DMI
@param[in] Address Address for generic IO range base address.
@param[in] Length Length of generic IO range.
@param[in] RangeIndex Index of choosen range
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiSetLpcGenIoRange (
IN UINT32 Address,
IN UINT32 Length,
IN UINT32 RangeIndex
)
{
UINT32 Data32;
//
// This cycle decoding is only allowed to set when DMIC.SRL is 0.
//
if (IsPchDmiLocked ()) {
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
Data32 = (UINT32) (((Length - 1) << 16) & B_LPC_CFG_GENX_DEC_IODRA);
Data32 |= (UINT32) Address;
Data32 |= B_LPC_CFG_GENX_DEC_EN;
//
// Program LPC Generic IO Range #, PCR[DMI] + 2730h ~ 273Fh to the same value programmed in LPC/eSPI PCI Offset 84h~93h.
//
PchPcrWrite32 (
PID_DMI, (UINT16) (R_PCH_DMI_PCR_LPCLGIR1 + RangeIndex * 4),
Data32
);
return EFI_SUCCESS;
}
/**
Set PCH eSPI eSPI CS1# generic IO range decoding in DMI
@param[in] Address Address for generic IO range base address.
@param[in] Length Length of generic IO range.
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiSetEspiCs1GenIoRange (
IN UINT32 Address,
IN UINT32 Length
)
{
UINT32 Data32;
//
// This cycle decoding is only allowed to set when DMIC.SRL is 0.
//
if (IsPchDmiLocked ()) {
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
Data32 = (UINT32) (((Length - 1) << 16) & B_LPC_CFG_GENX_DEC_IODRA);
Data32 |= (UINT32) Address;
Data32 |= B_LPC_CFG_GENX_DEC_EN;
//
// Program eSPI Generic IO Range #, PCR[DMI] + 27BCh to the same value programmed in eSPI PCI Offset A4h.
//
PchPcrWrite32 (PID_DMI, R_PCH_DMI_PCR_SEGIR, Data32);
return EFI_SUCCESS;
}
/**
Clear PCH LPC/eSPI generic IO range decoding in DMI
@param[in] RangeIndex Index of chosen range
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiClearLpcGenIoRange (
IN UINTN RangeIndex
)
{
//
// This cycle decoding is only allowed to set when DMIC.SRL is 0.
//
if (IsPchDmiLocked ()) {
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
//
// Program LPC Generic IO Range #, PCR[DMI] + 2730h ~ 273Fh to the same value programmed in LPC/eSPI PCI Offset 84h~93h.
//
PchPcrWrite32 (
PID_DMI, (UINT16) (R_PCH_DMI_PCR_LPCLGIR1 + RangeIndex * 4),
0
);
return EFI_SUCCESS;
}
/**
Clear PCH eSPI CS1# generic IO range decoding in DMI
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiClearEspiCs1GenIoRange (
VOID
)
{
//
// This cycle decoding is only allowed to set when DMIC.SRL is 0.
//
if (IsPchDmiLocked ()) {
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
//
// Program LPC Generic IO Range #, PCR[DMI] + 27BCh to the same value programmed in eSPI PCI Offset A4h.
//
PchPcrWrite32 (PID_DMI, R_PCH_DMI_PCR_SEGIR, 0);
return EFI_SUCCESS;
}
/**
Set PCH LPC/eSPI memory range decoding in DMI
@param[in] Address Address for memory base address.
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiSetLpcMemRange (
IN UINT32 Address
)
{
if (IsPchDmiLocked ()) {
DEBUG ((DEBUG_ERROR, "PchDmiSetLpcMemRange Error. DMI is locked.\n"));
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
//
// Program LPC Memory Range, PCR[DMI] + 2740h to the same value programmed in LPC/eSPI PCI Offset 98h.
//
PchPcrWrite32 (
PID_DMI, R_PCH_DMI_PCR_LPCGMR,
(Address | B_LPC_CFG_LGMR_LMRD_EN)
);
return EFI_SUCCESS;
}
/**
Set PCH eSPI CS1# memory range decoding in DMI
@param[in] Address Address for memory base address.
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiSetEspiCs1MemRange (
IN UINT32 Address
)
{
if (IsPchDmiLocked ()) {
DEBUG ((DEBUG_ERROR, "PchLpcMemRange2Set Error. DMI is locked.\n"));
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
//
// Program LPC Memory Range, PCR[DMI] + 27C0h to the same value programmed in eSPI PCI Offset A8h.
//
PchPcrWrite32 (
PID_DMI, R_PCH_DMI_PCR_SEGMR,
(Address | B_LPC_CFG_LGMR_LMRD_EN)
);
return EFI_SUCCESS;
}
/**
Check if Boot BIOS Strap is set for SPI.
@retval TRUE Boot BIOS Strap set for SPI
@retval FALSE Boot BIOS Strap set for LPC/eSPI
**/
BOOLEAN
PchDmiIsBootBiosStrapSetForSpi (
VOID
)
{
//
// Check General Control and Status (GCS) [10]
// '0': SPI
// '1': LPC/eSPI
//
return ((PchPcrRead32 (PID_DMI, R_PCH_DMI_PCR_GCS) & B_PCH_DMI_PCR_BBS) != B_PCH_DMI_PCR_BBS);
}
/**
Set PCH BIOS range decoding in DMI
Please check EDS for detail of BiosDecodeEnable bit definition.
bit 15: F8-FF Enable
bit 14: F0-F8 Enable
bit 13: E8-EF Enable
bit 12: E0-E8 Enable
bit 11: D8-DF Enable
bit 10: D0-D7 Enable
bit 9: C8-CF Enable
bit 8: C0-C7 Enable
bit 7: Legacy F Segment Enable
bit 6: Legacy E Segment Enable
bit 5: Reserved
bit 4: Reserved
bit 3: 70-7F Enable
bit 2: 60-6F Enable
bit 1: 50-5F Enable
bit 0: 40-4F Enable
@param[in] BiosDecodeEnable Bios decode enable setting.
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiSetBiosDecodeEnable (
IN UINT16 BiosDecodeEnable
)
{
if (IsPchDmiLocked ()) {
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
//
// program LPC BIOS Decode Enable, PCR[DMI] + 2744h to the same value programmed in LPC or SPI Offset D8h.
//
PchPcrWrite16 (PID_DMI, R_PCH_DMI_PCR_LPCBDE, BiosDecodeEnable);
return EFI_SUCCESS;
}
/**
Set PCH LPC/eSPI IO decode ranges in DMI
Please check EDS for detail of LPC/eSPI IO decode ranges bit definition.
Bit 12: FDD range
Bit 9:8: LPT range
Bit 6:4: ComB range
Bit 2:0: ComA range
@param[in] LpcIoDecodeRanges LPC/eSPI IO decode ranges bit settings.
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiSetLpcIoDecodeRanges (
IN UINT16 LpcIoDecodeRanges
)
{
//
// This cycle decoding is only allowed to set when DMI is not locked.
//
if (IsPchDmiLocked ()) {
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
//
// program LPC I/O Decode Ranges, PCR[DMI] + 2770h[15:0] to the same value programmed in LPC/eSPI PCI offset 80h.
//
PchPcrWrite16 (PID_DMI, R_PCH_DMI_PCR_LPCIOD, LpcIoDecodeRanges);
return EFI_SUCCESS;
}
/**
Set PCH LPC/eSPI IO enable decoding in DMI
@param[in] LpcIoEnableDecoding LPC/eSPI IO enable decoding bit settings.
@retval EFI_SUCCESS Successfully completed.
@retval EFI_UNSUPPORTED DMIC.SRL is set.
**/
EFI_STATUS
PchDmiSetLpcIoEnable (
IN UINT16 LpcIoEnableDecoding
)
{
//
// This cycle decoding is only allowed to set when DMI is not locked.
//
if (IsPchDmiLocked ()) {
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
//
// program LPC I/O Decode Ranges, PCR[DMI] + 2774h[15:0] to the same value programmed in LPC/eSPI PCI offset 82h.
//
PchPcrWrite16 (PID_DMI, R_PCH_DMI_PCR_LPCIOE, LpcIoEnableDecoding);
return EFI_SUCCESS;
}
/**
Set PCH IO port 80h cycle decoding to PCIE root port in DMI
@param[in] RpNumber PCIE root port physical number.
@retval EFI_SUCCESS Successfully completed.
**/
EFI_STATUS
PchDmiSetIoPort80Decode (
IN UINTN RpNumber
)
{
UINT16 DmiRpDestinationId;
PSF_PORT_DEST_ID PsfRpDestinationId;
if (IsPchDmiLocked ()) {
DEBUG ((DEBUG_ERROR, "PchIoPort80DecodeSet Error. DMI is locked.\n"));
ASSERT (FALSE);
return EFI_UNSUPPORTED;
}
///
/// IO port 80h is typically used by decoder/LED hardware for debug purposes.
/// By default PCH will forward IO port 80h cycles to LPC bus. The Reserved Page Route (RPR) bit
/// of General Control and Status register, located at PCR[DMI] + 274Ch[11] , allows software to
/// re-direct IO port 80h cycles to PCIe bus so that a target (for example, a debug card) on
/// PCIe bus can receive and claim these cycles.
/// The "RPR Destination ID", PCR[DMI] + 274Ch[31:16] need to be set accordingly to point
/// to the root port that decode this range. Reading from Port 80h may not return valid values
/// if the POST-card itself do not shadow the writes. Unlike LPC, PCIe does not shadow the Port 80 writes.
///
PsfRpDestinationId = PsfPcieDestinationId ((UINT32)RpNumber);
DmiRpDestinationId = (UINT16)((0x2 << 12) |
(PsfRpDestinationId.Fields.PsfId << 8) |
(PsfRpDestinationId.Fields.PortGroupId << 7) |
(PsfRpDestinationId.Fields.PortId << 3) |
PsfRpDestinationId.Fields.ChannelId);
//
// Program "RPR Destination ID", PCR[DMI] + 274Ch[31:16] to the Dest ID of RP.
//
PchPcrWrite16 (PID_DMI, R_PCH_DMI_PCR_GCS + 2, DmiRpDestinationId);
//
// Program "Reserved Page Route", PCR[DMI] + 274Ch[11] to '1'.
// Use byte write on GCS+1 and leave the BILD bit which is RWO.
//
PchPcrAndThenOr8 (PID_DMI, R_PCH_DMI_PCR_GCS + 1, 0xFF, (B_PCH_DMI_PCR_RPR >> 8));
return EFI_SUCCESS;
}
/**
Set DMI thermal throttling to recommended configuration.
It's intended only for P-DMI.
**/
VOID
PchDmiSetRecommendedThermalThrottling (
VOID
)
{
if (IsPchWithPdmi ()) {
PchDmi15SetRecommendedThermalThrottling ();
}
}
/**
Set DMI thermal throttling to custom configuration.
This function will configure Thermal Sensor 0/1/2/3 TargetWidth and set
DMI Thermal Sensor Autonomous Width Enable.
It's intended only for P-DMI.
@param[in] DmiThermalThrottling DMI Thermal Throttling structure.
**/
VOID
PchDmiSetCustomThermalThrottling (
IN DMI_THERMAL_THROTTLING DmiThermalThrottling
)
{
if (IsPchWithPdmi ()) {
PchDmi15SetCustomThermalThrottling (DmiThermalThrottling);
}
}
/**
Determines where to send the reserved page registers
Accesses to the I/O ranges 80h - 8Fh will be forwarded to PCIe Root Port
with the destination ID specified in GCS.RPRDID using DMI source decode.
**/
VOID
PchDmiSetReservedPageRegToPcieRootPort (
VOID
)
{
PchPcrAndThenOr8 (
PID_DMI, R_PCH_DMI_PCR_GCS + 1,
(UINT8) ~0,
(UINT8) (B_PCH_DMI_PCR_RPR >> 8)
);
}
/**
Determines where to send the reserved page registers
DMI will not perform source decode on the I/O ranges 80h - 8Fh. The cycles hitting these ranges will
end up in P2SB which will then forward the cycle to LPC or eSPI through IOSF Sideband.
**/
VOID
PchDmiSetReservedPageRegToLpc (
VOID
)
{
PchPcrAndThenOr8 (
PID_DMI, R_PCH_DMI_PCR_GCS + 1,
(UINT8) (~(B_PCH_DMI_PCR_RPR >> 8)),
0
);
}
/**
uCode Patch Region Enable (UPRE). Enables memory access targeting the uCode patch region (0xFEF00000 to 0xFEFFFFFF)
to be forwarded to SPI Flash. This can only be set if the boot flash is on SPI.
**/
VOID
PchDmiEnableUCodePatchRegion (
VOID
)
{
///
/// Setup "uCode Patch Region Enable", PCR [DMI] + 2748h[0] to '0b'
///
PchPcrAndThenOr32 (PID_DMI, R_PCH_DMI_PCR_UCPR, (UINT32) ~B_PCH_DMI_PCR_UCPR_UPRE, 0);
}