blob: 8b037a9dd09c012c780ade60422f03592bd264bb [file] [log] [blame]
Angel Pons182dbde2020-04-02 23:49:05 +02001/* SPDX-License-Identifier: GPL-2.0-only */
Aaron Durbin76c37002012-10-30 09:03:43 -05002
3#include <console/console.h>
Kyösti Mälkkif1b58b72019-03-01 13:43:02 +02004#include <device/pci_ops.h>
Aaron Durbin76c37002012-10-30 09:03:43 -05005#include <device/device.h>
6#include <device/pci.h>
Angel Pons2178b722020-05-31 00:55:35 +02007#include "iobp.h"
Aaron Durbin76c37002012-10-30 09:03:43 -05008#include "pch.h"
9
Elyes HAOUAS1dcd8db2018-12-05 10:59:42 +010010#ifdef __SIMPLE_DEVICE__
11static pci_devfn_t pch_get_lpc_device(void)
Duncan Laurie5cc51c02013-03-07 14:06:43 -080012{
Duncan Laurie5cc51c02013-03-07 14:06:43 -080013 return PCI_DEV(0, 0x1f, 0);
Duncan Laurie5cc51c02013-03-07 14:06:43 -080014}
Elyes HAOUAS1dcd8db2018-12-05 10:59:42 +010015#else
16static struct device *pch_get_lpc_device(void)
17{
Kyösti Mälkkic70eed12018-05-22 02:18:00 +030018 return pcidev_on_root(0x1f, 0);
Elyes HAOUAS1dcd8db2018-12-05 10:59:42 +010019}
20#endif
Aaron Durbin76c37002012-10-30 09:03:43 -050021
22int pch_silicon_revision(void)
23{
Duncan Laurie5cc51c02013-03-07 14:06:43 -080024 static int pch_revision_id = -1;
25
Aaron Durbin76c37002012-10-30 09:03:43 -050026 if (pch_revision_id < 0)
Duncan Laurie5cc51c02013-03-07 14:06:43 -080027 pch_revision_id = pci_read_config8(pch_get_lpc_device(),
28 PCI_REVISION_ID);
Aaron Durbin76c37002012-10-30 09:03:43 -050029 return pch_revision_id;
30}
31
Tristan Corrickd3f01b22018-12-06 22:46:58 +130032int pch_silicon_id(void)
33{
34 static int pch_id = -1;
35
36 if (pch_id < 0)
37 pch_id = pci_read_config16(pch_get_lpc_device(), PCI_DEVICE_ID);
38
39 return pch_id;
40}
41
Angel Pons31739932020-07-03 23:14:40 +020042enum pch_platform_type get_pch_platform_type(void)
Aaron Durbin76c37002012-10-30 09:03:43 -050043{
Angel Pons31739932020-07-03 23:14:40 +020044 const u16 did = pci_read_config16(pch_get_lpc_device(), PCI_DEVICE_ID);
Duncan Laurie5cc51c02013-03-07 14:06:43 -080045
Angel Pons31739932020-07-03 23:14:40 +020046 /* Check if this is a LPT-LP or WPT-LP device ID */
47 if ((did & 0xff00) == 0x9c00)
48 return PCH_TYPE_ULT;
49
50 /* Non-LP laptop SKUs have an odd device ID (least significant bit is one) */
51 if (did & 1)
52 return PCH_TYPE_MOBILE;
53
54 /* Desktop and Server SKUs have an even device ID */
55 return PCH_TYPE_DESKTOP;
Aaron Durbin76c37002012-10-30 09:03:43 -050056}
57
Duncan Laurie1ad55642013-03-07 14:08:04 -080058u16 get_pmbase(void)
59{
60 static u16 pmbase;
61
62 if (!pmbase)
63 pmbase = pci_read_config16(pch_get_lpc_device(),
64 PMBASE) & 0xfffc;
65 return pmbase;
66}
67
68u16 get_gpiobase(void)
69{
70 static u16 gpiobase;
71
72 if (!gpiobase)
73 gpiobase = pci_read_config16(pch_get_lpc_device(),
74 GPIOBASE) & 0xfffc;
75 return gpiobase;
76}
77
Kyösti Mälkki21d6a272019-11-05 18:50:38 +020078#ifndef __SIMPLE_DEVICE__
Duncan Laurie5cc51c02013-03-07 14:06:43 -080079
Duncan Laurie98c40622013-05-21 16:37:40 -070080/* Put device in D3Hot Power State */
Elyes HAOUAS38f1d132018-09-17 08:44:18 +020081static void pch_enable_d3hot(struct device *dev)
Aaron Durbin76c37002012-10-30 09:03:43 -050082{
Angel Ponsbf9bc502020-06-08 00:12:43 +020083 pci_or_config32(dev, PCH_PCS, PCH_PCS_PS_D3HOT);
Duncan Laurie98c40622013-05-21 16:37:40 -070084}
85
Frans Hendrikse6bf51f2019-05-01 10:48:31 +020086/* Set bit in function disable register to hide this device */
Elyes HAOUAS38f1d132018-09-17 08:44:18 +020087void pch_disable_devfn(struct device *dev)
Duncan Laurie98c40622013-05-21 16:37:40 -070088{
89 switch (dev->path.pci.devfn) {
Duncan Laurie26e7dd72012-12-19 09:12:31 -080090 case PCI_DEVFN(19, 0): /* Audio DSP */
91 RCBA32_OR(FD, PCH_DISABLE_ADSPD);
92 break;
93 case PCI_DEVFN(20, 0): /* XHCI */
94 RCBA32_OR(FD, PCH_DISABLE_XHCI);
95 break;
Duncan Laurie71346c02013-01-10 13:20:40 -080096 case PCI_DEVFN(21, 0): /* DMA */
Duncan Laurie98c40622013-05-21 16:37:40 -070097 pch_enable_d3hot(dev);
Angel Pons8963f7d2020-10-24 12:20:28 +020098 pch_iobp_update(SIO_IOBP_FUNCDIS0, ~0, SIO_IOBP_FUNCDIS_DIS);
Duncan Laurie71346c02013-01-10 13:20:40 -080099 break;
100 case PCI_DEVFN(21, 1): /* I2C0 */
Duncan Laurie98c40622013-05-21 16:37:40 -0700101 pch_enable_d3hot(dev);
Angel Pons8963f7d2020-10-24 12:20:28 +0200102 pch_iobp_update(SIO_IOBP_FUNCDIS1, ~0, SIO_IOBP_FUNCDIS_DIS);
Duncan Laurie71346c02013-01-10 13:20:40 -0800103 break;
104 case PCI_DEVFN(21, 2): /* I2C1 */
Duncan Laurie98c40622013-05-21 16:37:40 -0700105 pch_enable_d3hot(dev);
Angel Pons8963f7d2020-10-24 12:20:28 +0200106 pch_iobp_update(SIO_IOBP_FUNCDIS2, ~0, SIO_IOBP_FUNCDIS_DIS);
Duncan Laurie71346c02013-01-10 13:20:40 -0800107 break;
108 case PCI_DEVFN(21, 3): /* SPI0 */
Duncan Laurie98c40622013-05-21 16:37:40 -0700109 pch_enable_d3hot(dev);
Angel Pons8963f7d2020-10-24 12:20:28 +0200110 pch_iobp_update(SIO_IOBP_FUNCDIS3, ~0, SIO_IOBP_FUNCDIS_DIS);
Duncan Laurie71346c02013-01-10 13:20:40 -0800111 break;
112 case PCI_DEVFN(21, 4): /* SPI1 */
Duncan Laurie98c40622013-05-21 16:37:40 -0700113 pch_enable_d3hot(dev);
Angel Pons8963f7d2020-10-24 12:20:28 +0200114 pch_iobp_update(SIO_IOBP_FUNCDIS4, ~0, SIO_IOBP_FUNCDIS_DIS);
Duncan Laurie71346c02013-01-10 13:20:40 -0800115 break;
116 case PCI_DEVFN(21, 5): /* UART0 */
Duncan Laurie98c40622013-05-21 16:37:40 -0700117 pch_enable_d3hot(dev);
Angel Pons8963f7d2020-10-24 12:20:28 +0200118 pch_iobp_update(SIO_IOBP_FUNCDIS5, ~0, SIO_IOBP_FUNCDIS_DIS);
Duncan Laurie71346c02013-01-10 13:20:40 -0800119 break;
120 case PCI_DEVFN(21, 6): /* UART1 */
Duncan Laurie98c40622013-05-21 16:37:40 -0700121 pch_enable_d3hot(dev);
Angel Pons8963f7d2020-10-24 12:20:28 +0200122 pch_iobp_update(SIO_IOBP_FUNCDIS6, ~0, SIO_IOBP_FUNCDIS_DIS);
Duncan Laurie71346c02013-01-10 13:20:40 -0800123 break;
Aaron Durbin76c37002012-10-30 09:03:43 -0500124 case PCI_DEVFN(22, 0): /* MEI #1 */
125 RCBA32_OR(FD2, PCH_DISABLE_MEI1);
126 break;
127 case PCI_DEVFN(22, 1): /* MEI #2 */
128 RCBA32_OR(FD2, PCH_DISABLE_MEI2);
129 break;
130 case PCI_DEVFN(22, 2): /* IDE-R */
131 RCBA32_OR(FD2, PCH_DISABLE_IDER);
132 break;
133 case PCI_DEVFN(22, 3): /* KT */
134 RCBA32_OR(FD2, PCH_DISABLE_KT);
135 break;
Duncan Laurie71346c02013-01-10 13:20:40 -0800136 case PCI_DEVFN(23, 0): /* SDIO */
Duncan Laurie98c40622013-05-21 16:37:40 -0700137 pch_enable_d3hot(dev);
Angel Pons8963f7d2020-10-24 12:20:28 +0200138 pch_iobp_update(SIO_IOBP_FUNCDIS7, ~0, SIO_IOBP_FUNCDIS_DIS);
Duncan Laurie71346c02013-01-10 13:20:40 -0800139 break;
Aaron Durbin76c37002012-10-30 09:03:43 -0500140 case PCI_DEVFN(25, 0): /* Gigabit Ethernet */
141 RCBA32_OR(BUC, PCH_DISABLE_GBE);
142 break;
143 case PCI_DEVFN(26, 0): /* EHCI #2 */
144 RCBA32_OR(FD, PCH_DISABLE_EHCI2);
145 break;
146 case PCI_DEVFN(27, 0): /* HD Audio Controller */
147 RCBA32_OR(FD, PCH_DISABLE_HD_AUDIO);
148 break;
149 case PCI_DEVFN(28, 0): /* PCI Express Root Port 1 */
150 case PCI_DEVFN(28, 1): /* PCI Express Root Port 2 */
151 case PCI_DEVFN(28, 2): /* PCI Express Root Port 3 */
152 case PCI_DEVFN(28, 3): /* PCI Express Root Port 4 */
153 case PCI_DEVFN(28, 4): /* PCI Express Root Port 5 */
154 case PCI_DEVFN(28, 5): /* PCI Express Root Port 6 */
155 case PCI_DEVFN(28, 6): /* PCI Express Root Port 7 */
156 case PCI_DEVFN(28, 7): /* PCI Express Root Port 8 */
Duncan Laurie98c40622013-05-21 16:37:40 -0700157 RCBA32_OR(FD, PCH_DISABLE_PCIE(PCI_FUNC(dev->path.pci.devfn)));
Aaron Durbin76c37002012-10-30 09:03:43 -0500158 break;
159 case PCI_DEVFN(29, 0): /* EHCI #1 */
160 RCBA32_OR(FD, PCH_DISABLE_EHCI1);
161 break;
Aaron Durbin76c37002012-10-30 09:03:43 -0500162 case PCI_DEVFN(31, 0): /* LPC */
163 RCBA32_OR(FD, PCH_DISABLE_LPC);
164 break;
165 case PCI_DEVFN(31, 2): /* SATA #1 */
166 RCBA32_OR(FD, PCH_DISABLE_SATA1);
167 break;
168 case PCI_DEVFN(31, 3): /* SMBUS */
169 RCBA32_OR(FD, PCH_DISABLE_SMBUS);
170 break;
Duncan Laurie26e7dd72012-12-19 09:12:31 -0800171 case PCI_DEVFN(31, 5): /* SATA #2 */
Aaron Durbin76c37002012-10-30 09:03:43 -0500172 RCBA32_OR(FD, PCH_DISABLE_SATA2);
173 break;
174 case PCI_DEVFN(31, 6): /* Thermal Subsystem */
175 RCBA32_OR(FD, PCH_DISABLE_THERMAL);
176 break;
177 }
178}
179
Elyes HAOUAS38f1d132018-09-17 08:44:18 +0200180void pch_enable(struct device *dev)
Aaron Durbin76c37002012-10-30 09:03:43 -0500181{
Aaron Durbinc0254e62013-06-20 01:20:30 -0500182 /* PCH PCIe Root Ports are handled in PCIe driver. */
Aaron Durbin76c37002012-10-30 09:03:43 -0500183 if (PCI_SLOT(dev->path.pci.devfn) == PCH_PCIE_DEV_SLOT)
Aaron Durbinc0254e62013-06-20 01:20:30 -0500184 return;
Aaron Durbin76c37002012-10-30 09:03:43 -0500185
186 if (!dev->enabled) {
Elyes Haouas8b8ada62022-11-22 17:36:02 +0100187 printk(BIOS_DEBUG, "%s: Disabling device\n", dev_path(dev));
Aaron Durbin76c37002012-10-30 09:03:43 -0500188
189 /* Ensure memory, io, and bus master are all disabled */
Angel Ponsbf9bc502020-06-08 00:12:43 +0200190 pci_and_config16(dev, PCI_COMMAND,
191 ~(PCI_COMMAND_MASTER | PCI_COMMAND_MEMORY | PCI_COMMAND_IO));
Aaron Durbin76c37002012-10-30 09:03:43 -0500192
Aaron Durbin3fcd3562013-06-19 13:20:37 -0500193 /* Disable this device if possible */
194 pch_disable_devfn(dev);
Aaron Durbin76c37002012-10-30 09:03:43 -0500195 } else {
196 /* Enable SERR */
Elyes HAOUAS73ae0762020-04-28 10:13:05 +0200197 pci_or_config16(dev, PCI_COMMAND, PCI_COMMAND_SERR);
Aaron Durbin76c37002012-10-30 09:03:43 -0500198 }
199}
200
201struct chip_operations southbridge_intel_lynxpoint_ops = {
202 CHIP_NAME("Intel Series 8 (Lynx Point) Southbridge")
203 .enable_dev = pch_enable,
204};
Duncan Laurie5cc51c02013-03-07 14:06:43 -0800205
Kyösti Mälkki21d6a272019-11-05 18:50:38 +0200206#endif /* __SIMPLE_DEVICE__ */