blob: 0c3653879b816c199e4554d2f974dc2b5aa1af2c [file] [log] [blame]
Uwe Hermannb80dbf02007-04-22 19:08:13 +00001/*
Stefan Reinauer7e61e452008-01-18 10:35:56 +00002 * This file is part of the coreboot project.
Uwe Hermannb80dbf02007-04-22 19:08:13 +00003 *
4 * Copyright (C) 2005 Linux Networx
5 * (Written by Eric Biederman <ebiederman@lnxi.com> for Linux Networx)
6 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; version 2 of the License.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
Uwe Hermannb80dbf02007-04-22 19:08:13 +000015 */
Yinghai Lu13f1c2a2005-07-08 02:49:49 +000016
17#include <console/console.h>
Duncan Laurie90dcdd42011-10-25 14:15:11 -070018#include <delay.h>
Yinghai Lu13f1c2a2005-07-08 02:49:49 +000019#include <device/device.h>
20#include <device/pci.h>
21#include <device/pci_ids.h>
22#include <device/pciexp.h>
23
Kenji Chen31c6e632014-10-04 01:14:44 +080024unsigned int pciexp_find_extended_cap(device_t dev, unsigned int cap)
25{
26 unsigned int this_cap_offset, next_cap_offset;
27 unsigned int this_cap, cafe;
28
29 this_cap_offset = PCIE_EXT_CAP_OFFSET;
30 do {
Kyösti Mälkki91bfa8e2016-11-20 20:39:56 +020031 this_cap = pci_read_config32(dev, this_cap_offset);
Kenji Chen31c6e632014-10-04 01:14:44 +080032 next_cap_offset = this_cap >> 20;
33 this_cap &= 0xffff;
Kyösti Mälkki91bfa8e2016-11-20 20:39:56 +020034 cafe = pci_read_config32(dev, this_cap_offset + 4);
Kenji Chen31c6e632014-10-04 01:14:44 +080035 cafe &= 0xffff;
36 if (this_cap == cap)
37 return this_cap_offset;
38 else if (cafe == cap)
39 return this_cap_offset + 4;
40 else
41 this_cap_offset = next_cap_offset;
42 } while (next_cap_offset != 0);
43
44 return 0;
45}
Kenji Chen31c6e632014-10-04 01:14:44 +080046
Duncan Laurie90dcdd42011-10-25 14:15:11 -070047/*
48 * Re-train a PCIe link
49 */
50#define PCIE_TRAIN_RETRY 10000
51static int pciexp_retrain_link(device_t dev, unsigned cap)
52{
53 unsigned try = PCIE_TRAIN_RETRY;
54 u16 lnk;
55
56 /* Start link retraining */
57 lnk = pci_read_config16(dev, cap + PCI_EXP_LNKCTL);
58 lnk |= PCI_EXP_LNKCTL_RL;
59 pci_write_config16(dev, cap + PCI_EXP_LNKCTL, lnk);
60
61 /* Wait for training to complete */
62 while (try--) {
63 lnk = pci_read_config16(dev, cap + PCI_EXP_LNKSTA);
64 if (!(lnk & PCI_EXP_LNKSTA_LT))
65 return 0;
66 udelay(100);
67 }
68
69 printk(BIOS_ERR, "%s: Link Retrain timeout\n", dev_path(dev));
70 return -1;
71}
72
73/*
74 * Check the Slot Clock Configuration for root port and endpoint
75 * and enable Common Clock Configuration if possible. If CCC is
76 * enabled the link must be retrained.
77 */
78static void pciexp_enable_common_clock(device_t root, unsigned root_cap,
79 device_t endp, unsigned endp_cap)
80{
81 u16 root_scc, endp_scc, lnkctl;
82
83 /* Get Slot Clock Configuration for root port */
84 root_scc = pci_read_config16(root, root_cap + PCI_EXP_LNKSTA);
85 root_scc &= PCI_EXP_LNKSTA_SLC;
86
87 /* Get Slot Clock Configuration for endpoint */
88 endp_scc = pci_read_config16(endp, endp_cap + PCI_EXP_LNKSTA);
89 endp_scc &= PCI_EXP_LNKSTA_SLC;
90
91 /* Enable Common Clock Configuration and retrain */
92 if (root_scc && endp_scc) {
93 printk(BIOS_INFO, "Enabling Common Clock Configuration\n");
94
95 /* Set in endpoint */
96 lnkctl = pci_read_config16(endp, endp_cap + PCI_EXP_LNKCTL);
97 lnkctl |= PCI_EXP_LNKCTL_CCC;
98 pci_write_config16(endp, endp_cap + PCI_EXP_LNKCTL, lnkctl);
99
100 /* Set in root port */
101 lnkctl = pci_read_config16(root, root_cap + PCI_EXP_LNKCTL);
102 lnkctl |= PCI_EXP_LNKCTL_CCC;
103 pci_write_config16(root, root_cap + PCI_EXP_LNKCTL, lnkctl);
104
105 /* Retrain link if CCC was enabled */
106 pciexp_retrain_link(root, root_cap);
107 }
108}
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700109
Kane Chen18cb1342014-10-01 11:13:54 +0800110static void pciexp_enable_clock_power_pm(device_t endp, unsigned endp_cap)
111{
112 /* check if per port clk req is supported in device */
113 u32 endp_ca;
114 u16 lnkctl;
115 endp_ca = pci_read_config32(endp, endp_cap + PCI_EXP_LNKCAP);
116 if ((endp_ca & PCI_EXP_CLK_PM) == 0) {
117 printk(BIOS_INFO, "PCIE CLK PM is not supported by endpoint");
118 return;
119 }
120 lnkctl = pci_read_config16(endp, endp_cap + PCI_EXP_LNKCTL);
121 lnkctl = lnkctl | PCI_EXP_EN_CLK_PM;
122 pci_write_config16(endp, endp_cap + PCI_EXP_LNKCTL, lnkctl);
123}
Kane Chen18cb1342014-10-01 11:13:54 +0800124
Kenji Chen31c6e632014-10-04 01:14:44 +0800125static void pciexp_config_max_latency(device_t root, device_t dev)
126{
127 unsigned int cap;
128 cap = pciexp_find_extended_cap(dev, PCIE_EXT_CAP_LTR_ID);
Pratik Prajapati0cd0d282015-06-09 12:06:20 -0700129 if ((cap) && (root->ops->ops_pci != NULL) &&
130 (root->ops->ops_pci->set_L1_ss_latency != NULL))
131 root->ops->ops_pci->set_L1_ss_latency(dev, cap + 4);
Kenji Chen31c6e632014-10-04 01:14:44 +0800132}
133
134static void pciexp_enable_ltr(device_t dev)
135{
136 unsigned int cap;
137 cap = pci_find_capability(dev, PCI_CAP_ID_PCIE);
Pratik Prajapati0cd0d282015-06-09 12:06:20 -0700138 if(!cap) {
139 printk(BIOS_INFO, "Failed to enable LTR for dev = %s\n",
140 dev_path(dev));
141 return;
142 }
Kyösti Mälkki48c389e2013-07-26 08:53:59 +0300143 pci_update_config32(dev, cap + 0x28, ~(1 << 10), 1 << 10);
Kenji Chen31c6e632014-10-04 01:14:44 +0800144}
145
146static unsigned char pciexp_L1_substate_cal(device_t dev, unsigned int endp_cap,
147 unsigned int *data)
148{
149 unsigned char mult[4] = {2, 10, 100, 0};
150
151 unsigned int L1SubStateSupport = *data & 0xf;
152 unsigned int comm_mode_rst_time = (*data >> 8) & 0xff;
153 unsigned int power_on_scale = (*data >> 16) & 0x3;
154 unsigned int power_on_value = (*data >> 19) & 0x1f;
155
Kyösti Mälkki91bfa8e2016-11-20 20:39:56 +0200156 unsigned int endp_data = pci_read_config32(dev, endp_cap + 4);
Kenji Chen31c6e632014-10-04 01:14:44 +0800157 unsigned int endp_L1SubStateSupport = endp_data & 0xf;
158 unsigned int endp_comm_mode_restore_time = (endp_data >> 8) & 0xff;
159 unsigned int endp_power_on_scale = (endp_data >> 16) & 0x3;
160 unsigned int endp_power_on_value = (endp_data >> 19) & 0x1f;
161
162 L1SubStateSupport &= endp_L1SubStateSupport;
163
164 if (L1SubStateSupport == 0)
165 return 0;
166
167 if (power_on_value * mult[power_on_scale] <
168 endp_power_on_value * mult[endp_power_on_scale]) {
169 power_on_value = endp_power_on_value;
170 power_on_scale = endp_power_on_scale;
171 }
172 if (comm_mode_rst_time < endp_comm_mode_restore_time)
173 comm_mode_rst_time = endp_comm_mode_restore_time;
174
175 *data = (comm_mode_rst_time << 8) | (power_on_scale << 16)
176 | (power_on_value << 19) | L1SubStateSupport;
177
178 return 1;
179}
180
181static void pciexp_L1_substate_commit(device_t root, device_t dev,
182 unsigned int root_cap, unsigned int end_cap)
183{
184 device_t dev_t;
185 unsigned char L1_ss_ok;
Kyösti Mälkki91bfa8e2016-11-20 20:39:56 +0200186 unsigned int rp_L1_support = pci_read_config32(root, root_cap + 4);
Kenji Chen31c6e632014-10-04 01:14:44 +0800187 unsigned int L1SubStateSupport;
188 unsigned int comm_mode_rst_time;
189 unsigned int power_on_scale;
190 unsigned int endp_power_on_value;
191
192 for (dev_t = dev; dev_t; dev_t = dev_t->sibling) {
193 /*
194 * rp_L1_support is init'd above from root port.
195 * it needs coordination with endpoints to reach in common.
196 * if certain endpoint doesn't support L1 Sub-State, abort
197 * this feature enabling.
198 */
199 L1_ss_ok = pciexp_L1_substate_cal(dev_t, end_cap,
200 &rp_L1_support);
201 if (!L1_ss_ok)
202 return;
203 }
204
205 L1SubStateSupport = rp_L1_support & 0xf;
206 comm_mode_rst_time = (rp_L1_support >> 8) & 0xff;
207 power_on_scale = (rp_L1_support >> 16) & 0x3;
208 endp_power_on_value = (rp_L1_support >> 19) & 0x1f;
209
210 printk(BIOS_INFO, "L1 Sub-State supported from root port %d\n",
211 root->path.pci.devfn >> 3);
212 printk(BIOS_INFO, "L1 Sub-State Support = 0x%x\n", L1SubStateSupport);
213 printk(BIOS_INFO, "CommonModeRestoreTime = 0x%x\n", comm_mode_rst_time);
214 printk(BIOS_INFO, "Power On Value = 0x%x, Power On Scale = 0x%x\n",
215 endp_power_on_value, power_on_scale);
216
217 pciexp_enable_ltr(root);
218
Kyösti Mälkki48c389e2013-07-26 08:53:59 +0300219 pci_update_config32(root, root_cap + 0x08, ~0xff00,
Kenji Chen31c6e632014-10-04 01:14:44 +0800220 (comm_mode_rst_time << 8));
221
Kyösti Mälkki48c389e2013-07-26 08:53:59 +0300222 pci_update_config32(root, root_cap + 0x0c , 0xffffff04,
Kenji Chen31c6e632014-10-04 01:14:44 +0800223 (endp_power_on_value << 3) | (power_on_scale));
224
Kyösti Mälkki48c389e2013-07-26 08:53:59 +0300225 pci_update_config32(root, root_cap + 0x08, ~0xe3ff0000,
Kenji Chen31c6e632014-10-04 01:14:44 +0800226 (1 << 21) | (1 << 23) | (1 << 30));
227
Kyösti Mälkki48c389e2013-07-26 08:53:59 +0300228 pci_update_config32(root, root_cap + 0x08, ~0x1f,
Kenji Chen31c6e632014-10-04 01:14:44 +0800229 L1SubStateSupport);
230
231 for (dev_t = dev; dev_t; dev_t = dev_t->sibling) {
Kyösti Mälkki48c389e2013-07-26 08:53:59 +0300232 pci_update_config32(dev_t, end_cap + 0x0c , 0xffffff04,
Kenji Chen31c6e632014-10-04 01:14:44 +0800233 (endp_power_on_value << 3) | (power_on_scale));
234
Kyösti Mälkki48c389e2013-07-26 08:53:59 +0300235 pci_update_config32(dev_t, end_cap + 0x08, ~0xe3ff0000,
Kenji Chen31c6e632014-10-04 01:14:44 +0800236 (1 << 21) | (1 << 23) | (1 << 30));
237
Kyösti Mälkki48c389e2013-07-26 08:53:59 +0300238 pci_update_config32(dev_t, end_cap + 0x08, ~0x1f,
Kenji Chen31c6e632014-10-04 01:14:44 +0800239 L1SubStateSupport);
240
241 pciexp_enable_ltr(dev_t);
242
243 pciexp_config_max_latency(root, dev_t);
244 }
245}
246
247static void pciexp_config_L1_sub_state(device_t root, device_t dev)
248{
249 unsigned int root_cap, end_cap;
250
251 /* Do it for function 0 only */
252 if (dev->path.pci.devfn & 0x7)
253 return;
254
255 root_cap = pciexp_find_extended_cap(root, PCIE_EXT_CAP_L1SS_ID);
256 if (!root_cap)
257 return;
258
259 end_cap = pciexp_find_extended_cap(dev, PCIE_EXT_CAP_L1SS_ID);
260 if (!end_cap) {
261 end_cap = pciexp_find_extended_cap(dev, 0xcafe);
262 if (!end_cap)
263 return;
264 }
265
266 pciexp_L1_substate_commit(root, dev, root_cap, end_cap);
267}
Kenji Chen31c6e632014-10-04 01:14:44 +0800268
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700269/*
270 * Determine the ASPM L0s or L1 exit latency for a link
271 * by checking both root port and endpoint and returning
272 * the highest latency value.
273 */
274static int pciexp_aspm_latency(device_t root, unsigned root_cap,
275 device_t endp, unsigned endp_cap,
276 enum aspm_type type)
277{
278 int root_lat = 0, endp_lat = 0;
279 u32 root_lnkcap, endp_lnkcap;
280
281 root_lnkcap = pci_read_config32(root, root_cap + PCI_EXP_LNKCAP);
282 endp_lnkcap = pci_read_config32(endp, endp_cap + PCI_EXP_LNKCAP);
283
284 /* Make sure the link supports this ASPM type by checking
285 * capability bits 11:10 with aspm_type offset by 1 */
286 if (!(root_lnkcap & (1 << (type + 9))) ||
287 !(endp_lnkcap & (1 << (type + 9))))
288 return -1;
289
290 /* Find the one with higher latency */
291 switch (type) {
292 case PCIE_ASPM_L0S:
293 root_lat = (root_lnkcap & PCI_EXP_LNKCAP_L0SEL) >> 12;
294 endp_lat = (endp_lnkcap & PCI_EXP_LNKCAP_L0SEL) >> 12;
295 break;
296 case PCIE_ASPM_L1:
297 root_lat = (root_lnkcap & PCI_EXP_LNKCAP_L1EL) >> 15;
298 endp_lat = (endp_lnkcap & PCI_EXP_LNKCAP_L1EL) >> 15;
299 break;
300 default:
301 return -1;
302 }
303
304 return (endp_lat > root_lat) ? endp_lat : root_lat;
305}
306
307/*
308 * Enable ASPM on PCIe root port and endpoint.
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700309 */
Kyösti Mälkkidf96a702016-11-29 16:08:38 +0200310static void pciexp_enable_aspm(device_t root, unsigned root_cap,
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700311 device_t endp, unsigned endp_cap)
312{
313 const char *aspm_type_str[] = { "None", "L0s", "L1", "L0s and L1" };
314 enum aspm_type apmc = PCIE_ASPM_NONE;
315 int exit_latency, ok_latency;
316 u16 lnkctl;
317 u32 devcap;
318
319 /* Get endpoint device capabilities for acceptable limits */
320 devcap = pci_read_config32(endp, endp_cap + PCI_EXP_DEVCAP);
321
322 /* Enable L0s if it is within endpoint acceptable limit */
323 ok_latency = (devcap & PCI_EXP_DEVCAP_L0S) >> 6;
324 exit_latency = pciexp_aspm_latency(root, root_cap, endp, endp_cap,
325 PCIE_ASPM_L0S);
326 if (exit_latency >= 0 && exit_latency <= ok_latency)
327 apmc |= PCIE_ASPM_L0S;
328
329 /* Enable L1 if it is within endpoint acceptable limit */
330 ok_latency = (devcap & PCI_EXP_DEVCAP_L1) >> 9;
331 exit_latency = pciexp_aspm_latency(root, root_cap, endp, endp_cap,
332 PCIE_ASPM_L1);
333 if (exit_latency >= 0 && exit_latency <= ok_latency)
334 apmc |= PCIE_ASPM_L1;
335
336 if (apmc != PCIE_ASPM_NONE) {
337 /* Set APMC in root port first */
338 lnkctl = pci_read_config16(root, root_cap + PCI_EXP_LNKCTL);
339 lnkctl |= apmc;
340 pci_write_config16(root, root_cap + PCI_EXP_LNKCTL, lnkctl);
341
342 /* Set APMC in endpoint device next */
343 lnkctl = pci_read_config16(endp, endp_cap + PCI_EXP_LNKCTL);
344 lnkctl |= apmc;
345 pci_write_config16(endp, endp_cap + PCI_EXP_LNKCTL, lnkctl);
Kyösti Mälkkidf96a702016-11-29 16:08:38 +0200346
347 /* Enable ASPM role based error reporting. */
348 devcap = pci_read_config32(endp, endp_cap + PCI_EXP_DEVCAP);
349 devcap |= PCI_EXP_DEVCAP_RBER;
350 pci_write_config32(endp, endp_cap + PCI_EXP_DEVCAP, devcap);
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700351 }
352
353 printk(BIOS_INFO, "ASPM: Enabled %s\n", aspm_type_str[apmc]);
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700354}
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700355
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000356static void pciexp_tune_dev(device_t dev)
357{
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700358 device_t root = dev->bus->dev;
359 unsigned int root_cap, cap;
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000360
361 cap = pci_find_capability(dev, PCI_CAP_ID_PCIE);
Uwe Hermannd453dd02010-10-18 00:00:57 +0000362 if (!cap)
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000363 return;
Uwe Hermannd453dd02010-10-18 00:00:57 +0000364
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700365 root_cap = pci_find_capability(root, PCI_CAP_ID_PCIE);
366 if (!root_cap)
367 return;
Stefan Reinauerf6eb88a2010-01-17 13:54:08 +0000368
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700369 /* Check for and enable Common Clock */
Kyösti Mälkki91bfa8e2016-11-20 20:39:56 +0200370 if (IS_ENABLED(CONFIG_PCIEXP_COMMON_CLOCK))
371 pciexp_enable_common_clock(root, root_cap, dev, cap);
Uwe Hermanne4870472010-11-04 23:23:47 +0000372
Kane Chen18cb1342014-10-01 11:13:54 +0800373 /* Check if per port CLK req is supported by endpoint*/
Kyösti Mälkki91bfa8e2016-11-20 20:39:56 +0200374 if (IS_ENABLED(CONFIG_PCIEXP_CLK_PM))
375 pciexp_enable_clock_power_pm(dev, cap);
Kane Chen18cb1342014-10-01 11:13:54 +0800376
Kenji Chen31c6e632014-10-04 01:14:44 +0800377 /* Enable L1 Sub-State when both root port and endpoint support */
Kyösti Mälkki91bfa8e2016-11-20 20:39:56 +0200378 if (IS_ENABLED(CONFIG_PCIEXP_L1_SUB_STATE))
379 pciexp_config_L1_sub_state(root, dev);
Kenji Chen31c6e632014-10-04 01:14:44 +0800380
Duncan Laurie90dcdd42011-10-25 14:15:11 -0700381 /* Check for and enable ASPM */
Kyösti Mälkki91bfa8e2016-11-20 20:39:56 +0200382 if (IS_ENABLED(CONFIG_PCIEXP_ASPM))
383 pciexp_enable_aspm(root, root_cap, dev, cap);
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000384}
385
Kyösti Mälkkide271a82015-03-18 13:09:47 +0200386void pciexp_scan_bus(struct bus *bus, unsigned int min_devfn,
387 unsigned int max_devfn)
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000388{
389 device_t child;
Uwe Hermannd453dd02010-10-18 00:00:57 +0000390
Kyösti Mälkkide271a82015-03-18 13:09:47 +0200391 pci_scan_bus(bus, min_devfn, max_devfn);
Uwe Hermannd453dd02010-10-18 00:00:57 +0000392
393 for (child = bus->children; child; child = child->sibling) {
394 if ((child->path.pci.devfn < min_devfn) ||
395 (child->path.pci.devfn > max_devfn)) {
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000396 continue;
397 }
398 pciexp_tune_dev(child);
399 }
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000400}
401
Kyösti Mälkki580e7222015-03-19 21:04:23 +0200402void pciexp_scan_bridge(device_t dev)
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000403{
Kyösti Mälkki580e7222015-03-19 21:04:23 +0200404 do_pci_scan_bridge(dev, pciexp_scan_bus);
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000405}
406
407/** Default device operations for PCI Express bridges */
408static struct pci_operations pciexp_bus_ops_pci = {
409 .set_subsystem = 0,
410};
411
412struct device_operations default_pciexp_ops_bus = {
413 .read_resources = pci_bus_read_resources,
414 .set_resources = pci_dev_set_resources,
415 .enable_resources = pci_bus_enable_resources,
Uwe Hermannd453dd02010-10-18 00:00:57 +0000416 .init = 0,
417 .scan_bus = pciexp_scan_bridge,
Yinghai Lu13f1c2a2005-07-08 02:49:49 +0000418 .enable = 0,
419 .reset_bus = pci_bus_reset,
420 .ops_pci = &pciexp_bus_ops_pci,
421};