diff options
author | Yinghai Lu <yinghai@kernel.org> | 2013-01-21 16:20:51 -0500 |
---|---|---|
committer | Bjorn Helgaas <bhelgaas@google.com> | 2013-01-25 17:10:12 -0500 |
commit | 58d9a38f6facb28e935ec2747f6d9e9bf4684118 (patch) | |
tree | 252899cd333ccc3c23c2435acd3f377512092aa0 /drivers | |
parent | d59f53bc9bd80ee62072dea590fc623c67cb84a8 (diff) |
PCI: Skip attaching driver in device_add()
We want to add PCI devices to the device tree as early as possible but
delay attaching drivers.
device_add() adds a device to the device hierarchy and (via
device_attach()) attaches a matching driver and calls its .probe() method.
We want to separate adding the device to the hierarchy from attaching the
driver.
This patch does that by adding "match_driver" in struct pci_dev. When
false, we return failure from pci_bus_match(), which makes device_attach()
believe there's no matching driver.
Later, we set "match_driver = true" and call device_attach() again, which
now attaches the driver and calls its .probe() method.
[bhelgaas: changelog, explicitly init dev->match_driver,
fold device_attach() call into pci_bus_add_device()]
Signed-off-by: Yinghai Lu <yinghai@kernel.org>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Acked-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/pci/bus.c | 5 | ||||
-rw-r--r-- | drivers/pci/pci-driver.c | 6 |
2 files changed, 10 insertions, 1 deletions
diff --git a/drivers/pci/bus.c b/drivers/pci/bus.c index 847f3ca47bb8..c8709c6fdb7c 100644 --- a/drivers/pci/bus.c +++ b/drivers/pci/bus.c | |||
@@ -177,10 +177,15 @@ int pci_bus_add_device(struct pci_dev *dev) | |||
177 | if (retval) | 177 | if (retval) |
178 | return retval; | 178 | return retval; |
179 | 179 | ||
180 | dev->match_driver = false; | ||
180 | retval = device_add(&dev->dev); | 181 | retval = device_add(&dev->dev); |
181 | if (retval) | 182 | if (retval) |
182 | return retval; | 183 | return retval; |
183 | 184 | ||
185 | dev->match_driver = true; | ||
186 | retval = device_attach(&dev->dev); | ||
187 | WARN_ON(retval < 0); | ||
188 | |||
184 | dev->is_added = 1; | 189 | dev->is_added = 1; |
185 | pci_proc_attach_device(dev); | 190 | pci_proc_attach_device(dev); |
186 | pci_create_sysfs_dev_files(dev); | 191 | pci_create_sysfs_dev_files(dev); |
diff --git a/drivers/pci/pci-driver.c b/drivers/pci/pci-driver.c index f79cbcd3944b..acdcc3c6ecdd 100644 --- a/drivers/pci/pci-driver.c +++ b/drivers/pci/pci-driver.c | |||
@@ -1186,9 +1186,13 @@ pci_dev_driver(const struct pci_dev *dev) | |||
1186 | static int pci_bus_match(struct device *dev, struct device_driver *drv) | 1186 | static int pci_bus_match(struct device *dev, struct device_driver *drv) |
1187 | { | 1187 | { |
1188 | struct pci_dev *pci_dev = to_pci_dev(dev); | 1188 | struct pci_dev *pci_dev = to_pci_dev(dev); |
1189 | struct pci_driver *pci_drv = to_pci_driver(drv); | 1189 | struct pci_driver *pci_drv; |
1190 | const struct pci_device_id *found_id; | 1190 | const struct pci_device_id *found_id; |
1191 | 1191 | ||
1192 | if (!pci_dev->match_driver) | ||
1193 | return 0; | ||
1194 | |||
1195 | pci_drv = to_pci_driver(drv); | ||
1192 | found_id = pci_match_device(pci_drv, pci_dev); | 1196 | found_id = pci_match_device(pci_drv, pci_dev); |
1193 | if (found_id) | 1197 | if (found_id) |
1194 | return 1; | 1198 | return 1; |