Commit faa48a50 authored by Bjorn Helgaas's avatar Bjorn Helgaas

PCI: Remove spurious error for sriov_numvfs store and simplify flow

If we request "num_vfs" and the driver's sriov_configure() method enables
exactly that number ("num_vfs_enabled"), we complain "Invalid value for
number of VFs to enable" and return an error.  We should silently return
success instead.

Also, use kstrtou16() since numVFs is defined to be a 16-bit field and
rework to simplify control flow.
Reported-by: default avatarGreg Rose <gregory.v.rose@intel.com>
Reference: http://lkml.kernel.org/r/20121214101911.00002f59@unknownSigned-off-by: default avatarBjorn Helgaas <bhelgaas@google.com>
Tested-by: default avatarDonald Dutile <ddutile@redhat.com>
parent a49f0d1e
...@@ -422,77 +422,60 @@ static ssize_t sriov_numvfs_show(struct device *dev, ...@@ -422,77 +422,60 @@ static ssize_t sriov_numvfs_show(struct device *dev,
} }
/* /*
* num_vfs > 0; number of vfs to enable * num_vfs > 0; number of VFs to enable
* num_vfs = 0; disable all vfs * num_vfs = 0; disable all VFs
* *
* Note: SRIOV spec doesn't allow partial VF * Note: SRIOV spec doesn't allow partial VF
* disable, so its all or none. * disable, so it's all or none.
*/ */
static ssize_t sriov_numvfs_store(struct device *dev, static ssize_t sriov_numvfs_store(struct device *dev,
struct device_attribute *attr, struct device_attribute *attr,
const char *buf, size_t count) const char *buf, size_t count)
{ {
struct pci_dev *pdev = to_pci_dev(dev); struct pci_dev *pdev = to_pci_dev(dev);
int num_vfs_enabled = 0; int ret;
int num_vfs; u16 num_vfs;
int ret = 0;
u16 total;
if (kstrtoint(buf, 0, &num_vfs) < 0) ret = kstrtou16(buf, 0, &num_vfs);
return -EINVAL; if (ret < 0)
return ret;
if (num_vfs > pci_sriov_get_totalvfs(pdev))
return -ERANGE;
if (num_vfs == pdev->sriov->num_VFs)
return count; /* no change */
/* is PF driver loaded w/callback */ /* is PF driver loaded w/callback */
if (!pdev->driver || !pdev->driver->sriov_configure) { if (!pdev->driver || !pdev->driver->sriov_configure) {
dev_info(&pdev->dev, dev_info(&pdev->dev, "Driver doesn't support SRIOV configuration via sysfs\n");
"Driver doesn't support SRIOV configuration via sysfs\n");
return -ENOSYS; return -ENOSYS;
} }
/* if enabling vf's ... */ if (num_vfs == 0) {
total = pci_sriov_get_totalvfs(pdev); /* disable VFs */
/* Requested VFs to enable < totalvfs and none enabled already */ ret = pdev->driver->sriov_configure(pdev, 0);
if ((num_vfs > 0) && (num_vfs <= total)) { if (ret < 0)
if (pdev->sriov->num_VFs == 0) { return ret;
num_vfs_enabled = return count;
pdev->driver->sriov_configure(pdev, num_vfs);
if ((num_vfs_enabled >= 0) &&
(num_vfs_enabled != num_vfs)) {
dev_warn(&pdev->dev,
"Only %d VFs enabled\n",
num_vfs_enabled);
return count;
} else if (num_vfs_enabled < 0)
/* error code from driver callback */
return num_vfs_enabled;
} else if (num_vfs == pdev->sriov->num_VFs) {
dev_warn(&pdev->dev,
"%d VFs already enabled; no enable action taken\n",
num_vfs);
return count;
} else {
dev_warn(&pdev->dev,
"%d VFs already enabled. Disable before enabling %d VFs\n",
pdev->sriov->num_VFs, num_vfs);
return -EINVAL;
}
} }
/* disable vfs */ /* enable VFs */
if (num_vfs == 0) { if (pdev->sriov->num_VFs) {
if (pdev->sriov->num_VFs != 0) { dev_warn(&pdev->dev, "%d VFs already enabled. Disable before enabling %d VFs\n",
ret = pdev->driver->sriov_configure(pdev, 0); pdev->sriov->num_VFs, num_vfs);
return ret ? ret : count; return -EBUSY;
} else {
dev_warn(&pdev->dev,
"All VFs disabled; no disable action taken\n");
return count;
}
} }
dev_err(&pdev->dev, ret = pdev->driver->sriov_configure(pdev, num_vfs);
"Invalid value for number of VFs to enable: %d\n", num_vfs); if (ret < 0)
return ret;
return -EINVAL; if (ret != num_vfs)
dev_warn(&pdev->dev, "%d VFs requested; only %d enabled\n",
num_vfs, ret);
return count;
} }
static struct device_attribute sriov_totalvfs_attr = __ATTR_RO(sriov_totalvfs); static struct device_attribute sriov_totalvfs_attr = __ATTR_RO(sriov_totalvfs);
......
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