1 /* SPDX-License-Identifier: GPL-2.0-or-later */
3 #include <console/console.h>
4 #include <device/pci_def.h>
5 #include <device/pci_ops.h>
6 #include <southbridge/intel/lynxpoint/hsio/hsio.h>
7 #include <southbridge/intel/lynxpoint/pch.h>
10 static inline bool is_9_series_pch_h(void)
12 const uint16_t devid
= pci_read_config16(PCH_LPC_DEV
, PCI_DEVICE_ID
);
13 return (devid
& 0xfff0) == 0x8cc0;
16 static void early_sata_init(const uint8_t pch_revision
)
18 const bool is_mobile
= get_pch_platform_type() != PCH_TYPE_DESKTOP
;
20 const uint8_t lane_owner
= pci_read_config8(PCH_PCIE_DEV(0), 0x410);
21 printk(BIOS_DEBUG
, "HSIO lane owner: 0x%02x\n", lane_owner
);
24 pci_update_config32(PCH_SATA_DEV
, SATA_SCLKG
, ~0x1ff, 0x183);
26 /* BWG Step 3: Set OOB Retry Mode */
27 pci_or_config16(PCH_SATA_DEV
, SATA_PCS
, 1 << 15);
29 /* BWG Step 4: Program the SATA mPHY tables */
31 if (pch_revision
>= LPT_LP_STEP_B0
&& pch_revision
<= LPT_LP_STEP_B2
) {
32 program_hsio_sata_lpt_lp_bx(is_mobile
);
34 printk(BIOS_ERR
, "Unsupported PCH-LP stepping 0x%02x\n", pch_revision
);
37 /** FIXME: Figure out HSIO settings for 9 series PCH-H **/
38 if (pch_revision
>= LPT_H_STEP_C0
|| is_9_series_pch_h()) {
39 program_hsio_sata_lpt_h_cx(is_mobile
);
41 printk(BIOS_ERR
, "Unsupported PCH-H stepping 0x%02x\n", pch_revision
);
45 /** FIXME: Program SATA RxEq tables **/
48 /** FIXME: Only for desktop and mobile (skip this on workstation and server) **/
49 pci_or_config32(PCH_SATA_DEV
, 0x98, BIT(22));
52 pci_or_config32(PCH_SATA_DEV
, 0x98, BIT(19));
55 pci_update_config32(PCH_SATA_DEV
, 0x98, ~(0x3f << 7), 0x04 << 7);
58 pci_or_config32(PCH_SATA_DEV
, 0x98, BIT(20));
61 pci_update_config32(PCH_SATA_DEV
, 0x98, ~(3 << 5), 1 << 5);
64 pci_or_config32(PCH_SATA_DEV
, 0x98, BIT(18));
66 /* Enable SATA ports */
68 if (CONFIG(INTEL_LYNXPOINT_LP
)) {
69 for (uint8_t i
= 0; i
< 4; i
++) {
70 if ((lane_owner
& BIT(7 - i
)) == 0) {
76 for (uint8_t i
= 4; i
< 6; i
++) {
77 if ((lane_owner
& BIT(i
)) == 0) {
82 printk(BIOS_DEBUG
, "SATA port enables: 0x%02x\n", sata_pcs
);
83 pci_or_config8(PCH_SATA_DEV
, SATA_PCS
, sata_pcs
);
86 void early_pch_init_native(int s3resume
)
88 const uint8_t pch_revision
= pci_read_config8(PCH_LPC_DEV
, PCI_REVISION_ID
);
90 RCBA16(DISPBDF
) = 0x0010;
91 RCBA32_OR(FD2
, PCH_ENABLE_DBDF
);
93 /** FIXME: Check GEN_PMCON_3 and handle RTC failure? **/
95 RCBA32(PRSTS
) = BIT(4);
97 early_sata_init(pch_revision
);
99 pci_or_config8(PCH_LPC_DEV
, 0xa6, 1 << 1);
100 pci_and_config8(PCH_LPC_DEV
, 0xdc, ~(1 << 5 | 1 << 1));
102 /** TODO: Send GET HSIO VER and update ChipsetInit table? Is it needed? **/
104 /** FIXME: GbE handling? **/
106 pci_update_config32(PCH_LPC_DEV
, 0xac, ~(1 << 20), 0);
108 for (uint8_t i
= 0; i
< 8; i
++)
109 pci_update_config32(PCH_PCIE_DEV(i
), 0x338, ~(1 << 26), 0);
111 pci_update_config8(PCH_PCIE_DEV(0), 0xf4, ~(3 << 5), 1 << 7);
113 pci_update_config8(PCH_EHCI1_DEV
, 0x88, ~(1 << 2), 0);
115 pci_update_config8(PCH_EHCI2_DEV
, 0x88, ~(1 << 2), 0);
117 /** FIXME: Disable SATA2 device? **/
120 if (pch_revision
>= LPT_LP_STEP_B0
&& pch_revision
<= LPT_LP_STEP_B2
) {
121 program_hsio_xhci_lpt_lp_bx();
122 program_hsio_igbe_lpt_lp_bx();
124 printk(BIOS_ERR
, "Unsupported PCH-LP stepping 0x%02x\n", pch_revision
);
127 /** FIXME: Figure out HSIO settings for 9 series PCH-H **/
128 if (pch_revision
>= LPT_H_STEP_C0
|| is_9_series_pch_h()) {
129 program_hsio_xhci_lpt_h_cx();
130 program_hsio_igbe_lpt_h_cx();
132 printk(BIOS_ERR
, "Unsupported PCH-H stepping 0x%02x\n", pch_revision
);
136 early_thermal_init();
140 void pch_dmi_setup_physical_layer(void)
142 /** FIXME: We need to make sure the SA supports Gen2 as well **/
143 if ((RCBA32(0x21a4) & 0x0f) == 0x02) {
144 /* Set Gen 2 Common Clock N_FTS */
145 RCBA32_AND_OR(0x2340, ~0x00ff0000, 0x3a << 16);
147 /* Set Target Link Speed to DMI Gen2 */
148 RCBA8_AND_OR(DLCTL2
, ~0x07, 0x02);
152 #define VC_ACTIVE (1U << 31)
154 #define VCNEGPND (1 << 1)
156 void pch_dmi_tc_vc_mapping(const u32 vc0
, const u32 vc1
, const u32 vcp
, const u32 vcm
)
158 printk(BIOS_DEBUG
, "Programming PCH DMI VC/TC mappings...\n");
160 RCBA32_AND_OR(CIR0050
, ~(0xf << 20), 2 << 20);
162 RCBA32_OR(CIR0050
, 1 << 19 | 1 << 17);
164 RCBA32(CIR0050
); /* Ensure posted write hits */
166 /* Use the same virtual channel mapping on both ends of the DMI link */
169 RCBA32(V1CTL
); /* Ensure posted write hits */
171 RCBA32(VPCTL
); /* Ensure posted write hits */
174 /* Lock the registers */
175 RCBA32_OR(CIR0050
, 1U << 31);
176 RCBA32(CIR0050
); /* Ensure posted write hits */
178 printk(BIOS_DEBUG
, "Waiting for PCH DMI VC negotiation... ");
179 do {} while (RCBA16(V0STS
) & VCNEGPND
);
180 do {} while (RCBA16(V1STS
) & VCNEGPND
);
181 do {} while (RCBA16(VPSTS
) & VCNEGPND
);
182 do {} while (RCBA16(VMSTS
) & VCNEGPND
);
183 printk(BIOS_DEBUG
, "done!\n");