| |
| /** |
| * @file |
| * |
| * Config Southbridge GPP controller |
| * |
| * Init GPP features. |
| * |
| * @xrefitem bom "File Content Label" "Release Content" |
| * @e project: CIMx-SB |
| * @e sub-project |
| * @e \$Revision:$ @e \$Date:$ |
| * |
| */ |
| /* |
| ***************************************************************************** |
| * |
| * Copyright (c) 2011, Advanced Micro Devices, Inc. |
| * All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are met: |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * * Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * * Neither the name of Advanced Micro Devices, Inc. nor the names of |
| * its contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND |
| * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL ADVANCED MICRO DEVICES, INC. BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
| * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| * |
| * *************************************************************************** |
| * |
| */ |
| |
| #include "SBPLATFORM.h" |
| #include "cbtypes.h" |
| |
| /** |
| * PCIE_CAP_ID - PCIe Cap ID |
| * |
| */ |
| #define PCIE_CAP_ID 0x10 |
| |
| // |
| // Declaration of local functions |
| // |
| |
| /** |
| * PreInitGppLink - Enable GPP link training. |
| * |
| * @param[in] pConfig Southbridge configuration structure pointer. |
| * |
| */ |
| VOID PreInitGppLink (IN AMDSBCFG* pConfig); |
| UINT8 CheckGppLinkStatus (IN AMDSBCFG* pConfig); |
| VOID AfterGppLinkInit (IN AMDSBCFG* pConfig); |
| VOID sbGppForceGen2 (IN UINT32 portId ); |
| VOID sbGppForceGen1 (IN UINT32 portId ); |
| VOID sbGppDisableUnusedPadMap (IN AMDSBCFG* pConfig ); |
| VOID sbGppSetAspm (IN UINT32 pciAddress, IN UINT8 LxState); |
| UINT8 sbFindPciCap (IN UINT32 pciAddress, IN UINT8 targetCapId); |
| |
| // |
| // Declaration of external functions |
| // |
| |
| // |
| //----------------------------------------------------------------------------------- |
| // Early SB800 GPP initialization sequence: |
| // |
| // 1) Set port enable bit fields by current GPP link configuration mode |
| // 2) Deassert GPP reset and pull EP out of reset - Clear GPP_RESET (abcfg:0xC0[8] = 0) |
| // 3) Loop polling for the link status of all ports |
| // 4) Misc operations after link training: |
| // - (optional) Detect GFX device |
| // - Hide empty GPP configuration spaces (Disable empty GPP ports) |
| // - (optional) Power down unused GPP ports |
| // - (optional) Configure PCIE_P2P_Int_Map (abcfg:0xC4[7:0]) |
| // 5) GPP init completed |
| // |
| // |
| // *) Gen2 vs Gen1 |
| // Gen2 mode Gen1 mode |
| // --------------------------------------------------------------- |
| // STRAP_PHY_PLL_CLKF[6:0] 7'h32 7'h19 |
| // STRAP_BIF_GEN2_EN 1 0 |
| // |
| // PCIE_PHY_PLL clock locks @ 5GHz |
| // |
| // |
| |
| /** |
| * GPP early programming and link training. On exit all populated EPs should be fully operational. |
| * |
| * |
| * |
| * @param[in] pConfig Southbridge configuration structure pointer. |
| * |
| */ |
| VOID |
| sbPcieGppEarlyInit ( |
| IN AMDSBCFG* pConfig |
| ) |
| { |
| UINT8 TogglePort; |
| UINT8 portNum; |
| UINT32 reg32Value; |
| UINT8 retryCount; |
| UINT8 cimGppMemWrImprove; |
| UINT8 cimGppLaneReversal; |
| UINT8 cimAlinkPhyPllPowerDown; |
| |
| cimGppMemWrImprove = pConfig->GppMemWrImprove; |
| cimGppLaneReversal = (UINT8) pConfig->GppLaneReversal; |
| cimAlinkPhyPllPowerDown = (UINT8) pConfig->AlinkPhyPllPowerDown; |
| #if SB_CIMx_PARAMETER == 0 |
| cimGppMemWrImprove = cimGppMemWrImproveDefault; |
| cimGppLaneReversal = cimGppLaneReversalDefault; |
| cimAlinkPhyPllPowerDown = cimAlinkPhyPllPowerDownDefault; |
| #endif |
| |
| // |
| // Configure NB-SB link PCIE PHY PLL power down for L1 |
| // |
| if ( cimAlinkPhyPllPowerDown == TRUE ) { |
| UINT32 abValue; |
| // Set PCIE_P_CNTL in Alink PCIEIND space |
| writeAlink (SB_AX_INDXC_REG30 | ((UINT32) AXINDC << 29), 0x40); |
| abValue = readAlink (SB_AX_DATAC_REG34 | ((UINT32) AXINDC << 29)); |
| abValue |= BIT12 + BIT3 + BIT0; |
| abValue &= ~(BIT9 + BIT4); |
| writeAlink (SB_AX_DATAC_REG34 | ((UINT32) AXINDC << 29), abValue); |
| rwAlink (SB_AX_INDXC_REG02 | ((UINT32) AXINDC << 29), ~BIT8, (BIT8)); |
| } |
| |
| // |
| // Set ABCFG 0x031C[0] = 1 enable the lane reversal support. |
| // |
| reg32Value = readAlink (SB_ABCFG_REG31C | ((UINT32) ABCFG << 29)); |
| if ( cimGppLaneReversal ) { |
| writeAlink (SB_ABCFG_REG31C | ((UINT32) ABCFG << 29), reg32Value | BIT0); |
| } else { |
| writeAlink (SB_ABCFG_REG31C | ((UINT32) ABCFG << 29), reg32Value | 0x00); |
| } |
| // |
| // Set abcfg:0x90[20] = 1 to enable GPP bridge multi-function |
| // |
| reg32Value = readAlink (SB_ABCFG_REG90 | ((UINT32) ABCFG << 29)); |
| writeAlink (SB_ABCFG_REG90 | ((UINT32) ABCFG << 29), reg32Value | BIT20); |
| |
| |
| // |
| // Initialize and configure GPP |
| // |
| if (pConfig->GppFunctionEnable) { |
| // PreInit - Enable GPP link training |
| PreInitGppLink (pConfig); |
| |
| // |
| // GPP Upstream Memory Write Arbitration Enhancement ABCFG 0x54[26] = 1 |
| // GPP Memory Write Max Payload Improvement RCINDC_Reg 0x10[12:10] = 0x4 |
| // |
| if ( cimGppMemWrImprove == TRUE ) { |
| rwAlink (SB_ABCFG_REG54 | ((UINT32) ABCFG << 29), ~BIT26, (BIT26)); |
| rwAlink (SB_RCINDXC_REG10 | ((UINT32) RCINDXC << 29), ~(BIT12 + BIT11 + BIT10), (BIT12)); |
| } |
| |
| if ( pConfig->S3Resume ) { |
| for ( portNum = 0; portNum < MAX_GPP_PORTS; portNum++ ) { |
| reg32Value = readAlink ((SB_ABCFG_REG340 + portNum * 4) | ((UINT32) ABCFG << 29)); |
| writeAlink ((SB_ABCFG_REG340 + portNum * 4) | ((UINT32) ABCFG << 29), reg32Value & ~BIT21); |
| } |
| } |
| // |
| // a) Loop polling regA5 -> LcState (timeout ~100ms); |
| // b) if (LcState[5:0] == 0x10), training successful, go to g); |
| // c) if any of (LcState[13:8], [21:16], [29:24]) == 0x29 or 0x2A: |
| // d) Clear De-emphasis bit for relevant ports; |
| // e) Toggle GPP reset signal (via OEM callback); |
| // f) go back to a); |
| // g) exit; |
| // |
| for (retryCount = 0; retryCount < MAX_GPP_RESETS; retryCount++) { |
| // Polling each GPP port for link status |
| TogglePort = CheckGppLinkStatus (pConfig); |
| |
| if (TogglePort == 0) { |
| break; |
| } else { |
| // Check failure port and clear STRAP_BIF_DE_EMPHASIS_SEL_x_GPP bit (abcfg:0x34[0, 4, 8, C][21]=0) |
| for ( portNum = 0; portNum < MAX_GPP_PORTS; portNum++ ) { |
| if (TogglePort & (1 << portNum)) { |
| reg32Value = readAlink ((SB_ABCFG_REG340 + portNum * 4) | ((UINT32) ABCFG << 29)); |
| writeAlink ((SB_ABCFG_REG340 + portNum * 4) | ((UINT32) ABCFG << 29), reg32Value & ~BIT21); |
| } |
| sbGppForceGen1 (portNum); |
| } |
| |
| // Toggle GPP reset (Note this affects all SB800 GPP ports) |
| CallBackToOEM (CB_SBGPP_RESET_ASSERT, (UINT32)TogglePort, pConfig); |
| SbStall (500); |
| CallBackToOEM (CB_SBGPP_RESET_DEASSERT, (UINT32)TogglePort, pConfig); |
| } |
| }; |
| |
| // Misc operations after link training |
| AfterGppLinkInit (pConfig); |
| } else { |
| |
| // RPR 5.11 Power Saving With GPP Disable |
| // ABCFG 0xC0[8] = 0x0 |
| // ABCFG 0xC0[15:12] = 0xF |
| // Enable "Power Saving Feature for A-Link Express Lanes" |
| // Enable "Power Saving Feature for GPP Lanes" |
| // ABCFG 0x90[19] = 1 |
| // ABCFG 0x90[6] = 1 |
| // RCINDC_Reg 0x65 [27:0] = 0xFFFFFFF |
| // ABCFG 0xC0[7:4] = 0x0 |
| |
| rwAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29), ~BIT8, (BIT4 + BIT5 + BIT6 + BIT7)); |
| rwAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29), 0xFFFFFFFF, (BIT12 + BIT13 + BIT14 + BIT15)); |
| rwAlink (SB_AX_INDXC_REG40, ~(BIT9 + BIT4), (BIT0 + BIT3 + BIT12)); |
| rwAlink (RC_INDXC_REG40, ~(BIT9 + BIT4), (BIT0 + BIT3 + BIT12)); |
| rwAlink ((SB_ABCFG_REG90 | ((UINT32) ABCFG << 29)), 0xFFFFFFFF, (BIT6 + BIT19)); |
| rwAlink (RC_INDXC_REG65, 0xFFFFFFFF, 0x0fffffff); |
| rwAlink ((SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29)), ~(BIT4 + BIT5 + BIT6 + BIT7), 0); |
| } |
| sbGppDisableUnusedPadMap ( pConfig ); |
| } |
| |
| /** |
| * PreInitGppLink - Enable GPP link training. |
| * |
| * |
| * |
| * @param[in] pConfig Southbridge configuration structure pointer. |
| * |
| */ |
| VOID |
| PreInitGppLink ( |
| IN AMDSBCFG* pConfig |
| ) |
| { |
| UINT8 portMask[5] = {0x01, |
| 0x00, |
| 0x03, |
| 0x07, |
| 0x0F |
| }; |
| UINT8 cfgMode; |
| UINT8 portId; |
| UINT32 reg32Value; |
| UINT16 tmp16Value; |
| |
| // PCIE_GPP_ENABLE (abcfg:0xC0): |
| // |
| // GPP_LINK_CONFIG ([3:0]) PortA PortB PortC PortD Description |
| // ---------------------------------------------------------------------------------- |
| // 0000 0-3 x4 Config |
| // 0001 N/A |
| // 0010 0-1 2-3 0 2:2 Config |
| // 0011 0-1 2 3 2:1:1 Config |
| // 0100 0 1 2 3 1:1:1:1 Config |
| // |
| // For A12 and above: |
| // ABCFG:0xC0[12] - Port A hold training (default 1) |
| // ABCFG:0xC0[13] - Port B hold training (default 1) |
| // ABCFG:0xC0[14] - Port C hold training (default 1) |
| // ABCFG:0xC0[15] - Port D hold training (default 1) |
| // |
| // |
| // |
| // Set port enable bit fields based on current GPP link configuration mode |
| // |
| cfgMode = (UINT8) pConfig->GppLinkConfig; |
| if ( cfgMode > GPP_CFGMODE_X1111 || cfgMode == 1 ) { |
| cfgMode = GPP_CFGMODE_X4000; |
| pConfig->GppLinkConfig = GPP_CFGMODE_X4000; |
| } |
| reg32Value = (UINT32) portMask[cfgMode]; |
| |
| // Mask out non-applicable ports according to the target link configuration mode |
| for ( portId = 0; portId < MAX_GPP_PORTS; portId++ ) { |
| if (!(reg32Value & (1 << portId))) |
| pConfig->PORTCONFIG[portId].PortCfg.PortPresent = false; |
| else if (!pConfig->PORTCONFIG[portId].PortCfg.PortPresent) |
| reg32Value &= ~(1 << portId); |
| } |
| |
| // |
| // Deassert GPP reset and pull EP out of reset - Clear GPP_RESET (abcfg:0xC0[8] = 0) |
| // |
| tmp16Value = (UINT16) (~reg32Value << 12); |
| reg32Value = (UINT32) (tmp16Value + (reg32Value << 4) + cfgMode); |
| writeAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29), reg32Value); |
| |
| reg32Value = readAlink (0xC0 | ((UINT32) RCINDXC << 29)); |
| writeAlink (0xC0 | ((UINT32) RCINDXC << 29), reg32Value | 0x400); // Set STRAP_F0_MSI_EN |
| |
| // A-Link L1 Entry Delay Shortening |
| // AXINDP_Reg 0xA0[7:4] = 0x3 |
| rwAlink (SB_AX_INDXP_REGA0, 0xFFFFFF0F, 0x30); |
| rwAlink (SB_AX_INDXP_REGB1, 0xFFFFFFFF, BIT19); |
| rwAlink (SB_AX_INDXP_REGB1, 0xFFFFFFFF, BIT28); |
| |
| // RPR5.22 GPP L1 Entry Delay Shortening |
| // RCINDP_Reg 0xA0[7:4] = 0x1 Enter L1 sooner after ACK'ing PM request. |
| // This is done to reduce number of NAK received with L1 enabled. |
| for ( portId = 0; portId < MAX_GPP_PORTS; portId++ ) { |
| rwAlink (SB_RCINDXP_REGA0 | portId << 24, 0xFFFFFF0F, 0x10); |
| } |
| } |
| |
| /** |
| * CheckGppLinkStatus - loop polling the link status for each GPP port |
| * |
| * |
| * Return: ToggleStatus[3:0] = Port bitmap for those need to clear De-emphasis |
| * |
| * @param[in] pConfig Southbridge configuration structure pointer. |
| * |
| */ |
| UINT8 |
| CheckGppLinkStatus ( |
| IN AMDSBCFG* pConfig |
| ) |
| { |
| UINT32 retryCounter; |
| UINT32 portId; |
| UINT32 abIndex; |
| UINT32 Data32; |
| UINT8 portScanMap; |
| UINT8 portScanMap2; |
| UINT8 ToggleStatus; |
| UINT16 i; |
| SBGPPPORTCONFIG *portCfg; |
| |
| |
| portScanMap = 0; |
| retryCounter = MAX_TRAINING_RETRY; |
| ToggleStatus = 0; |
| |
| // Obtain a list of ports to be checked |
| for ( portId = 0; portId < MAX_GPP_PORTS; portId++ ) { |
| portCfg = &pConfig->PORTCONFIG[portId].PortCfg; |
| if ( portCfg->PortPresent == TRUE && portCfg->PortDetected == FALSE ) { |
| portScanMap |= 1 << portId; |
| } |
| } |
| portScanMap2 = portScanMap; |
| |
| // |
| // After training is enabled, Check LCSTATE for each port, if LCSTATE<= 4, then keep |
| // polling for up to 40ms. If LCSTATE still <= 4, then assume the port to be empty. |
| // |
| i = 400; |
| while ( --i && portScanMap2) { |
| for (portId = 0; portId < MAX_GPP_PORTS; portId++) { |
| portCfg = &pConfig->PORTCONFIG[portId].PortCfg; |
| if (((portCfg->PortHotPlug == FALSE) || ((portCfg->PortHotPlug == TRUE) && (pConfig->S3Resume == FALSE)) ) && (portScanMap2 & (1 << portId))) { |
| // |
| // Get port link state (reading LC_CURRENT_STATE of PCIEIND_P) |
| // |
| abIndex = SB_RCINDXP_REGA5 | ((UINT32) RCINDXP << 29) | (portId << 24); |
| Data32 = readAlink (abIndex) & 0x3F; |
| if ((UINT8) (Data32) > 4) { |
| portScanMap2 &= ~(1 << portId); // This port is not empty |
| break; |
| } |
| SbStall (100); // Delay 100us |
| } |
| } |
| } |
| portScanMap &= ~portScanMap2; // Mark remaining ports as empty |
| |
| |
| while ( --retryCounter && portScanMap ) { |
| for ( portId = 0; portId < MAX_GPP_PORTS; portId++ ) { |
| portCfg = &pConfig->PORTCONFIG[portId].PortCfg; |
| if (( portCfg->PortHotPlug == TRUE ) && ( pConfig->S3Resume )) { |
| continue; |
| } |
| if ( portCfg->PortPresent == TRUE && portCfg->PortDetected == FALSE ) { |
| // |
| // Get port link state (reading LC_CURRENT_STATE of PCIEIND_P) |
| // |
| SbStall (1000); // Delay 400us |
| abIndex = SB_RCINDXP_REGA5 | ((UINT32) RCINDXP << 29) | (portId << 24); |
| Data32 = readAlink (abIndex) & 0x3F3F3F3F; |
| |
| if ( (UINT8) (Data32) == 0x10 ) { |
| portCfg->PortDetected = TRUE; |
| portScanMap &= ~(1 << portId); |
| } else { |
| for (i = 0; i < 4; i++) { |
| // |
| // Compliance mode (0x7), downgrade from Gen2 to Gen1 (*A12) |
| // |
| if ((UINT8) (Data32) == 0x29 || (UINT8) (Data32) == 0x2A || (UINT8) (Data32) == 0x7 ) { |
| ToggleStatus |= (1 << portId); // A11 only: need to toggle GPP reset |
| portScanMap &= ~(1 << portId); |
| } |
| Data32 >>= 8; |
| } |
| } |
| } |
| } |
| } |
| return ToggleStatus; |
| } |
| |
| |
| /** |
| * AfterGppLinkInit |
| * - Search for display device behind each GPP port |
| * - If the port is empty AND not hotplug-capable: |
| * * Turn off link training |
| * * (optional) Power down the port |
| * * Hide the configuration space (Turn off the port) |
| * |
| * @param[in] pConfig Southbridge configuration structure pointer. |
| * |
| */ |
| VOID |
| AfterGppLinkInit ( |
| IN AMDSBCFG* pConfig |
| ) |
| { |
| UINT32 portId; |
| SBGPPPORTCONFIG *portCfg; |
| UINT32 regBusNumber; |
| UINT32 abValue; |
| UINT32 abIndex; |
| UINT32 i; |
| UINT32 Data32; |
| UINT8 bValue; |
| UINT8 cimGppGen2; |
| |
| cimGppGen2 = pConfig->GppGen2; |
| #if SB_CIMx_PARAMETER == 0 |
| cimGppGen2 = cimGppGen2Default; |
| #endif |
| |
| bValue = GPP_EFUSE_LOCATION; |
| getEfuseStatus (&bValue); |
| if ( (bValue & GPP_GEN2_EFUSE_BIT) != 0 ) { |
| cimGppGen2 = FALSE; |
| } else { |
| pConfig->CoreGen2Enable = TRUE; // Output for platform use |
| } |
| |
| //GPP Gen2 Speed Change |
| // if ((GPP Gen2 == enabled) and (RCINDP_Reg 0xA4[0] == 0x1)) { |
| // PCIe_Cfg 0x88[3:0] = 0x2 |
| // RCINDP_Reg 0xA2[13] = 0x0 |
| // RCINDP_Reg 0xC0[15] = 0x0 |
| // RCINDP_Reg 0xA4[29] = 0x1 |
| // } else { |
| // PCIe_Cfg 0x88[3:0] = 0x1 |
| // RCINDP_Reg 0xA4[0] = 0x0 |
| // RCINDP_Reg 0xA2[13] = 0x1 |
| // RCINDP_Reg 0xC0[15] = 0x0 |
| // RCINDP_Reg 0xA4[29] = 0x1 |
| // } |
| for ( portId = 0; portId < MAX_GPP_PORTS; portId++ ) { |
| portCfg = &pConfig->PORTCONFIG[portId].PortCfg; |
| abValue = readAlink (SB_RCINDXP_REGA4 | portId << 24) & BIT0; |
| if (( cimGppGen2 == TRUE ) && (abValue == BIT0) && (portCfg->PortDetected == TRUE)) { |
| portCfg->PortIsGen2 = TRUE; // Output for platform use |
| sbGppForceGen2 (portId); |
| //_asm {jmp $}; |
| SbStall (400); // Delay 400us |
| i = 500; |
| Data32 = 0; |
| while ( --i ) { |
| abIndex = SB_RCINDXP_REGA5 | ((UINT32) RCINDXP << 29) | (portId << 24); |
| Data32 = readAlink (abIndex) & 0x3F; |
| if ((UINT8) (Data32) == 0x10) { |
| break; |
| } |
| SbStall (400); // Delay 100us |
| } |
| if (!( (UINT8) (Data32) == 0x10 )) { |
| if (pConfig->GppCompliance == FALSE) { |
| portCfg->PortIsGen2 = FALSE; // Revert to default; output for platform use |
| sbGppForceGen1 (portId); |
| } |
| } |
| } else { |
| if (pConfig->GppCompliance == FALSE) { |
| sbGppForceGen1 (portId); |
| } |
| } |
| //RPR 5.9 Link Bandwidth Notification Capability Enable |
| //RCINDC 0xC1[0] = 1 |
| //PCIe Cfg 0x68[10] = 0 |
| //PCIe Cfg 0x68[11] = 0 |
| |
| rwAlink (SB_RCINDXC_REGC1, 0xFFFFFFFF, BIT0); |
| RWPCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x68), AccWidthUint16, ~(BIT10 + BIT11), 0); |
| } |
| |
| // Status = AGESA_SUCCESS; |
| pConfig->GppFoundGfxDev = 0; |
| abValue = readAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29)); |
| |
| for ( portId = 0; portId < MAX_GPP_PORTS; portId++ ) { |
| portCfg = &pConfig->PORTCONFIG[portId].PortCfg; |
| // Check if there is GFX device behind each GPP port |
| if ( portCfg->PortDetected == TRUE ) { |
| regBusNumber = (SBTEMP_BUS << 16) + (SBTEMP_BUS << 8); |
| WritePCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x18), AccWidthUint32, ®BusNumber); |
| // *** Stall (); |
| ReadPCI (PCI_ADDRESS (SBTEMP_BUS, 0, 0, 0x0B), AccWidthUint8, &bValue); |
| if ( bValue == 3 ) { |
| pConfig->GppFoundGfxDev |= (1 << portId); |
| } |
| regBusNumber = 0; |
| WritePCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x18), AccWidthUint32, ®BusNumber); |
| } |
| |
| // Mask off non-applicable ports |
| else if ( portCfg->PortPresent == FALSE ) { |
| abValue &= ~(1 << (portId + 4)); |
| } |
| // Mask off empty port if the port is not hotplug-capable |
| else if ( portCfg->PortHotPlug == FALSE ) { |
| abValue &= ~(1 << (portId + 4)); |
| } |
| // Clear STRAP_BIF_DE_EMPHASIS_SEL_x_GPP bit (abcfg:0x34[0, 4, 8, C][21]=0) to make hotplug working |
| if ( portCfg->PortHotPlug == TRUE ) { |
| rwAlink ((SB_ABCFG_REG340 + portId * 4) | ((UINT32) ABCFG << 29), ~BIT21, 0); |
| |
| // RPR5.12 Hot Plug: PCIe Native Support |
| // RCINDP_Reg 0x10[3] = 0x1 |
| // PCIe_Cfg 0x5A[8] = 0x1 |
| // PCIe_Cfg 0x6C[6] = 0x1 |
| // RCINDP_Reg 0x20[19] = 0x0 |
| |
| rwAlink ((SB_RCINDXP_REG10 | ((UINT32) RCINDXP << 29) | (portId << 24)), 0xFFFFFFFF, BIT3); |
| RWPCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x5b), AccWidthUint8, 0xff, BIT0); |
| RWPCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x6c), AccWidthUint8, 0xff, BIT6); |
| rwAlink ((SB_RCINDXP_REG20 | ((UINT32) RCINDXP << 29) | (portId << 24)), ~BIT19, 0); |
| } |
| } |
| if ( pConfig->GppUnhidePorts == FALSE ) { |
| if ((abValue & 0xF0) == 0) { |
| abValue = BIT8; // if all ports are empty set GPP_RESET |
| } else if ((abValue & 0xE0) != 0 && (abValue & 0x10) == 0) { |
| abValue |= BIT4; // PortA should always be visible whenever other ports are exist |
| } |
| |
| // Update GPP_Portx_Enable (abcfg:0xC0[7:5]) |
| writeAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29), abValue); |
| } |
| |
| // |
| // Common initialization for open GPP ports |
| // |
| for ( portId = 0; portId < MAX_GPP_PORTS; portId++ ) { |
| ReadPCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x80), AccWidthUint8, &bValue); |
| if (bValue != 0xff) { |
| // Set pciCfg:PCIE_DEVICE_CNTL2[3:0] = 4'h6 (0x80[3:0]) |
| bValue &= 0xf0; |
| bValue |= 0x06; |
| WritePCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x80), AccWidthUint8, &bValue); |
| |
| // Set PCIEIND_P:PCIE_RX_CNTL[RX_RCB_CPL_TIMEOUT_MODE] (0x70:[19]) = 1 |
| abIndex = SB_RCINDXP_REG70 | ((UINT32) RCINDXP << 29) | (portId << 24); |
| abValue = readAlink (abIndex) | BIT19; |
| writeAlink (abIndex, abValue); |
| |
| // Set PCIEIND_P:PCIE_TX_CNTL[TX_FLUSH_TLP_DIS] (0x20:[19]) = 0 |
| abIndex = SB_RCINDXP_REG20 | ((UINT32) RCINDXP << 29) | (portId << 24); |
| abValue = readAlink (abIndex) & ~BIT19; |
| writeAlink (abIndex, abValue); |
| |
| } |
| } |
| } |
| |
| |
| /** |
| * sbPcieGppLateInit - Late PCIE initialization for SB800 GPP component |
| * |
| * |
| * @param[in] pConfig Southbridge configuration structure pointer. |
| * |
| */ |
| VOID |
| sbPcieGppLateInit ( |
| IN AMDSBCFG* pConfig |
| ) |
| { |
| UINT32 reg32Value; |
| UINT8 portId; |
| UINT8 busNum; |
| UINT8 aspmValue; |
| UINT8 reg8Value; |
| UINT8 cimGppPhyPllPowerDown; |
| |
| reg8Value = 0x01; |
| // |
| // Configure ASPM |
| // |
| // writeAlink (0xC0 | ((UINT32) RCINDXC << 29), 0x400); // Set STRAP_F0_MSI_EN |
| aspmValue = (UINT8)pConfig->GppPortAspm; |
| cimGppPhyPllPowerDown = (UINT8) pConfig->GppPhyPllPowerDown; |
| #if SB_CIMx_PARAMETER == 0 |
| aspmValue = cimGppPortAspmDefault; |
| cimGppPhyPllPowerDown = cimGppPhyPllPowerDownDefault; |
| #endif |
| |
| for ( portId = 0; portId < MAX_GPP_PORTS; portId++ ) { |
| // write pci_reg3d with 0x01 to fix yellow mark for GPP bridge under Vista |
| // when native PCIE is enabled but MSI is not available |
| // SB02029: SB800 BIF/GPP allowing strap STRAP_BIF_INTERRUPT_PIN_SB controlled by AB reg |
| WritePCI (PCI_ADDRESS (0, 21, portId, 0x3d), AccWidthUint8, ®8Value); |
| ReadPCI (PCI_ADDRESS (0, 21, portId, 0x19), AccWidthUint8, &busNum); |
| if (busNum != 0xFF) { |
| ReadPCI (PCI_ADDRESS (busNum, 0, 0, 0x00), AccWidthUint32, ®32Value); |
| if (reg32Value != 0xffffffff) { |
| // Set ASPM on EP side |
| sbGppSetAspm (PCI_ADDRESS (busNum, 0, 0, 0), aspmValue & 0x3); |
| // Set ASPM on port side |
| sbGppSetAspm (PCI_ADDRESS (0, 21, portId, 0), aspmValue & 0x3); |
| } |
| } |
| aspmValue = aspmValue >> 2; |
| } |
| |
| // |
| // Configure Lock HWInit registers |
| // |
| reg32Value = readAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29)); |
| if (reg32Value & 0xF0) { |
| reg32Value = readAlink (SB_RCINDXC_REG10 | ((UINT32) RCINDXC << 29)); |
| writeAlink (SB_RCINDXC_REG10 | ((UINT32) RCINDXC << 29), reg32Value | BIT0); // Set HWINIT_WR_LOCK |
| |
| if ( cimGppPhyPllPowerDown == TRUE ) { |
| // |
| // RPR 5.4 Power Saving Feature for GPP Lanes |
| // |
| UINT32 abValue; |
| // Set PCIE_P_CNTL in Alink PCIEIND space |
| abValue = readAlink (RC_INDXC_REG40 | ((UINT32) RCINDXC << 29)); |
| abValue |= BIT12 + BIT3 + BIT0; |
| abValue &= ~(BIT9 + BIT4); |
| writeAlink (RC_INDXC_REG40 | ((UINT32) RCINDXC << 29), abValue); |
| } |
| } |
| |
| // |
| // Configure Lock HWInit registers |
| // |
| reg32Value = readAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29)); |
| // |
| // Disable hidden register decode and serial number capability |
| // |
| reg32Value = readAlink (SB_ABCFG_REG330 | ((UINT32) ABCFG << 29)); |
| writeAlink (SB_ABCFG_REG330 | ((UINT32) ABCFG << 29), reg32Value & ~(BIT26 + BIT10)); |
| } |
| |
| /** |
| * sbGppSetAspm - Set SPP ASPM |
| * |
| * |
| * @param[in] pciAddress PCI Address. |
| * @param[in] LxState Lane State. |
| * |
| */ |
| VOID |
| sbGppSetAspm ( |
| IN UINT32 pciAddress, |
| IN UINT8 LxState |
| ) |
| { |
| UINT8 pcieCapOffset; |
| UINT8 value8; |
| UINT8 maxFuncs; |
| UINT32 devBDF; |
| |
| maxFuncs = 1; |
| ReadPCI (pciAddress + 0x0E, AccWidthUint8, &value8); |
| |
| if (value8 & BIT7) { |
| maxFuncs = 8; // multi-function device |
| } |
| while (maxFuncs != 0) { |
| devBDF = pciAddress + (UINT32) ((maxFuncs - 1) << 16); |
| pcieCapOffset = sbFindPciCap (devBDF, PCIE_CAP_ID); |
| if (pcieCapOffset) { |
| // Read link capabilities register (0x0C[11:10] - ASPM support) |
| ReadPCI (devBDF + pcieCapOffset + 0x0D, AccWidthUint8, &value8); |
| if (value8 & BIT2) { |
| value8 = (value8 >> 2) & (BIT1 + BIT0); |
| // Set ASPM state in link control register |
| RWPCI (devBDF + pcieCapOffset + 0x10, AccWidthUint8, 0xffffffff, LxState & value8); |
| } |
| } |
| maxFuncs--; |
| } |
| } |
| |
| /** |
| * sbFindPciCap - Find PCI Cap |
| * |
| * |
| * @param[in] pciAddress PCI Address. |
| * @param[in] targetCapId Target Cap ID. |
| * |
| */ |
| UINT8 |
| sbFindPciCap ( |
| IN UINT32 pciAddress, |
| IN UINT8 targetCapId |
| ) |
| { |
| UINT8 NextCapPtr; |
| UINT8 CapId; |
| |
| NextCapPtr = 0x34; |
| while (NextCapPtr != 0) { |
| ReadPCI (pciAddress + NextCapPtr, AccWidthUint8, &NextCapPtr); |
| if (NextCapPtr == 0xff) { |
| return 0; |
| } |
| if (NextCapPtr != 0) { |
| ReadPCI (pciAddress + NextCapPtr, AccWidthUint8, &CapId); |
| if (CapId == targetCapId) { |
| break; |
| } else { |
| NextCapPtr++; |
| } |
| } |
| } |
| return NextCapPtr; |
| } |
| |
| /** |
| * sbGppForceGen2 - Set SPP to GENII |
| * |
| * |
| * @param[in] portId |
| * |
| */ |
| VOID |
| sbGppForceGen2 ( |
| IN UINT32 portId |
| ) |
| { |
| RWPCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x88), AccWidthUint8, 0xf0, 0x02); |
| rwAlink (SB_RCINDXP_REGA2 | portId << 24, ~BIT13, 0); |
| rwAlink (SB_RCINDXP_REGC0 | portId << 24, ~BIT15, 0); |
| rwAlink (SB_RCINDXP_REGA4 | portId << 24, 0xFFFFFFFF, BIT29); |
| } |
| |
| /** |
| * sbGppForceGen1 - Set SPP to GENI |
| * |
| * |
| * @param[in] portId |
| * |
| */ |
| VOID |
| sbGppForceGen1 ( |
| IN UINT32 portId |
| ) |
| { |
| RWPCI (PCI_ADDRESS (0, GPP_DEV_NUM, portId, 0x88), AccWidthUint8, 0xf0, 0x01); |
| rwAlink (SB_RCINDXP_REGA4 | portId << 24, ~BIT0, 0); |
| rwAlink (SB_RCINDXP_REGA2 | portId << 24, 0xFFFFFFFF, BIT13); |
| rwAlink (SB_RCINDXP_REGC0 | portId << 24, ~BIT15, 0); |
| rwAlink (SB_RCINDXP_REGA4 | portId << 24, 0xFFFFFFFF, BIT29); |
| } |
| |
| /** |
| * sbGppDisableUnusedPadMap - Return GPP Pad Map |
| * |
| * |
| * @param[in] pConfig |
| * |
| */ |
| VOID |
| sbGppDisableUnusedPadMap ( |
| IN AMDSBCFG* pConfig |
| ) |
| { |
| UINT32 Data32; |
| UINT32 HoldData32; |
| SBGPPPORTCONFIG *portCfg; |
| UINT8 cimGppLaneReversal; |
| UINT8 cimAlinkPhyPllPowerDown; |
| UINT8 cimGppPhyPllPowerDown; |
| |
| cimAlinkPhyPllPowerDown = (UINT8) pConfig->AlinkPhyPllPowerDown; |
| cimGppLaneReversal = (UINT8) pConfig->GppLaneReversal; |
| cimGppPhyPllPowerDown = (UINT8) pConfig->GppPhyPllPowerDown; |
| #if SB_CIMx_PARAMETER == 0 |
| cimGppLaneReversal = cimGppLaneReversalDefault; |
| cimAlinkPhyPllPowerDown = cimAlinkPhyPllPowerDownDefault; |
| cimGppPhyPllPowerDown = cimGppPhyPllPowerDownDefault; |
| #endif |
| |
| Data32 = 0; |
| HoldData32 = 0; |
| switch ( pConfig->GppLinkConfig ) { |
| case GPP_CFGMODE_X4000: |
| portCfg = &pConfig->PORTCONFIG[0].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= 0x0f0f; |
| HoldData32 |= 0x1000; |
| } |
| break; |
| case GPP_CFGMODE_X2200: |
| portCfg = &pConfig->PORTCONFIG[0].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0c0c:0x0303; |
| HoldData32 |= 0x1000; |
| } |
| portCfg = &pConfig->PORTCONFIG[1].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0303:0x0c0c; |
| HoldData32 |= 0x2000; |
| } |
| break; |
| case GPP_CFGMODE_X2110: |
| portCfg = &pConfig->PORTCONFIG[0].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0c0c:0x0303; |
| HoldData32 |= 0x1000; |
| } |
| portCfg = &pConfig->PORTCONFIG[1].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0202:0x0404; |
| HoldData32 |= 0x2000; |
| } |
| portCfg = &pConfig->PORTCONFIG[2].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0101:0x0808; |
| HoldData32 |= 0x4000; |
| } |
| break; |
| case GPP_CFGMODE_X1111: |
| portCfg = &pConfig->PORTCONFIG[0].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0808:0x0101; |
| HoldData32 |= 0x1000; |
| } |
| portCfg = &pConfig->PORTCONFIG[1].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0404:0x0202; |
| HoldData32 |= 0x2000; |
| } |
| portCfg = &pConfig->PORTCONFIG[2].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0202:0x0404; |
| HoldData32 |= 0x4000; |
| } |
| portCfg = &pConfig->PORTCONFIG[3].PortCfg; |
| if ( portCfg->PortDetected == FALSE ) { |
| Data32 |= ( cimGppLaneReversal )? 0x0101:0x0808; |
| HoldData32 |= 0x8000; |
| } |
| break; |
| default: |
| break; |
| } |
| |
| // RPR 5.11 Power Saving With GPP Disable |
| // ABCFG 0xC0[8] = 0x0 |
| // ABCFG 0xC0[15:12] = 0xF |
| // Enable "Power Saving Feature for A-Link Express Lanes" |
| // Enable "Power Saving Feature for GPP Lanes" |
| // ABCFG 0x90[19] = 1 |
| // ABCFG 0x90[6] = 1 |
| // RCINDC_Reg 0x65 [27:0] = 0xFFFFFFF |
| // ABCFG 0xC0[7:4] = 0x0 |
| if ( (Data32 & 0xf) == 0xf ) Data32 |= 0x0cff0000; |
| if ( cimAlinkPhyPllPowerDown && cimGppPhyPllPowerDown ) { |
| rwAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29), ~BIT8, 0); |
| rwAlink (SB_ABCFG_REGC0 | ((UINT32) ABCFG << 29), 0xFFFFFFFF, HoldData32); |
| rwAlink (SB_AX_INDXC_REG40, ~(BIT9 + BIT4), (BIT0 + BIT3 + BIT12)); |
| rwAlink (RC_INDXC_REG40, ~(BIT9 + BIT4), (BIT0 + BIT3 + BIT12)); |
| rwAlink ((SB_ABCFG_REG90 | ((UINT32) ABCFG << 29)), 0xFFFFFFFF, (BIT6 + BIT19)); |
| rwAlink (RC_INDXC_REG65, 0xFFFFFFFF, Data32); |
| } |
| } |