/** @file Header file for PCH PCI Express helpers library Copyright (c) 2019 Intel Corporation. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent **/ #ifndef _PCH_PCI_EXPRESS_HELPERS_LIB_H_ #define _PCH_PCI_EXPRESS_HELPERS_LIB_H_ #include typedef enum { TpoScale2us, TpoScale10us, TpoScale100us, TpoScaleMax } T_PO_SCALE; typedef struct { UINT32 Value; T_PO_SCALE Scale; } T_POWER_ON; // // Function prototypes // /** Get PCIe port number for enabled port. @param[in] RpBase Root Port pci segment base address @return Root Port number (1 based) **/ UINT32 PciePortNum ( IN UINT64 RpBase ); /** Get PCIe root port index @param[in] RpBase Root Port pci segment base address @return Root Port index (0 based) **/ UINT32 PciePortIndex ( IN UINT64 RpBase ); /** Translate PCIe Port/Lane pair to 0-based PCIe lane number. @param[in] RpIndex Root Port index @param[in] RpLane Root Port Lane (0-3) @retval PCIe lane number (0-based) **/ UINT32 PchPciePhysicalLane ( UINT32 RpIndex, UINT32 RpLane ); /** Checks if lane reversal is enabled on a given root port @param[in] RpIndex Root port index (0-based) @retval TRUE if lane reversal is enbabled, FALSE otherwise **/ BOOLEAN IsPcieLaneReversalEnabled ( IN UINT32 RpIndex ); /** Calculates the index of the first port on the same controller. @param[in] RpIndex Root Port Number (0-based) @retval Index of the first port on the first controller. **/ UINT32 PchGetPcieFirstPortIndex ( IN UINT32 RpIndex ); /* Returns Tpower_on capability of device @param[in] DeviceBase device's PCI segment base address @param[in] L1ssCapOffset offset to L1substates capability in device's extended config space @retval structure containing Tpoweron scale and value */ T_POWER_ON GetTpoCapability ( UINT64 DeviceBase, UINT32 L1ssCapOffset ); /* Converts Tpower_on from value:scale notation to microseconds @param[in] TpoScale T power on scale @param[in] TpoValue T power on value @retval number of microseconds */ UINT32 TpoToUs ( UINT32 TpoScale, UINT32 TpoValue ); /** Find the Offset to a given Capabilities ID CAPID list: 0x01 = PCI Power Management Interface 0x04 = Slot Identification 0x05 = MSI Capability 0x10 = PCI Express Capability @param[in] DeviceBase device's base address @param[in] CapId CAPID to search for @retval 0 CAPID not found @retval Other CAPID found, Offset of desired CAPID **/ UINT8 PcieBaseFindCapId ( IN UINT64 DeviceBase, IN UINT8 CapId ); /** Find the Offset to a given Capabilities ID CAPID list: 0x01 = PCI Power Management Interface 0x04 = Slot Identification 0x05 = MSI Capability 0x10 = PCI Express Capability @param[in] Segment Pci Segment Number @param[in] Bus Pci Bus Number @param[in] Device Pci Device Number @param[in] Function Pci Function Number @param[in] CapId CAPID to search for @retval 0 CAPID not found @retval Other CAPID found, Offset of desired CAPID **/ UINT8 PcieFindCapId ( IN UINT8 Segment, IN UINT8 Bus, IN UINT8 Device, IN UINT8 Function, IN UINT8 CapId ); /** Search and return the offset of desired Pci Express Capability ID CAPID list: 0x0001 = Advanced Error Reporting Capability 0x0002 = Virtual Channel Capability 0x0003 = Device Serial Number Capability 0x0004 = Power Budgeting Capability @param[in] DeviceBase device base address @param[in] CapId Extended CAPID to search for @retval 0 CAPID not found, this includes situation where device doesn't exist @retval Other CAPID found, Offset of desired CAPID **/ UINT16 PcieBaseFindExtendedCapId ( IN UINT64 DeviceBase, IN UINT16 CapId ); /** Search and return the offset of desired Pci Express Capability ID CAPID list: 0x0001 = Advanced Error Rreporting Capability 0x0002 = Virtual Channel Capability 0x0003 = Device Serial Number Capability 0x0004 = Power Budgeting Capability @param[in] Segment Pci Segment Number @param[in] Bus Pci Bus Number @param[in] Device Pci Device Number @param[in] Function Pci Function Number @param[in] CapId Extended CAPID to search for @retval 0 CAPID not found @retval Other CAPID found, Offset of desired CAPID **/ UINT16 PcieFindExtendedCapId ( IN UINT8 Segment, IN UINT8 Bus, IN UINT8 Device, IN UINT8 Function, IN UINT16 CapId ); /* Checks device's Slot Clock Configuration @param[in] Base device's base address @retval TRUE when device device uses slot clock, FALSE otherwise */ BOOLEAN GetScc ( UINT64 Base, UINT8 PcieCapOffset ); /* Sets Common Clock Configuration bit for given device. @param[in] Base device's base address */ VOID EnableCcc ( UINT64 Base, UINT8 PcieCapOffset ); /* Retrains link behind given device. It only makes sense to call it for downstream ports. If called for upstream port nothing will happen, it won't enter infinite loop. @param[in] Base device's base address */ VOID RetrainLink ( UINT64 Base, UINT8 PcieCapOffset, BOOLEAN WaitUntilDone ); /* Checks if device at given address exists @retval TRUE when device exists; FALSE otherwise */ BOOLEAN IsDevicePresent ( UINT64 Base ); /* Checks if device is a multifunction device @param[in] Base device's base address @retval TRUE if multifunction; FALSE otherwise */ BOOLEAN IsMultifunctionDevice ( UINT64 Base ); /* Initializes the following features in rootport and devices behind it: Maximum Payload Size (generic) Rootport packet split (proprietary) EonOfInterrupt forwarding (proprietary) Common Clock Configuration (generic) Generic: any code written according to PCIE Express base specification can do that. Proprietary: code uses registers and features that are specific to Intel silicon and probably only this Reference Code knows how to handle that. If OEM implemented generic feature enabling in his platform code or trusts Operating System to do it, then those features can be deleted from here. CCC requires link retrain, which takes a while. CCC must happen before L0s/L1 programming. If there was guarantee no code would access PCI while links retrain, it would be possible to skip this waiting @param[in] RpSegment address of rootport on PCIe @param[in] RpBus address of rootport on PCIe @param[in] RpDevice address of rootport on PCIe @param[in] RpFunction address of rootport on PCIe @param[in] BusMin minimum Bus number that can be assigned below this rootport @param[in] BusMax maximum Bus number that can be assigned below this rootport */ VOID RootportDownstreamConfiguration ( UINT8 RpSegment, UINT8 RpBus, UINT8 RpDevice, UINT8 RpFunction, UINT8 BusMin, UINT8 BusMax ); /* Configures the following power-management related features in rootport and devices behind it: LTR limit (generic) LTR override (proprietary) Clock Power Management (generic) L1 substates (generic except for the override table) L1.LOW substate (proprietary) L0s and L1 (generic) Generic: any code written according to PCIE Express base specification can do that. Proprietary: code uses registers and features that are specific to Intel silicon and probably only this Reference Code knows how to handle that. If OEM implemented generic feature enabling in his platform code or trusts Operating System to do it, then those features can be deleted from here. @param[in] RpSegment address of rootport on PCIe @param[in] RpBus address of rootport on PCIe @param[in] RpDevice address of rootport on PCIe @param[in] RpFunction address of rootport on PCIe @param[in] BusLimit maximum Bus number that can be assigned below this rootport @param[in] AspmOverrideTableSize size of override array @param[in] AspmOverrideTable array of device that need exceptions in configuration */ VOID RootportDownstreamPmConfiguration ( UINT8 RpSegment, UINT8 RpBus, UINT8 RpDevice, UINT8 RpFunction, UINT8 BusMin, UINT8 BusMax, PCH_PCIE_ROOT_PORT_CONFIG *RpConfig, UINT32 AspmOverrideTableSize, PCH_PCIE_DEVICE_OVERRIDE *AspmOverrideTable ); /** Get current PCIe link speed. @param[in] RpBase Root Port base address @return Link speed **/ UINT32 GetLinkSpeed ( UINT64 RpBase ); /** Get max PCIe link speed supported by the root port. @param[in] RpBase Root Port pci segment base address @return Max link speed **/ UINT32 GetMaxLinkSpeed ( UINT64 RpBase ); /** PCIe controller configuration. **/ typedef enum { Pcie4x1 = 0, Pcie1x2_2x1 = 1, Pcie2x2 = 2, Pcie1x4 = 3 } PCIE_CONTROLLER_CONFIG; #endif // _PEI_DXE_SMM_PCH_PCI_EXPRESS_HELPERS_LIB_H_