|
@@ -244,6 +244,24 @@ struct bus_attribute pci_bus_attrs[] = {
|
|
|
__ATTR_NULL
|
|
|
};
|
|
|
|
|
|
+static ssize_t
|
|
|
+dev_rescan_store(struct device *dev, struct device_attribute *attr,
|
|
|
+ const char *buf, size_t count)
|
|
|
+{
|
|
|
+ unsigned long val;
|
|
|
+ struct pci_dev *pdev = to_pci_dev(dev);
|
|
|
+
|
|
|
+ if (strict_strtoul(buf, 0, &val) < 0)
|
|
|
+ return -EINVAL;
|
|
|
+
|
|
|
+ if (val) {
|
|
|
+ mutex_lock(&pci_remove_rescan_mutex);
|
|
|
+ pci_rescan_bus(pdev->bus);
|
|
|
+ mutex_unlock(&pci_remove_rescan_mutex);
|
|
|
+ }
|
|
|
+ return count;
|
|
|
+}
|
|
|
+
|
|
|
static void remove_callback(struct device *dev)
|
|
|
{
|
|
|
struct pci_dev *pdev = to_pci_dev(dev);
|
|
@@ -298,6 +316,7 @@ struct device_attribute pci_dev_attrs[] = {
|
|
|
__ATTR(msi_bus, 0644, msi_bus_show, msi_bus_store),
|
|
|
#ifdef CONFIG_HOTPLUG
|
|
|
__ATTR(remove, (S_IWUSR|S_IWGRP), NULL, remove_store),
|
|
|
+ __ATTR(rescan, (S_IWUSR|S_IWGRP), NULL, dev_rescan_store),
|
|
|
#endif
|
|
|
__ATTR_NULL,
|
|
|
};
|