summaryrefslogtreecommitdiff
path: root/src/southbridge/intel/lynxpoint/smihandler.c
diff options
context:
space:
mode:
authorDuncan Laurie <dlaurie@chromium.org>2013-05-29 15:27:55 -0700
committerStefan Reinauer <stefan.reinauer@coreboot.org>2013-11-25 23:55:15 +0100
commit2d9d39a7041cf531246845194f76cb9f65eaa08d (patch)
tree51f60a7d9ede4ee6f2b44e270d85a08195dd6aef /src/southbridge/intel/lynxpoint/smihandler.c
parent5afca1357fdaebc5c4ad2b2a963f3c239648ba76 (diff)
downloadcoreboot-2d9d39a7041cf531246845194f76cb9f65eaa08d.tar.xz
lynxpoint: Enable USB clock gating, late setup, and sleep prep
Both EHCI and XHCI controllers have additional setup steps that are not part of the PEI reference code so they need to be done later. Both controllers also have specific clock gating setup requirements that are now implemented. Additionally they both have specific requirements when entering sleep states. XHCI needs something in S3/S4/S5 and EHCI only has steps for S4/S5 entry. Change-Id: Ic62cbc8b6255455e56b72dd5d52e27a311999330 Signed-off-by: Duncan Laurie <dlaurie@chromium.org> Reviewed-on: https://gerrit.chromium.org/gerrit/57033 Reviewed-by: Aaron Durbin <adurbin@chromium.org> Reviewed-on: http://review.coreboot.org/4217 Tested-by: build bot (Jenkins) Reviewed-by: Ronald G. Minnich <rminnich@gmail.com>
Diffstat (limited to 'src/southbridge/intel/lynxpoint/smihandler.c')
-rw-r--r--src/southbridge/intel/lynxpoint/smihandler.c98
1 files changed, 98 insertions, 0 deletions
diff --git a/src/southbridge/intel/lynxpoint/smihandler.c b/src/southbridge/intel/lynxpoint/smihandler.c
index 203a1bbfb5..add53ed5a9 100644
--- a/src/southbridge/intel/lynxpoint/smihandler.c
+++ b/src/southbridge/intel/lynxpoint/smihandler.c
@@ -20,6 +20,7 @@
* MA 02110-1301 USA
*/
+#include <delay.h>
#include <types.h>
#include <arch/hlt.h>
#include <arch/io.h>
@@ -105,6 +106,98 @@ static void busmaster_disable_on_bus(int bus)
}
}
+/* Handler for EHCI controller on entry to S3/S4/S5 */
+static void ehci_sleep_prepare(device_t dev, u8 slp_typ)
+{
+ u32 reg32;
+ u32 bar0_base;
+ u16 pwr_state;
+ u16 pci_cmd;
+
+ /* Check if the controller is disabled or not present */
+ bar0_base = pci_read_config32(dev, PCI_BASE_ADDRESS_0);
+ if (bar0_base == 0 || bar0_base == 0xffffffff)
+ return;
+ pci_cmd = pci_read_config32(dev, PCI_COMMAND);
+
+ switch (slp_typ) {
+ case SLP_TYP_S4:
+ case SLP_TYP_S5:
+ /* Check if controller is in D3 power state */
+ pwr_state = pci_read_config16(dev, EHCI_PWR_CNTL_STS);
+ if ((pwr_state & EHCI_PWR_STS_MASK) == EHCI_PWR_STS_SET_D3) {
+ /* Put in D0 */
+ pwr_state &= ~EHCI_PWR_STS_MASK;
+ pwr_state |= EHCI_PWR_STS_SET_D0;
+ pci_write_config16(dev, EHCI_PWR_CNTL_STS, pwr_state);
+
+ /* Make sure memory bar is set */
+ pci_write_config32(dev, PCI_BASE_ADDRESS_0, bar0_base);
+
+ /* Make sure memory space is enabled */
+ pci_write_config16(dev, PCI_COMMAND, pci_cmd |
+ PCI_COMMAND_MASTER | PCI_COMMAND_MEMORY);
+ }
+
+ /*
+ * If Run/Stop (bit0) is clear in USB2.0_CMD:
+ * - Clear Async Schedule Enable (bit5) and
+ * - Clear Periodic Schedule Enable (bit4) and
+ * - Set Run/Stop (bit0)
+ */
+ reg32 = read32(bar0_base + 0x20);
+ if (reg32 & (1 << 0)) {
+ reg32 &= ~((1 << 5) | (1 << 4));
+ reg32 |= (1 << 0);
+ write32(bar0_base + 0x20, reg32);
+ }
+
+ /* Check for Port Enabled in PORTSC */
+ reg32 = read32(bar0_base + 0x64);
+ if (reg32 & (1 << 2)) {
+ /* Set suspend bit in PORTSC if not already set */
+ if (!(reg32 & (1 << 7))) {
+ reg32 |= (1 << 7);
+ write32(bar0_base + 0x64, reg32);
+ }
+
+ /* Delay 25ms !! */
+ udelay(25 * 1000);
+
+ /* Clear Run/Stop bit */
+ reg32 = read32(bar0_base + 0x20);
+ reg32 &= (1 << 0);
+ write32(bar0_base + 0x20, reg32);
+ }
+
+ pwr_state = pci_read_config16(dev, EHCI_PWR_CNTL_STS);
+ if ((pwr_state & EHCI_PWR_STS_MASK) == EHCI_PWR_STS_SET_D3) {
+ /* Restore pci command reg */
+ pci_write_config16(dev, PCI_COMMAND, pci_cmd);
+
+ /* Enable D3 */
+ pwr_state |= EHCI_PWR_STS_SET_D3;
+ pci_write_config16(dev, EHCI_PWR_CNTL_STS, pwr_state);
+ }
+ }
+}
+
+/* Handler for XHCI controller on entry to S3/S4/S5 */
+static void xhci_sleep_prepare(device_t dev, u8 slp_typ)
+{
+ u16 reg16;
+
+ switch (slp_typ) {
+ case SLP_TYP_S3:
+ case SLP_TYP_S4:
+ case SLP_TYP_S5:
+ /* Set D3Hot state and PME enable bit */
+ reg16 = pci_read_config16(dev, 0x74);
+ reg16 |= (1 << 8) | (1 << 1) | (1 << 0);
+ pci_write_config16(dev, 0x74, reg16);
+ }
+}
+
static void southbridge_smi_sleep(void)
{
u8 reg8;
@@ -132,6 +225,11 @@ static void southbridge_smi_sleep(void)
/* Do any mainboard sleep handling */
mainboard_smi_sleep(slp_typ-2);
+ /* USB sleep preparations */
+ ehci_sleep_prepare(PCH_EHCI1_DEV, slp_typ);
+ ehci_sleep_prepare(PCH_EHCI2_DEV, slp_typ);
+ xhci_sleep_prepare(PCH_XHCI_DEV, slp_typ);
+
#if CONFIG_ELOG_GSMI
/* Log S3, S4, and S5 entry */
if (slp_typ >= 5)