Commit ea9a8854 authored by Alexander Duyck's avatar Alexander Duyck Committed by Bjorn Helgaas

PCI: Set SR-IOV NumVFs to zero after enumeration

The enumeration path should leave NumVFs set to zero.  But after
4449f079 ("PCI: Calculate maximum number of buses required for VFs"),
we call virtfn_max_buses() in the enumeration path, which changes NumVFs.
This NumVFs change is visible via lspci and sysfs until a driver enables
SR-IOV.

Iterate from TotalVFs down to zero so NumVFs is zero when we're finished
computing the maximum number of buses.  Validate offset and stride in
the loop, so we can test it at every possible NumVFs setting.  Rename
virtfn_max_buses() to compute_max_vf_buses() to hint that it does have a
side effect of updating iov->max_VF_buses.

[bhelgaas: changelog, rename, allow numVF==1 && stride==0, rework loop,
reverse sense of error path]
Fixes: 4449f079 ("PCI: Calculate maximum number of buses required for VFs")
Based-on-patch-by: default avatarEthan Zhao <ethan.zhao@oracle.com>
Signed-off-by: default avatarAlexander Duyck <aduyck@mirantis.com>
Signed-off-by: default avatarBjorn Helgaas <bhelgaas@google.com>
parent ff45f9dd
...@@ -54,24 +54,29 @@ static inline void pci_iov_set_numvfs(struct pci_dev *dev, int nr_virtfn) ...@@ -54,24 +54,29 @@ static inline void pci_iov_set_numvfs(struct pci_dev *dev, int nr_virtfn)
* The PF consumes one bus number. NumVFs, First VF Offset, and VF Stride * The PF consumes one bus number. NumVFs, First VF Offset, and VF Stride
* determine how many additional bus numbers will be consumed by VFs. * determine how many additional bus numbers will be consumed by VFs.
* *
* Iterate over all valid NumVFs and calculate the maximum number of bus * Iterate over all valid NumVFs, validate offset and stride, and calculate
* numbers that could ever be required. * the maximum number of bus numbers that could ever be required.
*/ */
static inline u8 virtfn_max_buses(struct pci_dev *dev) static int compute_max_vf_buses(struct pci_dev *dev)
{ {
struct pci_sriov *iov = dev->sriov; struct pci_sriov *iov = dev->sriov;
int nr_virtfn; int nr_virtfn, busnr, rc = 0;
u8 max = 0;
int busnr;
for (nr_virtfn = 1; nr_virtfn <= iov->total_VFs; nr_virtfn++) { for (nr_virtfn = iov->total_VFs; nr_virtfn; nr_virtfn--) {
pci_iov_set_numvfs(dev, nr_virtfn); pci_iov_set_numvfs(dev, nr_virtfn);
if (!iov->offset || (nr_virtfn > 1 && !iov->stride)) {
rc = -EIO;
goto out;
}
busnr = pci_iov_virtfn_bus(dev, nr_virtfn - 1); busnr = pci_iov_virtfn_bus(dev, nr_virtfn - 1);
if (busnr > max) if (busnr > iov->max_VF_buses)
max = busnr; iov->max_VF_buses = busnr;
} }
return max; out:
pci_iov_set_numvfs(dev, 0);
return rc;
} }
static struct pci_bus *virtfn_add_bus(struct pci_bus *bus, int busnr) static struct pci_bus *virtfn_add_bus(struct pci_bus *bus, int busnr)
...@@ -384,7 +389,7 @@ static int sriov_init(struct pci_dev *dev, int pos) ...@@ -384,7 +389,7 @@ static int sriov_init(struct pci_dev *dev, int pos)
int rc; int rc;
int nres; int nres;
u32 pgsz; u32 pgsz;
u16 ctrl, total, offset, stride; u16 ctrl, total;
struct pci_sriov *iov; struct pci_sriov *iov;
struct resource *res; struct resource *res;
struct pci_dev *pdev; struct pci_dev *pdev;
...@@ -410,11 +415,6 @@ static int sriov_init(struct pci_dev *dev, int pos) ...@@ -410,11 +415,6 @@ static int sriov_init(struct pci_dev *dev, int pos)
found: found:
pci_write_config_word(dev, pos + PCI_SRIOV_CTRL, ctrl); pci_write_config_word(dev, pos + PCI_SRIOV_CTRL, ctrl);
pci_write_config_word(dev, pos + PCI_SRIOV_NUM_VF, 0);
pci_read_config_word(dev, pos + PCI_SRIOV_VF_OFFSET, &offset);
pci_read_config_word(dev, pos + PCI_SRIOV_VF_STRIDE, &stride);
if (!offset || (total > 1 && !stride))
return -EIO;
pci_read_config_word(dev, pos + PCI_SRIOV_TOTAL_VF, &total); pci_read_config_word(dev, pos + PCI_SRIOV_TOTAL_VF, &total);
if (!total) if (!total)
...@@ -456,8 +456,6 @@ static int sriov_init(struct pci_dev *dev, int pos) ...@@ -456,8 +456,6 @@ static int sriov_init(struct pci_dev *dev, int pos)
iov->nres = nres; iov->nres = nres;
iov->ctrl = ctrl; iov->ctrl = ctrl;
iov->total_VFs = total; iov->total_VFs = total;
iov->offset = offset;
iov->stride = stride;
iov->pgsz = pgsz; iov->pgsz = pgsz;
iov->self = dev; iov->self = dev;
pci_read_config_dword(dev, pos + PCI_SRIOV_CAP, &iov->cap); pci_read_config_dword(dev, pos + PCI_SRIOV_CAP, &iov->cap);
...@@ -474,10 +472,15 @@ static int sriov_init(struct pci_dev *dev, int pos) ...@@ -474,10 +472,15 @@ static int sriov_init(struct pci_dev *dev, int pos)
dev->sriov = iov; dev->sriov = iov;
dev->is_physfn = 1; dev->is_physfn = 1;
iov->max_VF_buses = virtfn_max_buses(dev); rc = compute_max_vf_buses(dev);
if (rc)
goto fail_max_buses;
return 0; return 0;
fail_max_buses:
dev->sriov = NULL;
dev->is_physfn = 0;
failed: failed:
for (i = 0; i < PCI_SRIOV_NUM_BARS; i++) { for (i = 0; i < PCI_SRIOV_NUM_BARS; i++) {
res = &dev->resource[i + PCI_IOV_RESOURCES]; res = &dev->resource[i + PCI_IOV_RESOURCES];
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment