aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/pci
diff options
context:
space:
mode:
authorRajat Jain <rajatxjain@gmail.com>2014-09-08 17:19:49 -0400
committerBjorn Helgaas <bhelgaas@google.com>2014-09-09 01:05:00 -0400
commit89665a6a71408796565bfd29cfa6a7877b17a667 (patch)
tree95fa4fcde378e302264775083c3234866956f0c9 /drivers/pci
parent52addcf9d6669fa439387610bc65c92fa0980cef (diff)
PCI: Check only the Vendor ID to identify Configuration Request Retry
Per PCIe r3.0, sec 2.3.2, if a Root Complex - has Configuration Request Retry Status Software Visibility enabled, - issues a Configuration Read of both bytes of the Vendor ID, and - receives a Completion with Configuration Request Retry Status (CRS), it must complete the request to the host by fabricating data of 0x0001 for the Vendor ID and 0xff for any additional bytes in the request. Linux issues a single config read for the four bytes containing the Vendor ID and the Device ID. Previously we checked all four bytes for 0xffff0001 to identify CRS. However, it is only the Vendor ID that really indicates CRS, because it's sufficient to read only those two bytes. Checking the Device ID verifies spec compliance but doesn't add any information. Some Root Complexes appear to indicate CRS by returning 0x0001 for the Vendor ID along with the actual the Device ID. Previously we interpreted that as a valid Vendor/Device ID pair, although 0x0001 is reserved and cannot be a valid Vendor ID. [bhelgaas: changelog] Link: http://lkml.kernel.org/r/4729FC36.3040000@gmail.com Signed-off-by: Rajat Jain <rajatxjain@gmail.com> Signed-off-by: Rajat Jain <rajatjain@juniper.net> Signed-off-by: Guenter Roeck <groeck@juniper.net> Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Diffstat (limited to 'drivers/pci')
-rw-r--r--drivers/pci/probe.c9
1 files changed, 7 insertions, 2 deletions
diff --git a/drivers/pci/probe.c b/drivers/pci/probe.c
index e3cf8a2e6292..9c26663c91d2 100644
--- a/drivers/pci/probe.c
+++ b/drivers/pci/probe.c
@@ -1292,8 +1292,13 @@ bool pci_bus_read_dev_vendor_id(struct pci_bus *bus, int devfn, u32 *l,
1292 *l == 0x0000ffff || *l == 0xffff0000) 1292 *l == 0x0000ffff || *l == 0xffff0000)
1293 return false; 1293 return false;
1294 1294
1295 /* Configuration request Retry Status */ 1295 /*
1296 while (*l == 0xffff0001) { 1296 * Configuration Request Retry Status. Some root ports return the
1297 * actual device ID instead of the synthetic ID (0xFFFF) required
1298 * by the PCIe spec. Ignore the device ID and only check for
1299 * (vendor id == 1).
1300 */
1301 while ((*l & 0xffff) == 0x0001) {
1297 if (!crs_timeout) 1302 if (!crs_timeout)
1298 return false; 1303 return false;
1299 1304