aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/base/bus.c
diff options
context:
space:
mode:
authorGreg Kroah-Hartman <gregkh@suse.de>2005-06-22 19:09:05 -0400
committerGreg Kroah-Hartman <gregkh@suse.de>2005-06-30 01:48:04 -0400
commitafdce75f1eaebcf358b7594ba7969aade105c3b0 (patch)
tree5374a0e85e03c8706a1dd95478b9d0a3312917e0 /drivers/base/bus.c
parent151ef38f7c0ec1b0420f04438b0316e3a30bf2e4 (diff)
[PATCH] driver core: Add the ability to bind drivers to devices from userspace
This adds a single file, "bind", to the sysfs directory of every driver registered with the driver core. To bind a device to a driver, write the bus id of the device you wish to bind to that specific driver to the "bind" file (remember to not add a trailing \n). If that bus id matches a device on that bus, and it does not currently have a driver bound to it, the probe sequence will be initiated with that driver and device. Note, this requires that the driver itself be willing and able to accept that device (usually through a device id type table). This patch does not make it possible to override the driver's id table. Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Diffstat (limited to 'drivers/base/bus.c')
-rw-r--r--drivers/base/bus.c26
1 files changed, 26 insertions, 0 deletions
diff --git a/drivers/base/bus.c b/drivers/base/bus.c
index fa41ee90a53..7e17488271a 100644
--- a/drivers/base/bus.c
+++ b/drivers/base/bus.c
@@ -160,6 +160,30 @@ static ssize_t driver_unbind(struct device_driver *drv,
160} 160}
161static DRIVER_ATTR(unbind, S_IWUSR, NULL, driver_unbind); 161static DRIVER_ATTR(unbind, S_IWUSR, NULL, driver_unbind);
162 162
163/*
164 * Manually attach a device to a driver.
165 * Note: the driver must want to bind to the device,
166 * it is not possible to override the driver's id table.
167 */
168static ssize_t driver_bind(struct device_driver *drv,
169 const char *buf, size_t count)
170{
171 struct bus_type *bus = get_bus(drv->bus);
172 struct device *dev;
173 int err = -ENODEV;
174
175 dev = bus_find_device(bus, NULL, (void *)buf, driver_helper);
176 if ((dev) &&
177 (dev->driver == NULL)) {
178 down(&dev->sem);
179 err = driver_probe_device(drv, dev);
180 up(&dev->sem);
181 put_device(dev);
182 }
183 return err;
184}
185static DRIVER_ATTR(bind, S_IWUSR, NULL, driver_bind);
186
163 187
164static struct device * next_device(struct klist_iter * i) 188static struct device * next_device(struct klist_iter * i)
165{ 189{
@@ -425,6 +449,7 @@ int bus_add_driver(struct device_driver * drv)
425 449
426 driver_add_attrs(bus, drv); 450 driver_add_attrs(bus, drv);
427 driver_create_file(drv, &driver_attr_unbind); 451 driver_create_file(drv, &driver_attr_unbind);
452 driver_create_file(drv, &driver_attr_bind);
428 } 453 }
429 return error; 454 return error;
430} 455}
@@ -442,6 +467,7 @@ int bus_add_driver(struct device_driver * drv)
442void bus_remove_driver(struct device_driver * drv) 467void bus_remove_driver(struct device_driver * drv)
443{ 468{
444 if (drv->bus) { 469 if (drv->bus) {
470 driver_remove_file(drv, &driver_attr_bind);
445 driver_remove_file(drv, &driver_attr_unbind); 471 driver_remove_file(drv, &driver_attr_unbind);
446 driver_remove_attrs(drv->bus, drv); 472 driver_remove_attrs(drv->bus, drv);
447 klist_remove(&drv->knode_bus); 473 klist_remove(&drv->knode_bus);