Merge master.kernel.org:/pub/scm/linux/kernel/git/gregkh/driver-2.6

* master.kernel.org:/pub/scm/linux/kernel/git/gregkh/driver-2.6: (46 commits)
  dev_dbg: check dev_dbg() arguments
  drivers/base/attribute_container.c: use mutex instead of binary semaphore
  mod_sysfs_setup() doesn't return errno when kobject_add_dir() failure occurs
  s2ram: add arch irq disable/enable hooks
  define platform wakeup hook, use in pci_enable_wake()
  security: prevent permission checking of file removal via sysfs_remove_group()
  device_schedule_callback() needs a module reference
  s390: cio: Delay uevents for subchannels
  sysfs: bin.c printk fix
  Driver core: use mutex instead of semaphore in DMA pool handler
  driver core: bus_add_driver should return an error if no bus
  debugfs: Add debugfs_create_u64()
  the overdue removal of the mount/umount uevents
  kobject: Comment and warning fixes to kobject.c
  Driver core: warn when userspace writes to the uevent file in a non-supported way
  Driver core: make uevent-environment available in uevent-file
  kobject core: remove rwsem from struct subsystem
  qeth: Remove usage of subsys.rwsem
  PHY: remove rwsem use from phy core
  IEEE1394: remove rwsem use from ieee1394 core
  ...
This commit is contained in:
Linus Torvalds 2007-04-27 12:58:54 -07:00
commit d868772fff
56 changed files with 852 additions and 542 deletions

View File

@ -134,15 +134,6 @@ Who: Arjan van de Ven <arjan@linux.intel.com>
--------------------------- ---------------------------
What: mount/umount uevents
When: February 2007
Why: These events are not correct, and do not properly let userspace know
when a file system has been mounted or unmounted. Userspace should
poll the /proc/mounts file instead to detect this properly.
Who: Greg Kroah-Hartman <gregkh@suse.de>
---------------------------
What: USB driver API moves to EXPORT_SYMBOL_GPL What: USB driver API moves to EXPORT_SYMBOL_GPL
When: February 2008 When: February 2008
Files: include/linux/usb.h, drivers/usb/core/driver.c Files: include/linux/usb.h, drivers/usb/core/driver.c

View File

@ -475,9 +475,6 @@ static struct of_platform_driver of_pci_phb_driver = {
.name = "of-pci", .name = "of-pci",
.match_table = of_pci_phb_ids, .match_table = of_pci_phb_ids,
.probe = of_pci_phb_probe, .probe = of_pci_phb_probe,
.driver = {
.multithread_probe = 1,
},
}; };
static __init int of_pci_phb_init(void) static __init int of_pci_phb_init(void)

View File

@ -47,14 +47,13 @@ static int amba_match(struct device *dev, struct device_driver *drv)
static int amba_uevent(struct device *dev, char **envp, int nr_env, char *buf, int bufsz) static int amba_uevent(struct device *dev, char **envp, int nr_env, char *buf, int bufsz)
{ {
struct amba_device *pcdev = to_amba_device(dev); struct amba_device *pcdev = to_amba_device(dev);
int retval = 0, i = 0, len = 0;
if (nr_env < 2) retval = add_uevent_var(envp, nr_env, &i,
return -ENOMEM; buf, bufsz, &len,
"AMBA_ID=%08x", pcdev->periphid);
snprintf(buf, bufsz, "AMBA_ID=%08x", pcdev->periphid); envp[i] = NULL;
*envp++ = buf; return retval;
*envp++ = NULL;
return 0;
} }
#else #else
#define amba_uevent NULL #define amba_uevent NULL

View File

@ -62,7 +62,7 @@ EXPORT_SYMBOL_GPL(attribute_container_classdev_to_container);
static struct list_head attribute_container_list; static struct list_head attribute_container_list;
static DECLARE_MUTEX(attribute_container_mutex); static DEFINE_MUTEX(attribute_container_mutex);
/** /**
* attribute_container_register - register an attribute container * attribute_container_register - register an attribute container
@ -77,9 +77,9 @@ attribute_container_register(struct attribute_container *cont)
klist_init(&cont->containers,internal_container_klist_get, klist_init(&cont->containers,internal_container_klist_get,
internal_container_klist_put); internal_container_klist_put);
down(&attribute_container_mutex); mutex_lock(&attribute_container_mutex);
list_add_tail(&cont->node, &attribute_container_list); list_add_tail(&cont->node, &attribute_container_list);
up(&attribute_container_mutex); mutex_unlock(&attribute_container_mutex);
return 0; return 0;
} }
@ -94,7 +94,7 @@ int
attribute_container_unregister(struct attribute_container *cont) attribute_container_unregister(struct attribute_container *cont)
{ {
int retval = -EBUSY; int retval = -EBUSY;
down(&attribute_container_mutex); mutex_lock(&attribute_container_mutex);
spin_lock(&cont->containers.k_lock); spin_lock(&cont->containers.k_lock);
if (!list_empty(&cont->containers.k_list)) if (!list_empty(&cont->containers.k_list))
goto out; goto out;
@ -102,7 +102,7 @@ attribute_container_unregister(struct attribute_container *cont)
list_del(&cont->node); list_del(&cont->node);
out: out:
spin_unlock(&cont->containers.k_lock); spin_unlock(&cont->containers.k_lock);
up(&attribute_container_mutex); mutex_unlock(&attribute_container_mutex);
return retval; return retval;
} }
@ -145,7 +145,7 @@ attribute_container_add_device(struct device *dev,
{ {
struct attribute_container *cont; struct attribute_container *cont;
down(&attribute_container_mutex); mutex_lock(&attribute_container_mutex);
list_for_each_entry(cont, &attribute_container_list, node) { list_for_each_entry(cont, &attribute_container_list, node) {
struct internal_container *ic; struct internal_container *ic;
@ -173,7 +173,7 @@ attribute_container_add_device(struct device *dev,
attribute_container_add_class_device(&ic->classdev); attribute_container_add_class_device(&ic->classdev);
klist_add_tail(&ic->node, &cont->containers); klist_add_tail(&ic->node, &cont->containers);
} }
up(&attribute_container_mutex); mutex_unlock(&attribute_container_mutex);
} }
/* FIXME: can't break out of this unless klist_iter_exit is also /* FIXME: can't break out of this unless klist_iter_exit is also
@ -211,7 +211,7 @@ attribute_container_remove_device(struct device *dev,
{ {
struct attribute_container *cont; struct attribute_container *cont;
down(&attribute_container_mutex); mutex_lock(&attribute_container_mutex);
list_for_each_entry(cont, &attribute_container_list, node) { list_for_each_entry(cont, &attribute_container_list, node) {
struct internal_container *ic; struct internal_container *ic;
struct klist_iter iter; struct klist_iter iter;
@ -234,7 +234,7 @@ attribute_container_remove_device(struct device *dev,
} }
} }
} }
up(&attribute_container_mutex); mutex_unlock(&attribute_container_mutex);
} }
/** /**
@ -255,7 +255,7 @@ attribute_container_device_trigger(struct device *dev,
{ {
struct attribute_container *cont; struct attribute_container *cont;
down(&attribute_container_mutex); mutex_lock(&attribute_container_mutex);
list_for_each_entry(cont, &attribute_container_list, node) { list_for_each_entry(cont, &attribute_container_list, node) {
struct internal_container *ic; struct internal_container *ic;
struct klist_iter iter; struct klist_iter iter;
@ -273,7 +273,7 @@ attribute_container_device_trigger(struct device *dev,
fn(cont, dev, &ic->classdev); fn(cont, dev, &ic->classdev);
} }
} }
up(&attribute_container_mutex); mutex_unlock(&attribute_container_mutex);
} }
/** /**
@ -295,12 +295,12 @@ attribute_container_trigger(struct device *dev,
{ {
struct attribute_container *cont; struct attribute_container *cont;
down(&attribute_container_mutex); mutex_lock(&attribute_container_mutex);
list_for_each_entry(cont, &attribute_container_list, node) { list_for_each_entry(cont, &attribute_container_list, node) {
if (cont->match(cont, dev)) if (cont->match(cont, dev))
fn(cont, dev); fn(cont, dev);
} }
up(&attribute_container_mutex); mutex_unlock(&attribute_container_mutex);
} }
/** /**

View File

@ -16,7 +16,7 @@ extern int cpu_dev_init(void);
extern int attribute_container_init(void); extern int attribute_container_init(void);
extern int bus_add_device(struct device * dev); extern int bus_add_device(struct device * dev);
extern int bus_attach_device(struct device * dev); extern void bus_attach_device(struct device * dev);
extern void bus_remove_device(struct device * dev); extern void bus_remove_device(struct device * dev);
extern struct bus_type *get_bus(struct bus_type * bus); extern struct bus_type *get_bus(struct bus_type * bus);
extern void put_bus(struct bus_type * bus); extern void put_bus(struct bus_type * bus);

View File

@ -27,6 +27,9 @@
#define to_driver(obj) container_of(obj, struct device_driver, kobj) #define to_driver(obj) container_of(obj, struct device_driver, kobj)
static int __must_check bus_rescan_devices_helper(struct device *dev,
void *data);
static ssize_t static ssize_t
drv_attr_show(struct kobject * kobj, struct attribute * attr, char * buf) drv_attr_show(struct kobject * kobj, struct attribute * attr, char * buf)
{ {
@ -60,8 +63,19 @@ static struct sysfs_ops driver_sysfs_ops = {
static void driver_release(struct kobject * kobj) static void driver_release(struct kobject * kobj)
{ {
struct device_driver * drv = to_driver(kobj); /*
complete(&drv->unloaded); * Yes this is an empty release function, it is this way because struct
* device is always a static object, not a dynamic one. Yes, this is
* not nice and bad, but remember, drivers are code, reference counted
* by the module count, not a device, which is really data. And yes,
* in the future I do want to have all drivers be created dynamically,
* and am working toward that goal, but it will take a bit longer...
*
* But do not let this example give _anyone_ the idea that they can
* create a release function without any code in it at all, to do that
* is almost always wrong. If you have any questions about this,
* please send an email to <greg@kroah.com>
*/
} }
static struct kobj_type ktype_driver = { static struct kobj_type ktype_driver = {
@ -133,7 +147,6 @@ static decl_subsys(bus, &ktype_bus, NULL);
#ifdef CONFIG_HOTPLUG #ifdef CONFIG_HOTPLUG
/* Manually detach a device from its associated driver. */ /* Manually detach a device from its associated driver. */
static int driver_helper(struct device *dev, void *data) static int driver_helper(struct device *dev, void *data)
{ {
@ -199,6 +212,33 @@ static ssize_t driver_bind(struct device_driver *drv,
} }
static DRIVER_ATTR(bind, S_IWUSR, NULL, driver_bind); static DRIVER_ATTR(bind, S_IWUSR, NULL, driver_bind);
static ssize_t show_drivers_autoprobe(struct bus_type *bus, char *buf)
{
return sprintf(buf, "%d\n", bus->drivers_autoprobe);
}
static ssize_t store_drivers_autoprobe(struct bus_type *bus,
const char *buf, size_t count)
{
if (buf[0] == '0')
bus->drivers_autoprobe = 0;
else
bus->drivers_autoprobe = 1;
return count;
}
static ssize_t store_drivers_probe(struct bus_type *bus,
const char *buf, size_t count)
{
struct device *dev;
dev = bus_find_device(bus, NULL, (void *)buf, driver_helper);
if (!dev)
return -ENODEV;
if (bus_rescan_devices_helper(dev, NULL) != 0)
return -EINVAL;
return count;
}
#endif #endif
static struct device * next_device(struct klist_iter * i) static struct device * next_device(struct klist_iter * i)
@ -418,21 +458,21 @@ out_put:
* - Add device to bus's list of devices. * - Add device to bus's list of devices.
* - Try to attach to driver. * - Try to attach to driver.
*/ */
int bus_attach_device(struct device * dev) void bus_attach_device(struct device * dev)
{ {
struct bus_type *bus = dev->bus; struct bus_type *bus = dev->bus;
int ret = 0; int ret = 0;
if (bus) { if (bus) {
dev->is_registered = 1; dev->is_registered = 1;
ret = device_attach(dev); if (bus->drivers_autoprobe)
if (ret >= 0) { ret = device_attach(dev);
WARN_ON(ret < 0);
if (ret >= 0)
klist_add_tail(&dev->knode_bus, &bus->klist_devices); klist_add_tail(&dev->knode_bus, &bus->klist_devices);
ret = 0; else
} else
dev->is_registered = 0; dev->is_registered = 0;
} }
return ret;
} }
/** /**
@ -515,9 +555,41 @@ static void remove_bind_files(struct device_driver *drv)
driver_remove_file(drv, &driver_attr_bind); driver_remove_file(drv, &driver_attr_bind);
driver_remove_file(drv, &driver_attr_unbind); driver_remove_file(drv, &driver_attr_unbind);
} }
static int add_probe_files(struct bus_type *bus)
{
int retval;
bus->drivers_probe_attr.attr.name = "drivers_probe";
bus->drivers_probe_attr.attr.mode = S_IWUSR;
bus->drivers_probe_attr.attr.owner = bus->owner;
bus->drivers_probe_attr.store = store_drivers_probe;
retval = bus_create_file(bus, &bus->drivers_probe_attr);
if (retval)
goto out;
bus->drivers_autoprobe_attr.attr.name = "drivers_autoprobe";
bus->drivers_autoprobe_attr.attr.mode = S_IWUSR | S_IRUGO;
bus->drivers_autoprobe_attr.attr.owner = bus->owner;
bus->drivers_autoprobe_attr.show = show_drivers_autoprobe;
bus->drivers_autoprobe_attr.store = store_drivers_autoprobe;
retval = bus_create_file(bus, &bus->drivers_autoprobe_attr);
if (retval)
bus_remove_file(bus, &bus->drivers_probe_attr);
out:
return retval;
}
static void remove_probe_files(struct bus_type *bus)
{
bus_remove_file(bus, &bus->drivers_autoprobe_attr);
bus_remove_file(bus, &bus->drivers_probe_attr);
}
#else #else
static inline int add_bind_files(struct device_driver *drv) { return 0; } static inline int add_bind_files(struct device_driver *drv) { return 0; }
static inline void remove_bind_files(struct device_driver *drv) {} static inline void remove_bind_files(struct device_driver *drv) {}
static inline int add_probe_files(struct bus_type *bus) { return 0; }
static inline void remove_probe_files(struct bus_type *bus) {}
#endif #endif
/** /**
@ -531,7 +603,7 @@ int bus_add_driver(struct device_driver *drv)
int error = 0; int error = 0;
if (!bus) if (!bus)
return 0; return -EINVAL;
pr_debug("bus %s: add driver %s\n", bus->name, drv->name); pr_debug("bus %s: add driver %s\n", bus->name, drv->name);
error = kobject_set_name(&drv->kobj, "%s", drv->name); error = kobject_set_name(&drv->kobj, "%s", drv->name);
@ -541,9 +613,11 @@ int bus_add_driver(struct device_driver *drv)
if ((error = kobject_register(&drv->kobj))) if ((error = kobject_register(&drv->kobj)))
goto out_put_bus; goto out_put_bus;
error = driver_attach(drv); if (drv->bus->drivers_autoprobe) {
if (error) error = driver_attach(drv);
goto out_unregister; if (error)
goto out_unregister;
}
klist_add_tail(&drv->knode_bus, &bus->klist_drivers); klist_add_tail(&drv->knode_bus, &bus->klist_drivers);
module_add_driver(drv->owner, drv); module_add_driver(drv->owner, drv);
@ -605,8 +679,6 @@ static int __must_check bus_rescan_devices_helper(struct device *dev,
ret = device_attach(dev); ret = device_attach(dev);
if (dev->parent) if (dev->parent)
up(&dev->parent->sem); up(&dev->parent->sem);
if (ret > 0)
ret = 0;
} }
return ret < 0 ? ret : 0; return ret < 0 ? ret : 0;
} }
@ -762,6 +834,12 @@ int bus_register(struct bus_type * bus)
klist_init(&bus->klist_devices, klist_devices_get, klist_devices_put); klist_init(&bus->klist_devices, klist_devices_get, klist_devices_put);
klist_init(&bus->klist_drivers, NULL, NULL); klist_init(&bus->klist_drivers, NULL, NULL);
bus->drivers_autoprobe = 1;
retval = add_probe_files(bus);
if (retval)
goto bus_probe_files_fail;
retval = bus_add_attrs(bus); retval = bus_add_attrs(bus);
if (retval) if (retval)
goto bus_attrs_fail; goto bus_attrs_fail;
@ -770,6 +848,8 @@ int bus_register(struct bus_type * bus)
return 0; return 0;
bus_attrs_fail: bus_attrs_fail:
remove_probe_files(bus);
bus_probe_files_fail:
kset_unregister(&bus->drivers); kset_unregister(&bus->drivers);
bus_drivers_fail: bus_drivers_fail:
kset_unregister(&bus->devices); kset_unregister(&bus->devices);
@ -779,7 +859,6 @@ out:
return retval; return retval;
} }
/** /**
* bus_unregister - remove a bus from the system * bus_unregister - remove a bus from the system
* @bus: bus. * @bus: bus.
@ -791,6 +870,7 @@ void bus_unregister(struct bus_type * bus)
{ {
pr_debug("bus %s: unregistering\n", bus->name); pr_debug("bus %s: unregistering\n", bus->name);
bus_remove_attrs(bus); bus_remove_attrs(bus);
remove_probe_files(bus);
kset_unregister(&bus->drivers); kset_unregister(&bus->drivers);
kset_unregister(&bus->devices); kset_unregister(&bus->devices);
subsystem_unregister(&bus->subsys); subsystem_unregister(&bus->subsys);

View File

@ -145,6 +145,7 @@ int class_register(struct class * cls)
INIT_LIST_HEAD(&cls->children); INIT_LIST_HEAD(&cls->children);
INIT_LIST_HEAD(&cls->devices); INIT_LIST_HEAD(&cls->devices);
INIT_LIST_HEAD(&cls->interfaces); INIT_LIST_HEAD(&cls->interfaces);
kset_init(&cls->class_dirs);
init_MUTEX(&cls->sem); init_MUTEX(&cls->sem);
error = kobject_set_name(&cls->subsys.kset.kobj, "%s", cls->name); error = kobject_set_name(&cls->subsys.kset.kobj, "%s", cls->name);
if (error) if (error)
@ -163,7 +164,6 @@ int class_register(struct class * cls)
void class_unregister(struct class * cls) void class_unregister(struct class * cls)
{ {
pr_debug("device class '%s': unregistering\n", cls->name); pr_debug("device class '%s': unregistering\n", cls->name);
kobject_unregister(cls->virtual_dir);
remove_class_attrs(cls); remove_class_attrs(cls);
subsystem_unregister(&cls->subsys); subsystem_unregister(&cls->subsys);
} }

View File

@ -43,7 +43,8 @@ int (*platform_notify_remove)(struct device * dev) = NULL;
const char *dev_driver_string(struct device *dev) const char *dev_driver_string(struct device *dev)
{ {
return dev->driver ? dev->driver->name : return dev->driver ? dev->driver->name :
(dev->bus ? dev->bus->name : ""); (dev->bus ? dev->bus->name :
(dev->class ? dev->class->name : ""));
} }
EXPORT_SYMBOL(dev_driver_string); EXPORT_SYMBOL(dev_driver_string);
@ -119,6 +120,8 @@ static int dev_uevent_filter(struct kset *kset, struct kobject *kobj)
if (ktype == &ktype_device) { if (ktype == &ktype_device) {
struct device *dev = to_dev(kobj); struct device *dev = to_dev(kobj);
if (dev->uevent_suppress)
return 0;
if (dev->bus) if (dev->bus)
return 1; return 1;
if (dev->class) if (dev->class)
@ -156,6 +159,11 @@ static int dev_uevent(struct kset *kset, struct kobject *kobj, char **envp,
"MINOR=%u", MINOR(dev->devt)); "MINOR=%u", MINOR(dev->devt));
} }
if (dev->type && dev->type->name)
add_uevent_var(envp, num_envp, &i,
buffer, buffer_size, &length,
"DEVTYPE=%s", dev->type->name);
if (dev->driver) if (dev->driver)
add_uevent_var(envp, num_envp, &i, add_uevent_var(envp, num_envp, &i,
buffer, buffer_size, &length, buffer, buffer_size, &length,
@ -238,71 +246,152 @@ static struct kset_uevent_ops device_uevent_ops = {
.uevent = dev_uevent, .uevent = dev_uevent,
}; };
static ssize_t show_uevent(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct kobject *top_kobj;
struct kset *kset;
char *envp[32];
char data[PAGE_SIZE];
char *pos;
int i;
size_t count = 0;
int retval;
/* search the kset, the device belongs to */
top_kobj = &dev->kobj;
if (!top_kobj->kset && top_kobj->parent) {
do {
top_kobj = top_kobj->parent;
} while (!top_kobj->kset && top_kobj->parent);
}
if (!top_kobj->kset)
goto out;
kset = top_kobj->kset;
if (!kset->uevent_ops || !kset->uevent_ops->uevent)
goto out;
/* respect filter */
if (kset->uevent_ops && kset->uevent_ops->filter)
if (!kset->uevent_ops->filter(kset, &dev->kobj))
goto out;
/* let the kset specific function add its keys */
pos = data;
retval = kset->uevent_ops->uevent(kset, &dev->kobj,
envp, ARRAY_SIZE(envp),
pos, PAGE_SIZE);
if (retval)
goto out;
/* copy keys to file */
for (i = 0; envp[i]; i++) {
pos = &buf[count];
count += sprintf(pos, "%s\n", envp[i]);
}
out:
return count;
}
static ssize_t store_uevent(struct device *dev, struct device_attribute *attr, static ssize_t store_uevent(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count) const char *buf, size_t count)
{ {
if (memcmp(buf, "add", 3) != 0)
dev_err(dev, "uevent: unsupported action-string; this will "
"be ignored in a future kernel version");
kobject_uevent(&dev->kobj, KOBJ_ADD); kobject_uevent(&dev->kobj, KOBJ_ADD);
return count; return count;
} }
static int device_add_groups(struct device *dev) static int device_add_attributes(struct device *dev,
struct device_attribute *attrs)
{ {
int i;
int error = 0; int error = 0;
int i;
if (dev->groups) { if (attrs) {
for (i = 0; dev->groups[i]; i++) { for (i = 0; attr_name(attrs[i]); i++) {
error = sysfs_create_group(&dev->kobj, dev->groups[i]); error = device_create_file(dev, &attrs[i]);
if (error) { if (error)
while (--i >= 0) break;
sysfs_remove_group(&dev->kobj, dev->groups[i]);
goto out;
}
} }
if (error)
while (--i >= 0)
device_remove_file(dev, &attrs[i]);
} }
out:
return error; return error;
} }
static void device_remove_groups(struct device *dev) static void device_remove_attributes(struct device *dev,
struct device_attribute *attrs)
{ {
int i; int i;
if (dev->groups) {
for (i = 0; dev->groups[i]; i++) { if (attrs)
sysfs_remove_group(&dev->kobj, dev->groups[i]); for (i = 0; attr_name(attrs[i]); i++)
device_remove_file(dev, &attrs[i]);
}
static int device_add_groups(struct device *dev,
struct attribute_group **groups)
{
int error = 0;
int i;
if (groups) {
for (i = 0; groups[i]; i++) {
error = sysfs_create_group(&dev->kobj, groups[i]);
if (error) {
while (--i >= 0)
sysfs_remove_group(&dev->kobj, groups[i]);
break;
}
} }
} }
return error;
}
static void device_remove_groups(struct device *dev,
struct attribute_group **groups)
{
int i;
if (groups)
for (i = 0; groups[i]; i++)
sysfs_remove_group(&dev->kobj, groups[i]);
} }
static int device_add_attrs(struct device *dev) static int device_add_attrs(struct device *dev)
{ {
struct class *class = dev->class; struct class *class = dev->class;
struct device_type *type = dev->type; struct device_type *type = dev->type;
int error = 0; int error;
int i;
if (class && class->dev_attrs) { if (class) {
for (i = 0; attr_name(class->dev_attrs[i]); i++) { error = device_add_attributes(dev, class->dev_attrs);
error = device_create_file(dev, &class->dev_attrs[i]);
if (error)
break;
}
if (error) if (error)
while (--i >= 0) return error;
device_remove_file(dev, &class->dev_attrs[i]);
} }
if (type && type->attrs) { if (type) {
for (i = 0; attr_name(type->attrs[i]); i++) { error = device_add_groups(dev, type->groups);
error = device_create_file(dev, &type->attrs[i]);
if (error)
break;
}
if (error) if (error)
while (--i >= 0) goto err_remove_class_attrs;
device_remove_file(dev, &type->attrs[i]);
} }
error = device_add_groups(dev, dev->groups);
if (error)
goto err_remove_type_groups;
return 0;
err_remove_type_groups:
if (type)
device_remove_groups(dev, type->groups);
err_remove_class_attrs:
if (class)
device_remove_attributes(dev, class->dev_attrs);
return error; return error;
} }
@ -310,17 +399,14 @@ static void device_remove_attrs(struct device *dev)
{ {
struct class *class = dev->class; struct class *class = dev->class;
struct device_type *type = dev->type; struct device_type *type = dev->type;
int i;
if (class && class->dev_attrs) { device_remove_groups(dev, dev->groups);
for (i = 0; attr_name(class->dev_attrs[i]); i++)
device_remove_file(dev, &class->dev_attrs[i]);
}
if (type && type->attrs) { if (type)
for (i = 0; attr_name(type->attrs[i]); i++) device_remove_groups(dev, type->groups);
device_remove_file(dev, &type->attrs[i]);
} if (class)
device_remove_attributes(dev, class->dev_attrs);
} }
@ -394,9 +480,10 @@ void device_remove_bin_file(struct device *dev, struct bin_attribute *attr)
EXPORT_SYMBOL_GPL(device_remove_bin_file); EXPORT_SYMBOL_GPL(device_remove_bin_file);
/** /**
* device_schedule_callback - helper to schedule a callback for a device * device_schedule_callback_owner - helper to schedule a callback for a device
* @dev: device. * @dev: device.
* @func: callback function to invoke later. * @func: callback function to invoke later.
* @owner: module owning the callback routine
* *
* Attribute methods must not unregister themselves or their parent device * Attribute methods must not unregister themselves or their parent device
* (which would amount to the same thing). Attempts to do so will deadlock, * (which would amount to the same thing). Attempts to do so will deadlock,
@ -407,20 +494,23 @@ EXPORT_SYMBOL_GPL(device_remove_bin_file);
* argument in the workqueue's process context. @dev will be pinned until * argument in the workqueue's process context. @dev will be pinned until
* @func returns. * @func returns.
* *
* This routine is usually called via the inline device_schedule_callback(),
* which automatically sets @owner to THIS_MODULE.
*
* Returns 0 if the request was submitted, -ENOMEM if storage could not * Returns 0 if the request was submitted, -ENOMEM if storage could not
* be allocated. * be allocated, -ENODEV if a reference to @owner isn't available.
* *
* NOTE: This routine won't work if CONFIG_SYSFS isn't set! It uses an * NOTE: This routine won't work if CONFIG_SYSFS isn't set! It uses an
* underlying sysfs routine (since it is intended for use by attribute * underlying sysfs routine (since it is intended for use by attribute
* methods), and if sysfs isn't available you'll get nothing but -ENOSYS. * methods), and if sysfs isn't available you'll get nothing but -ENOSYS.
*/ */
int device_schedule_callback(struct device *dev, int device_schedule_callback_owner(struct device *dev,
void (*func)(struct device *)) void (*func)(struct device *), struct module *owner)
{ {
return sysfs_schedule_callback(&dev->kobj, return sysfs_schedule_callback(&dev->kobj,
(void (*)(void *)) func, dev); (void (*)(void *)) func, dev, owner);
} }
EXPORT_SYMBOL_GPL(device_schedule_callback); EXPORT_SYMBOL_GPL(device_schedule_callback_owner);
static void klist_children_get(struct klist_node *n) static void klist_children_get(struct klist_node *n)
{ {
@ -477,34 +567,58 @@ static struct kobject * get_device_parent(struct device *dev,
return NULL; return NULL;
} }
#else #else
static struct kobject * virtual_device_parent(struct device *dev) static struct kobject *virtual_device_parent(struct device *dev)
{ {
if (!dev->class) static struct kobject *virtual_dir = NULL;
return ERR_PTR(-ENODEV);
if (!dev->class->virtual_dir) { if (!virtual_dir)
static struct kobject *virtual_dir = NULL; virtual_dir = kobject_add_dir(&devices_subsys.kset.kobj, "virtual");
if (!virtual_dir) return virtual_dir;
virtual_dir = kobject_add_dir(&devices_subsys.kset.kobj, "virtual");
dev->class->virtual_dir = kobject_add_dir(virtual_dir, dev->class->name);
}
return dev->class->virtual_dir;
} }
static struct kobject * get_device_parent(struct device *dev, static struct kobject * get_device_parent(struct device *dev,
struct device *parent) struct device *parent)
{ {
/* if this is a class device, and has no parent, create one */ if (dev->class) {
if ((dev->class) && (parent == NULL)) { struct kobject *kobj = NULL;
return virtual_device_parent(dev); struct kobject *parent_kobj;
} else if (parent) struct kobject *k;
/*
* If we have no parent, we live in "virtual".
* Class-devices with a bus-device as parent, live
* in a class-directory to prevent namespace collisions.
*/
if (parent == NULL)
parent_kobj = virtual_device_parent(dev);
else if (parent->class)
return &parent->kobj;
else
parent_kobj = &parent->kobj;
/* find our class-directory at the parent and reference it */
spin_lock(&dev->class->class_dirs.list_lock);
list_for_each_entry(k, &dev->class->class_dirs.list, entry)
if (k->parent == parent_kobj) {
kobj = kobject_get(k);
break;
}
spin_unlock(&dev->class->class_dirs.list_lock);
if (kobj)
return kobj;
/* or create a new class-directory at the parent device */
return kobject_kset_add_dir(&dev->class->class_dirs,
parent_kobj, dev->class->name);
}
if (parent)
return &parent->kobj; return &parent->kobj;
return NULL; return NULL;
} }
#endif #endif
static int setup_parent(struct device *dev, struct device *parent) static int setup_parent(struct device *dev, struct device *parent)
{ {
struct kobject *kobj; struct kobject *kobj;
@ -541,7 +655,6 @@ int device_add(struct device *dev)
pr_debug("DEV: registering device: ID = '%s'\n", dev->bus_id); pr_debug("DEV: registering device: ID = '%s'\n", dev->bus_id);
parent = get_device(dev->parent); parent = get_device(dev->parent);
error = setup_parent(dev, parent); error = setup_parent(dev, parent);
if (error) if (error)
goto Error; goto Error;
@ -562,10 +675,11 @@ int device_add(struct device *dev)
BUS_NOTIFY_ADD_DEVICE, dev); BUS_NOTIFY_ADD_DEVICE, dev);
dev->uevent_attr.attr.name = "uevent"; dev->uevent_attr.attr.name = "uevent";
dev->uevent_attr.attr.mode = S_IWUSR; dev->uevent_attr.attr.mode = S_IRUGO | S_IWUSR;
if (dev->driver) if (dev->driver)
dev->uevent_attr.attr.owner = dev->driver->owner; dev->uevent_attr.attr.owner = dev->driver->owner;
dev->uevent_attr.store = store_uevent; dev->uevent_attr.store = store_uevent;
dev->uevent_attr.show = show_uevent;
error = device_create_file(dev, &dev->uevent_attr); error = device_create_file(dev, &dev->uevent_attr);
if (error) if (error)
goto attrError; goto attrError;
@ -614,16 +728,12 @@ int device_add(struct device *dev)
if ((error = device_add_attrs(dev))) if ((error = device_add_attrs(dev)))
goto AttrsError; goto AttrsError;
if ((error = device_add_groups(dev)))
goto GroupError;
if ((error = device_pm_add(dev))) if ((error = device_pm_add(dev)))
goto PMError; goto PMError;
if ((error = bus_add_device(dev))) if ((error = bus_add_device(dev)))
goto BusError; goto BusError;
if (!dev->uevent_suppress) kobject_uevent(&dev->kobj, KOBJ_ADD);
kobject_uevent(&dev->kobj, KOBJ_ADD); bus_attach_device(dev);
if ((error = bus_attach_device(dev)))
goto AttachError;
if (parent) if (parent)
klist_add_tail(&dev->knode_parent, &parent->klist_children); klist_add_tail(&dev->knode_parent, &parent->klist_children);
@ -639,19 +749,15 @@ int device_add(struct device *dev)
up(&dev->class->sem); up(&dev->class->sem);
} }
Done: Done:
kfree(class_name); kfree(class_name);
put_device(dev); put_device(dev);
return error; return error;
AttachError:
bus_remove_device(dev);
BusError: BusError:
device_pm_remove(dev); device_pm_remove(dev);
PMError: PMError:
if (dev->bus) if (dev->bus)
blocking_notifier_call_chain(&dev->bus->bus_notifier, blocking_notifier_call_chain(&dev->bus->bus_notifier,
BUS_NOTIFY_DEL_DEVICE, dev); BUS_NOTIFY_DEL_DEVICE, dev);
device_remove_groups(dev);
GroupError:
device_remove_attrs(dev); device_remove_attrs(dev);
AttrsError: AttrsError:
if (dev->devt_attr) { if (dev->devt_attr) {
@ -677,15 +783,6 @@ int device_add(struct device *dev)
#endif #endif
sysfs_remove_link(&dev->kobj, "device"); sysfs_remove_link(&dev->kobj, "device");
} }
down(&dev->class->sem);
/* notify any interfaces that the device is now gone */
list_for_each_entry(class_intf, &dev->class->interfaces, node)
if (class_intf->remove_dev)
class_intf->remove_dev(dev, class_intf);
/* remove the device from the class list */
list_del_init(&dev->node);
up(&dev->class->sem);
} }
ueventattrError: ueventattrError:
device_remove_file(dev, &dev->uevent_attr); device_remove_file(dev, &dev->uevent_attr);
@ -796,9 +893,33 @@ void device_del(struct device * dev)
/* remove the device from the class list */ /* remove the device from the class list */
list_del_init(&dev->node); list_del_init(&dev->node);
up(&dev->class->sem); up(&dev->class->sem);
/* If we live in a parent class-directory, unreference it */
if (dev->kobj.parent->kset == &dev->class->class_dirs) {
struct device *d;
int other = 0;
/*
* if we are the last child of our class, delete
* our class-directory at this parent
*/
down(&dev->class->sem);
list_for_each_entry(d, &dev->class->devices, node) {
if (d == dev)
continue;
if (d->kobj.parent == dev->kobj.parent) {
other = 1;
break;
}
}
if (!other)
kobject_del(dev->kobj.parent);
kobject_put(dev->kobj.parent);
up(&dev->class->sem);
}
} }
device_remove_file(dev, &dev->uevent_attr); device_remove_file(dev, &dev->uevent_attr);
device_remove_groups(dev);
device_remove_attrs(dev); device_remove_attrs(dev);
bus_remove_device(dev); bus_remove_device(dev);

View File

@ -94,19 +94,11 @@ int device_bind_driver(struct device *dev)
return ret; return ret;
} }
struct stupid_thread_structure {
struct device_driver *drv;
struct device *dev;
};
static atomic_t probe_count = ATOMIC_INIT(0); static atomic_t probe_count = ATOMIC_INIT(0);
static DECLARE_WAIT_QUEUE_HEAD(probe_waitqueue); static DECLARE_WAIT_QUEUE_HEAD(probe_waitqueue);
static int really_probe(void *void_data) static int really_probe(struct device *dev, struct device_driver *drv)
{ {
struct stupid_thread_structure *data = void_data;
struct device_driver *drv = data->drv;
struct device *dev = data->dev;
int ret = 0; int ret = 0;
atomic_inc(&probe_count); atomic_inc(&probe_count);
@ -154,7 +146,6 @@ probe_failed:
*/ */
ret = 0; ret = 0;
done: done:
kfree(data);
atomic_dec(&probe_count); atomic_dec(&probe_count);
wake_up(&probe_waitqueue); wake_up(&probe_waitqueue);
return ret; return ret;
@ -186,16 +177,14 @@ int driver_probe_done(void)
* format of the ID structures, nor what is to be considered a match and * format of the ID structures, nor what is to be considered a match and
* what is not. * what is not.
* *
* This function returns 1 if a match is found, an error if one occurs * This function returns 1 if a match is found, -ENODEV if the device is
* (that is not -ENODEV or -ENXIO), and 0 otherwise. * not registered, and 0 otherwise.
* *
* This function must be called with @dev->sem held. When called for a * This function must be called with @dev->sem held. When called for a
* USB interface, @dev->parent->sem must be held as well. * USB interface, @dev->parent->sem must be held as well.
*/ */
int driver_probe_device(struct device_driver * drv, struct device * dev) int driver_probe_device(struct device_driver * drv, struct device * dev)
{ {
struct stupid_thread_structure *data;
struct task_struct *probe_task;
int ret = 0; int ret = 0;
if (!device_is_registered(dev)) if (!device_is_registered(dev))
@ -206,19 +195,7 @@ int driver_probe_device(struct device_driver * drv, struct device * dev)
pr_debug("%s: Matched Device %s with Driver %s\n", pr_debug("%s: Matched Device %s with Driver %s\n",
drv->bus->name, dev->bus_id, drv->name); drv->bus->name, dev->bus_id, drv->name);
data = kmalloc(sizeof(*data), GFP_KERNEL); ret = really_probe(dev, drv);
if (!data)
return -ENOMEM;
data->drv = drv;
data->dev = dev;
if (drv->multithread_probe) {
probe_task = kthread_run(really_probe, data,
"probe-%s", dev->bus_id);
if (IS_ERR(probe_task))
ret = really_probe(data);
} else
ret = really_probe(data);
done: done:
return ret; return ret;
@ -230,30 +207,57 @@ static int __device_attach(struct device_driver * drv, void * data)
return driver_probe_device(drv, dev); return driver_probe_device(drv, dev);
} }
static int device_probe_drivers(void *data)
{
struct device *dev = data;
int ret = 0;
if (dev->bus) {
down(&dev->sem);
ret = bus_for_each_drv(dev->bus, NULL, dev, __device_attach);
up(&dev->sem);
}
return ret;
}
/** /**
* device_attach - try to attach device to a driver. * device_attach - try to attach device to a driver.
* @dev: device. * @dev: device.
* *
* Walk the list of drivers that the bus has and call * Walk the list of drivers that the bus has and call
* driver_probe_device() for each pair. If a compatible * driver_probe_device() for each pair. If a compatible
* pair is found, break out and return. * pair is found, break out and return. If the bus specifies
* multithreaded probing, walking the list of drivers is done
* on a probing thread.
* *
* Returns 1 if the device was bound to a driver; * Returns 1 if the device was bound to a driver;
* 0 if no matching device was found; error code otherwise. * 0 if no matching device was found or multithreaded probing is done;
* -ENODEV if the device is not registered.
* *
* When called for a USB interface, @dev->parent->sem must be held. * When called for a USB interface, @dev->parent->sem must be held.
*/ */
int device_attach(struct device * dev) int device_attach(struct device * dev)
{ {
int ret = 0; int ret = 0;
struct task_struct *probe_task = ERR_PTR(-ENOMEM);
down(&dev->sem); down(&dev->sem);
if (dev->driver) { if (dev->driver) {
ret = device_bind_driver(dev); ret = device_bind_driver(dev);
if (ret == 0) if (ret == 0)
ret = 1; ret = 1;
} else else {
ret = bus_for_each_drv(dev->bus, NULL, dev, __device_attach); dev->driver = NULL;
ret = 0;
}
} else {
if (dev->bus->multithread_probe)
probe_task = kthread_run(device_probe_drivers, dev,
"probe-%s", dev->bus_id);
if(IS_ERR(probe_task))
ret = bus_for_each_drv(dev->bus, NULL, dev,
__device_attach);
}
up(&dev->sem); up(&dev->sem);
return ret; return ret;
} }

View File

@ -37,7 +37,7 @@ struct dma_page { /* cacheable header for 'allocation' bytes */
#define POOL_TIMEOUT_JIFFIES ((100 /* msec */ * HZ) / 1000) #define POOL_TIMEOUT_JIFFIES ((100 /* msec */ * HZ) / 1000)
static DECLARE_MUTEX (pools_lock); static DEFINE_MUTEX (pools_lock);
static ssize_t static ssize_t
show_pools (struct device *dev, struct device_attribute *attr, char *buf) show_pools (struct device *dev, struct device_attribute *attr, char *buf)
@ -55,7 +55,7 @@ show_pools (struct device *dev, struct device_attribute *attr, char *buf)
size -= temp; size -= temp;
next += temp; next += temp;
down (&pools_lock); mutex_lock(&pools_lock);
list_for_each_entry(pool, &dev->dma_pools, pools) { list_for_each_entry(pool, &dev->dma_pools, pools) {
unsigned pages = 0; unsigned pages = 0;
unsigned blocks = 0; unsigned blocks = 0;
@ -73,7 +73,7 @@ show_pools (struct device *dev, struct device_attribute *attr, char *buf)
size -= temp; size -= temp;
next += temp; next += temp;
} }
up (&pools_lock); mutex_unlock(&pools_lock);
return PAGE_SIZE - size; return PAGE_SIZE - size;
} }
@ -143,7 +143,7 @@ dma_pool_create (const char *name, struct device *dev,
if (dev) { if (dev) {
int ret; int ret;
down (&pools_lock); mutex_lock(&pools_lock);
if (list_empty (&dev->dma_pools)) if (list_empty (&dev->dma_pools))
ret = device_create_file (dev, &dev_attr_pools); ret = device_create_file (dev, &dev_attr_pools);
else else
@ -155,7 +155,7 @@ dma_pool_create (const char *name, struct device *dev,
kfree(retval); kfree(retval);
retval = NULL; retval = NULL;
} }
up (&pools_lock); mutex_unlock(&pools_lock);
} else } else
INIT_LIST_HEAD (&retval->pools); INIT_LIST_HEAD (&retval->pools);
@ -231,11 +231,11 @@ pool_free_page (struct dma_pool *pool, struct dma_page *page)
void void
dma_pool_destroy (struct dma_pool *pool) dma_pool_destroy (struct dma_pool *pool)
{ {
down (&pools_lock); mutex_lock(&pools_lock);
list_del (&pool->pools); list_del (&pool->pools);
if (pool->dev && list_empty (&pool->dev->dma_pools)) if (pool->dev && list_empty (&pool->dev->dma_pools))
device_remove_file (pool->dev, &dev_attr_pools); device_remove_file (pool->dev, &dev_attr_pools);
up (&pools_lock); mutex_unlock(&pools_lock);
while (!list_empty (&pool->page_list)) { while (!list_empty (&pool->page_list)) {
struct dma_page *page; struct dma_page *page;

View File

@ -149,10 +149,6 @@ void put_driver(struct device_driver * drv)
* We pass off most of the work to the bus_add_driver() call, * We pass off most of the work to the bus_add_driver() call,
* since most of the things we have to do deal with the bus * since most of the things we have to do deal with the bus
* structures. * structures.
*
* The one interesting aspect is that we setup @drv->unloaded
* as a completion that gets complete when the driver reference
* count reaches 0.
*/ */
int driver_register(struct device_driver * drv) int driver_register(struct device_driver * drv)
{ {
@ -162,35 +158,19 @@ int driver_register(struct device_driver * drv)
printk(KERN_WARNING "Driver '%s' needs updating - please use bus_type methods\n", drv->name); printk(KERN_WARNING "Driver '%s' needs updating - please use bus_type methods\n", drv->name);
} }
klist_init(&drv->klist_devices, NULL, NULL); klist_init(&drv->klist_devices, NULL, NULL);
init_completion(&drv->unloaded);
return bus_add_driver(drv); return bus_add_driver(drv);
} }
/** /**
* driver_unregister - remove driver from system. * driver_unregister - remove driver from system.
* @drv: driver. * @drv: driver.
* *
* Again, we pass off most of the work to the bus-level call. * Again, we pass off most of the work to the bus-level call.
*
* Though, once that is done, we wait until @drv->unloaded is completed.
* This will block until the driver refcount reaches 0, and it is
* released. Only modular drivers will call this function, and we
* have to guarantee that it won't complete, letting the driver
* unload until all references are gone.
*/ */
void driver_unregister(struct device_driver * drv) void driver_unregister(struct device_driver * drv)
{ {
bus_remove_driver(drv); bus_remove_driver(drv);
/*
* If the driver is a module, we are probably in
* the module unload path, and we want to wait
* for everything to unload before we can actually
* finish the unload.
*/
if (drv->owner)
wait_for_completion(&drv->unloaded);
} }
/** /**

View File

@ -31,8 +31,6 @@ enum {
FW_STATUS_LOADING, FW_STATUS_LOADING,
FW_STATUS_DONE, FW_STATUS_DONE,
FW_STATUS_ABORT, FW_STATUS_ABORT,
FW_STATUS_READY,
FW_STATUS_READY_NOHOTPLUG,
}; };
static int loading_timeout = 60; /* In seconds */ static int loading_timeout = 60; /* In seconds */
@ -96,9 +94,6 @@ static int firmware_uevent(struct device *dev, char **envp, int num_envp,
struct firmware_priv *fw_priv = dev_get_drvdata(dev); struct firmware_priv *fw_priv = dev_get_drvdata(dev);
int i = 0, len = 0; int i = 0, len = 0;
if (!test_bit(FW_STATUS_READY, &fw_priv->status))
return -ENODEV;
if (add_uevent_var(envp, num_envp, &i, buffer, buffer_size, &len, if (add_uevent_var(envp, num_envp, &i, buffer, buffer_size, &len,
"FIRMWARE=%s", fw_priv->fw_id)) "FIRMWARE=%s", fw_priv->fw_id))
return -ENOMEM; return -ENOMEM;
@ -333,6 +328,7 @@ static int fw_register_device(struct device **dev_p, const char *fw_name,
f_dev->parent = device; f_dev->parent = device;
f_dev->class = &firmware_class; f_dev->class = &firmware_class;
dev_set_drvdata(f_dev, fw_priv); dev_set_drvdata(f_dev, fw_priv);
f_dev->uevent_suppress = 1;
retval = device_register(f_dev); retval = device_register(f_dev);
if (retval) { if (retval) {
printk(KERN_ERR "%s: device_register failed\n", printk(KERN_ERR "%s: device_register failed\n",
@ -382,9 +378,7 @@ static int fw_setup_device(struct firmware *fw, struct device **dev_p,
} }
if (uevent) if (uevent)
set_bit(FW_STATUS_READY, &fw_priv->status); f_dev->uevent_suppress = 0;
else
set_bit(FW_STATUS_READY_NOHOTPLUG, &fw_priv->status);
*dev_p = f_dev; *dev_p = f_dev;
goto out; goto out;

View File

@ -29,6 +29,9 @@ LIST_HEAD(dpm_off_irq);
DECLARE_MUTEX(dpm_sem); DECLARE_MUTEX(dpm_sem);
DECLARE_MUTEX(dpm_list_sem); DECLARE_MUTEX(dpm_list_sem);
int (*platform_enable_wakeup)(struct device *dev, int is_on);
/** /**
* device_pm_set_parent - Specify power dependency. * device_pm_set_parent - Specify power dependency.
* @dev: Device who needs power. * @dev: Device who needs power.

View File

@ -26,7 +26,9 @@ int resume_device(struct device * dev)
TRACE_DEVICE(dev); TRACE_DEVICE(dev);
TRACE_RESUME(0); TRACE_RESUME(0);
down(&dev->sem); down(&dev->sem);
if (dev->power.pm_parent if (dev->power.pm_parent
&& dev->power.pm_parent->power.power_state.event) { && dev->power.pm_parent->power.power_state.event) {
dev_err(dev, "PM: resume from %d, parent %s still %d\n", dev_err(dev, "PM: resume from %d, parent %s still %d\n",
@ -34,15 +36,24 @@ int resume_device(struct device * dev)
dev->power.pm_parent->bus_id, dev->power.pm_parent->bus_id,
dev->power.pm_parent->power.power_state.event); dev->power.pm_parent->power.power_state.event);
} }
if (dev->bus && dev->bus->resume) { if (dev->bus && dev->bus->resume) {
dev_dbg(dev,"resuming\n"); dev_dbg(dev,"resuming\n");
error = dev->bus->resume(dev); error = dev->bus->resume(dev);
} }
if (dev->class && dev->class->resume) {
if (!error && dev->type && dev->type->resume) {
dev_dbg(dev,"resuming\n");
error = dev->type->resume(dev);
}
if (!error && dev->class && dev->class->resume) {
dev_dbg(dev,"class resume\n"); dev_dbg(dev,"class resume\n");
error = dev->class->resume(dev); error = dev->class->resume(dev);
} }
up(&dev->sem); up(&dev->sem);
TRACE_RESUME(error); TRACE_RESUME(error);
return error; return error;
} }

View File

@ -36,7 +36,6 @@ void device_shutdown(void)
{ {
struct device * dev, *devn; struct device * dev, *devn;
down_write(&devices_subsys.rwsem);
list_for_each_entry_safe_reverse(dev, devn, &devices_subsys.kset.list, list_for_each_entry_safe_reverse(dev, devn, &devices_subsys.kset.list,
kobj.entry) { kobj.entry) {
if (dev->bus && dev->bus->shutdown) { if (dev->bus && dev->bus->shutdown) {
@ -47,7 +46,6 @@ void device_shutdown(void)
dev->driver->shutdown(dev); dev->driver->shutdown(dev);
} }
} }
up_write(&devices_subsys.rwsem);
sysdev_shutdown(); sysdev_shutdown();
} }

View File

@ -78,6 +78,18 @@ int suspend_device(struct device * dev, pm_message_t state)
suspend_report_result(dev->class->suspend, error); suspend_report_result(dev->class->suspend, error);
} }
if (!error && dev->type && dev->type->suspend && !dev->power.power_state.event) {
dev_dbg(dev, "%s%s\n",
suspend_verb(state.event),
((state.event == PM_EVENT_SUSPEND)
&& device_may_wakeup(dev))
? ", may wakeup"
: ""
);
error = dev->type->suspend(dev, state);
suspend_report_result(dev->type->suspend, error);
}
if (!error && dev->bus && dev->bus->suspend && !dev->power.power_state.event) { if (!error && dev->bus && dev->bus->suspend && !dev->power.power_state.event) {
dev_dbg(dev, "%s%s\n", dev_dbg(dev, "%s%s\n",
suspend_verb(state.event), suspend_verb(state.event),

View File

@ -310,14 +310,12 @@ static int proc_ide_read_driver
ide_driver_t *ide_drv; ide_driver_t *ide_drv;
int len; int len;
down_read(&dev->bus->subsys.rwsem);
if (dev->driver) { if (dev->driver) {
ide_drv = container_of(dev->driver, ide_driver_t, gen_driver); ide_drv = container_of(dev->driver, ide_driver_t, gen_driver);
len = sprintf(page, "%s version %s\n", len = sprintf(page, "%s version %s\n",
dev->driver->name, ide_drv->version); dev->driver->name, ide_drv->version);
} else } else
len = sprintf(page, "ide-default version 0.9.newide\n"); len = sprintf(page, "ide-default version 0.9.newide\n");
up_read(&dev->bus->subsys.rwsem);
PROC_IDE_READ_RETURN(page,start,off,count,eof,len); PROC_IDE_READ_RETURN(page,start,off,count,eof,len);
} }
@ -327,7 +325,6 @@ static int ide_replace_subdriver(ide_drive_t *drive, const char *driver)
int ret = 1; int ret = 1;
int err; int err;
down_write(&dev->bus->subsys.rwsem);
device_release_driver(dev); device_release_driver(dev);
/* FIXME: device can still be in use by previous driver */ /* FIXME: device can still be in use by previous driver */
strlcpy(drive->driver_req, driver, sizeof(drive->driver_req)); strlcpy(drive->driver_req, driver, sizeof(drive->driver_req));
@ -345,7 +342,6 @@ static int ide_replace_subdriver(ide_drive_t *drive, const char *driver)
} }
if (dev->driver && !strcmp(dev->driver->name, driver)) if (dev->driver && !strcmp(dev->driver->name, driver))
ret = 0; ret = 0;
up_write(&dev->bus->subsys.rwsem);
return ret; return ret;
} }

View File

@ -370,9 +370,7 @@ static ssize_t fw_set_ignore_driver(struct device *dev, struct device_attribute
if (state == 1) { if (state == 1) {
ud->ignore_driver = 1; ud->ignore_driver = 1;
down_write(&ieee1394_bus_type.subsys.rwsem);
device_release_driver(dev); device_release_driver(dev);
up_write(&ieee1394_bus_type.subsys.rwsem);
} else if (state == 0) } else if (state == 0)
ud->ignore_driver = 0; ud->ignore_driver = 0;
@ -1163,6 +1161,7 @@ static int nodemgr_uevent(struct class_device *cdev, char **envp, int num_envp,
struct unit_directory *ud; struct unit_directory *ud;
int i = 0; int i = 0;
int length = 0; int length = 0;
int retval = 0;
/* ieee1394:venNmoNspNverN */ /* ieee1394:venNmoNspNverN */
char buf[8 + 1 + 3 + 8 + 2 + 8 + 2 + 8 + 3 + 8 + 1]; char buf[8 + 1 + 3 + 8 + 2 + 8 + 2 + 8 + 3 + 8 + 1];
@ -1176,14 +1175,11 @@ static int nodemgr_uevent(struct class_device *cdev, char **envp, int num_envp,
#define PUT_ENVP(fmt,val) \ #define PUT_ENVP(fmt,val) \
do { \ do { \
int printed; \ retval = add_uevent_var(envp, num_envp, &i, \
envp[i++] = buffer; \ buffer, buffer_size, &length, \
printed = snprintf(buffer, buffer_size - length, \ fmt, val); \
fmt, val); \ if (retval) \
if ((buffer_size - (length+printed) <= 0) || (i >= num_envp)) \ return retval; \
return -ENOMEM; \
length += printed+1; \
buffer += printed+1; \
} while (0) } while (0)
PUT_ENVP("VENDOR_ID=%06x", ud->vendor_id); PUT_ENVP("VENDOR_ID=%06x", ud->vendor_id);
@ -1393,12 +1389,10 @@ static void nodemgr_suspend_ne(struct node_entry *ne)
if (ud->ne != ne) if (ud->ne != ne)
continue; continue;
down_write(&ieee1394_bus_type.subsys.rwsem);
if (ud->device.driver && if (ud->device.driver &&
(!ud->device.driver->suspend || (!ud->device.driver->suspend ||
ud->device.driver->suspend(&ud->device, PMSG_SUSPEND))) ud->device.driver->suspend(&ud->device, PMSG_SUSPEND)))
device_release_driver(&ud->device); device_release_driver(&ud->device);
up_write(&ieee1394_bus_type.subsys.rwsem);
} }
up(&nodemgr_ud_class.sem); up(&nodemgr_ud_class.sem);
} }
@ -1418,10 +1412,8 @@ static void nodemgr_resume_ne(struct node_entry *ne)
if (ud->ne != ne) if (ud->ne != ne)
continue; continue;
down_read(&ieee1394_bus_type.subsys.rwsem);
if (ud->device.driver && ud->device.driver->resume) if (ud->device.driver && ud->device.driver->resume)
ud->device.driver->resume(&ud->device); ud->device.driver->resume(&ud->device);
up_read(&ieee1394_bus_type.subsys.rwsem);
} }
up(&nodemgr_ud_class.sem); up(&nodemgr_ud_class.sem);
@ -1442,7 +1434,6 @@ static void nodemgr_update_pdrv(struct node_entry *ne)
if (ud->ne != ne) if (ud->ne != ne)
continue; continue;
down_write(&ieee1394_bus_type.subsys.rwsem);
if (ud->device.driver) { if (ud->device.driver) {
pdrv = container_of(ud->device.driver, pdrv = container_of(ud->device.driver,
struct hpsb_protocol_driver, struct hpsb_protocol_driver,
@ -1450,7 +1441,6 @@ static void nodemgr_update_pdrv(struct node_entry *ne)
if (pdrv->update && pdrv->update(ud)) if (pdrv->update && pdrv->update(ud))
device_release_driver(&ud->device); device_release_driver(&ud->device);
} }
up_write(&ieee1394_bus_type.subsys.rwsem);
} }
up(&nodemgr_ud_class.sem); up(&nodemgr_ud_class.sem);
} }

View File

@ -190,16 +190,14 @@ static void gameport_run_poll_handler(unsigned long d)
* Basic gameport -> driver core mappings * Basic gameport -> driver core mappings
*/ */
static void gameport_bind_driver(struct gameport *gameport, struct gameport_driver *drv) static int gameport_bind_driver(struct gameport *gameport, struct gameport_driver *drv)
{ {
int error; int error;
down_write(&gameport_bus.subsys.rwsem);
gameport->dev.driver = &drv->driver; gameport->dev.driver = &drv->driver;
if (drv->connect(gameport, drv)) { if (drv->connect(gameport, drv)) {
gameport->dev.driver = NULL; gameport->dev.driver = NULL;
goto out; return -ENODEV;
} }
error = device_bind_driver(&gameport->dev); error = device_bind_driver(&gameport->dev);
@ -211,31 +209,21 @@ static void gameport_bind_driver(struct gameport *gameport, struct gameport_driv
drv->description, error); drv->description, error);
drv->disconnect(gameport); drv->disconnect(gameport);
gameport->dev.driver = NULL; gameport->dev.driver = NULL;
goto out; return error;
} }
out: return 0;
up_write(&gameport_bus.subsys.rwsem);
}
static void gameport_release_driver(struct gameport *gameport)
{
down_write(&gameport_bus.subsys.rwsem);
device_release_driver(&gameport->dev);
up_write(&gameport_bus.subsys.rwsem);
} }
static void gameport_find_driver(struct gameport *gameport) static void gameport_find_driver(struct gameport *gameport)
{ {
int error; int error;
down_write(&gameport_bus.subsys.rwsem);
error = device_attach(&gameport->dev); error = device_attach(&gameport->dev);
if (error < 0) if (error < 0)
printk(KERN_WARNING printk(KERN_WARNING
"gameport: device_attach() failed for %s (%s), error: %d\n", "gameport: device_attach() failed for %s (%s), error: %d\n",
gameport->phys, gameport->name, error); gameport->phys, gameport->name, error);
up_write(&gameport_bus.subsys.rwsem);
} }
@ -483,13 +471,12 @@ static ssize_t gameport_rebind_driver(struct device *dev, struct device_attribut
{ {
struct gameport *gameport = to_gameport_port(dev); struct gameport *gameport = to_gameport_port(dev);
struct device_driver *drv; struct device_driver *drv;
int retval; int error;
retval = mutex_lock_interruptible(&gameport_mutex); error = mutex_lock_interruptible(&gameport_mutex);
if (retval) if (error)
return retval; return error;
retval = count;
if (!strncmp(buf, "none", count)) { if (!strncmp(buf, "none", count)) {
gameport_disconnect_port(gameport); gameport_disconnect_port(gameport);
} else if (!strncmp(buf, "reconnect", count)) { } else if (!strncmp(buf, "reconnect", count)) {
@ -499,15 +486,15 @@ static ssize_t gameport_rebind_driver(struct device *dev, struct device_attribut
gameport_find_driver(gameport); gameport_find_driver(gameport);
} else if ((drv = driver_find(buf, &gameport_bus)) != NULL) { } else if ((drv = driver_find(buf, &gameport_bus)) != NULL) {
gameport_disconnect_port(gameport); gameport_disconnect_port(gameport);
gameport_bind_driver(gameport, to_gameport_driver(drv)); error = gameport_bind_driver(gameport, to_gameport_driver(drv));
put_driver(drv); put_driver(drv);
} else { } else {
retval = -EINVAL; error = -EINVAL;
} }
mutex_unlock(&gameport_mutex); mutex_unlock(&gameport_mutex);
return retval; return error ? error : count;
} }
static struct device_attribute gameport_device_attrs[] = { static struct device_attribute gameport_device_attrs[] = {
@ -655,7 +642,7 @@ static void gameport_disconnect_port(struct gameport *gameport)
do { do {
parent = s->parent; parent = s->parent;
gameport_release_driver(s); device_release_driver(&s->dev);
gameport_destroy_port(s); gameport_destroy_port(s);
} while ((s = parent) != gameport); } while ((s = parent) != gameport);
} }
@ -663,7 +650,7 @@ static void gameport_disconnect_port(struct gameport *gameport)
/* /*
* Ok, no children left, now disconnect this port * Ok, no children left, now disconnect this port
*/ */
gameport_release_driver(gameport); device_release_driver(&gameport->dev);
} }
void gameport_rescan(struct gameport *gameport) void gameport_rescan(struct gameport *gameport)

View File

@ -115,18 +115,18 @@ static int serio_match_port(const struct serio_device_id *ids, struct serio *ser
* Basic serio -> driver core mappings * Basic serio -> driver core mappings
*/ */
static void serio_bind_driver(struct serio *serio, struct serio_driver *drv) static int serio_bind_driver(struct serio *serio, struct serio_driver *drv)
{ {
int error; int error;
down_write(&serio_bus.subsys.rwsem);
if (serio_match_port(drv->id_table, serio)) { if (serio_match_port(drv->id_table, serio)) {
serio->dev.driver = &drv->driver; serio->dev.driver = &drv->driver;
if (serio_connect_driver(serio, drv)) { if (serio_connect_driver(serio, drv)) {
serio->dev.driver = NULL; serio->dev.driver = NULL;
goto out; return -ENODEV;
} }
error = device_bind_driver(&serio->dev); error = device_bind_driver(&serio->dev);
if (error) { if (error) {
printk(KERN_WARNING printk(KERN_WARNING
@ -136,31 +136,21 @@ static void serio_bind_driver(struct serio *serio, struct serio_driver *drv)
drv->description, error); drv->description, error);
serio_disconnect_driver(serio); serio_disconnect_driver(serio);
serio->dev.driver = NULL; serio->dev.driver = NULL;
goto out; return error;
} }
} }
out: return 0;
up_write(&serio_bus.subsys.rwsem);
}
static void serio_release_driver(struct serio *serio)
{
down_write(&serio_bus.subsys.rwsem);
device_release_driver(&serio->dev);
up_write(&serio_bus.subsys.rwsem);
} }
static void serio_find_driver(struct serio *serio) static void serio_find_driver(struct serio *serio)
{ {
int error; int error;
down_write(&serio_bus.subsys.rwsem);
error = device_attach(&serio->dev); error = device_attach(&serio->dev);
if (error < 0) if (error < 0)
printk(KERN_WARNING printk(KERN_WARNING
"serio: device_attach() failed for %s (%s), error: %d\n", "serio: device_attach() failed for %s (%s), error: %d\n",
serio->phys, serio->name, error); serio->phys, serio->name, error);
up_write(&serio_bus.subsys.rwsem);
} }
@ -470,13 +460,12 @@ static ssize_t serio_rebind_driver(struct device *dev, struct device_attribute *
{ {
struct serio *serio = to_serio_port(dev); struct serio *serio = to_serio_port(dev);
struct device_driver *drv; struct device_driver *drv;
int retval; int error;
retval = mutex_lock_interruptible(&serio_mutex); error = mutex_lock_interruptible(&serio_mutex);
if (retval) if (error)
return retval; return error;
retval = count;
if (!strncmp(buf, "none", count)) { if (!strncmp(buf, "none", count)) {
serio_disconnect_port(serio); serio_disconnect_port(serio);
} else if (!strncmp(buf, "reconnect", count)) { } else if (!strncmp(buf, "reconnect", count)) {
@ -486,15 +475,15 @@ static ssize_t serio_rebind_driver(struct device *dev, struct device_attribute *
serio_find_driver(serio); serio_find_driver(serio);
} else if ((drv = driver_find(buf, &serio_bus)) != NULL) { } else if ((drv = driver_find(buf, &serio_bus)) != NULL) {
serio_disconnect_port(serio); serio_disconnect_port(serio);
serio_bind_driver(serio, to_serio_driver(drv)); error = serio_bind_driver(serio, to_serio_driver(drv));
put_driver(drv); put_driver(drv);
} else { } else {
retval = -EINVAL; error = -EINVAL;
} }
mutex_unlock(&serio_mutex); mutex_unlock(&serio_mutex);
return retval; return error ? error : count;
} }
static ssize_t serio_show_bind_mode(struct device *dev, struct device_attribute *attr, char *buf) static ssize_t serio_show_bind_mode(struct device *dev, struct device_attribute *attr, char *buf)
@ -665,7 +654,7 @@ static void serio_disconnect_port(struct serio *serio)
do { do {
parent = s->parent; parent = s->parent;
serio_release_driver(s); device_release_driver(&s->dev);
serio_destroy_port(s); serio_destroy_port(s);
} while ((s = parent) != serio); } while ((s = parent) != serio);
} }
@ -673,7 +662,7 @@ static void serio_disconnect_port(struct serio *serio)
/* /*
* Ok, no children left, now disconnect this port * Ok, no children left, now disconnect this port
*/ */
serio_release_driver(serio); device_release_driver(&serio->dev);
} }
void serio_rescan(struct serio *serio) void serio_rescan(struct serio *serio)

View File

@ -86,31 +86,26 @@ mmc_bus_uevent(struct device *dev, char **envp, int num_envp, char *buf,
{ {
struct mmc_card *card = dev_to_mmc_card(dev); struct mmc_card *card = dev_to_mmc_card(dev);
char ccc[13]; char ccc[13];
int i = 0; int retval = 0, i = 0, length = 0;
#define add_env(fmt,val) \ #define add_env(fmt,val) do { \
({ \ retval = add_uevent_var(envp, num_envp, &i, \
int len, ret = -ENOMEM; \ buf, buf_size, &length, \
if (i < num_envp) { \ fmt, val); \
envp[i++] = buf; \ if (retval) \
len = snprintf(buf, buf_size, fmt, val) + 1; \ return retval; \
buf_size -= len; \ } while (0);
buf += len; \
if (buf_size >= 0) \
ret = 0; \
} \
ret; \
})
for (i = 0; i < 12; i++) for (i = 0; i < 12; i++)
ccc[i] = card->csd.cmdclass & (1 << i) ? '1' : '0'; ccc[i] = card->csd.cmdclass & (1 << i) ? '1' : '0';
ccc[12] = '\0'; ccc[12] = '\0';
i = 0;
add_env("MMC_CCC=%s", ccc); add_env("MMC_CCC=%s", ccc);
add_env("MMC_MANFID=%06x", card->cid.manfid); add_env("MMC_MANFID=%06x", card->cid.manfid);
add_env("MMC_NAME=%s", mmc_card_name(card)); add_env("MMC_NAME=%s", mmc_card_name(card));
add_env("MMC_OEMID=%04x", card->cid.oemid); add_env("MMC_OEMID=%04x", card->cid.oemid);
#undef add_env
envp[i] = NULL;
return 0; return 0;
} }

View File

@ -276,21 +276,15 @@ static int fixed_mdio_register_device(int number, int speed, int duplex)
artificially, we are binding the driver here by hand; artificially, we are binding the driver here by hand;
it will be the same for all the fixed phys anyway. it will be the same for all the fixed phys anyway.
*/ */
down_write(&phydev->dev.bus->subsys.rwsem);
phydev->dev.driver = &fixed_mdio_driver.driver; phydev->dev.driver = &fixed_mdio_driver.driver;
err = phydev->dev.driver->probe(&phydev->dev); err = phydev->dev.driver->probe(&phydev->dev);
if(err < 0) { if(err < 0) {
printk(KERN_ERR "Phy %s: problems with fixed driver\n",phydev->dev.bus_id); printk(KERN_ERR "Phy %s: problems with fixed driver\n",phydev->dev.bus_id);
up_write(&phydev->dev.bus->subsys.rwsem);
goto probe_fail; goto probe_fail;
} }
err = device_bind_driver(&phydev->dev); err = device_bind_driver(&phydev->dev);
up_write(&phydev->dev.bus->subsys.rwsem);
if (err) if (err)
goto probe_fail; goto probe_fail;

View File

@ -208,16 +208,12 @@ struct phy_device *phy_attach(struct net_device *dev,
* exist, and we should use the genphy driver. */ * exist, and we should use the genphy driver. */
if (NULL == d->driver) { if (NULL == d->driver) {
int err; int err;
down_write(&d->bus->subsys.rwsem);
d->driver = &genphy_driver.driver; d->driver = &genphy_driver.driver;
err = d->driver->probe(d); err = d->driver->probe(d);
if (err >= 0) if (err >= 0)
err = device_bind_driver(d); err = device_bind_driver(d);
up_write(&d->bus->subsys.rwsem);
if (err) if (err)
return ERR_PTR(err); return ERR_PTR(err);
} }
@ -258,11 +254,8 @@ void phy_detach(struct phy_device *phydev)
* was using the generic driver), we unbind the device * was using the generic driver), we unbind the device
* from the generic driver so that there's a chance a * from the generic driver so that there's a chance a
* real driver could be loaded */ * real driver could be loaded */
if (phydev->dev.driver == &genphy_driver.driver) { if (phydev->dev.driver == &genphy_driver.driver)
down_write(&phydev->dev.bus->subsys.rwsem);
device_release_driver(&phydev->dev); device_release_driver(&phydev->dev);
up_write(&phydev->dev.bus->subsys.rwsem);
}
} }
EXPORT_SYMBOL(phy_detach); EXPORT_SYMBOL(phy_detach);

View File

@ -434,11 +434,6 @@ int __pci_register_driver(struct pci_driver *drv, struct module *owner,
drv->driver.mod_name = mod_name; drv->driver.mod_name = mod_name;
drv->driver.kobj.ktype = &pci_driver_kobj_type; drv->driver.kobj.ktype = &pci_driver_kobj_type;
if (pci_multithread_probe)
drv->driver.multithread_probe = pci_multithread_probe;
else
drv->driver.multithread_probe = drv->multithread_probe;
spin_lock_init(&drv->dynids.lock); spin_lock_init(&drv->dynids.lock);
INIT_LIST_HEAD(&drv->dynids.list); INIT_LIST_HEAD(&drv->dynids.list);
@ -574,6 +569,7 @@ struct bus_type pci_bus_type = {
static int __init pci_driver_init(void) static int __init pci_driver_init(void)
{ {
pci_bus_type.multithread_probe = pci_multithread_probe;
return bus_register(&pci_bus_type); return bus_register(&pci_bus_type);
} }

View File

@ -13,6 +13,7 @@
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/pci.h> #include <linux/pci.h>
#include <linux/pm.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/spinlock.h> #include <linux/spinlock.h>
#include <linux/string.h> #include <linux/string.h>
@ -891,31 +892,48 @@ pci_disable_device(struct pci_dev *dev)
} }
/** /**
* pci_enable_wake - enable device to generate PME# when suspended * pci_enable_wake - enable PCI device as wakeup event source
* @dev: - PCI device to operate on * @dev: PCI device affected
* @state: - Current state of device. * @state: PCI state from which device will issue wakeup events
* @enable: - Flag to enable or disable generation * @enable: True to enable event generation; false to disable
*
* Set the bits in the device's PM Capabilities to generate PME# when
* the system is suspended.
* *
* -EIO is returned if device doesn't have PM Capabilities. * This enables the device as a wakeup event source, or disables it.
* -EINVAL is returned if device supports it, but can't generate wake events. * When such events involves platform-specific hooks, those hooks are
* 0 if operation is successful. * called automatically by this routine.
* *
* Devices with legacy power management (no standard PCI PM capabilities)
* always require such platform hooks. Depending on the platform, devices
* supporting the standard PCI PME# signal may require such platform hooks;
* they always update bits in config space to allow PME# generation.
*
* -EIO is returned if the device can't ever be a wakeup event source.
* -EINVAL is returned if the device can't generate wakeup events from
* the specified PCI state. Returns zero if the operation is successful.
*/ */
int pci_enable_wake(struct pci_dev *dev, pci_power_t state, int enable) int pci_enable_wake(struct pci_dev *dev, pci_power_t state, int enable)
{ {
int pm; int pm;
int status;
u16 value; u16 value;
/* Note that drivers should verify device_may_wakeup(&dev->dev)
* before calling this function. Platform code should report
* errors when drivers try to enable wakeup on devices that
* can't issue wakeups, or on which wakeups were disabled by
* userspace updating the /sys/devices.../power/wakeup file.
*/
status = call_platform_enable_wakeup(&dev->dev, enable);
/* find PCI PM capability in list */ /* find PCI PM capability in list */
pm = pci_find_capability(dev, PCI_CAP_ID_PM); pm = pci_find_capability(dev, PCI_CAP_ID_PM);
/* If device doesn't support PM Capabilities, but request is to disable /* If device doesn't support PM Capabilities, but caller wants to
* wake events, it's a nop; otherwise fail */ * disable wake events, it's a NOP. Otherwise fail unless the
if (!pm) * platform hooks handled this legacy device already.
return enable ? -EIO : 0; */
if (!pm)
return enable ? status : 0;
/* Check device's ability to generate PME# */ /* Check device's ability to generate PME# */
pci_read_config_word(dev,pm+PCI_PM_PMC,&value); pci_read_config_word(dev,pm+PCI_PM_PMC,&value);
@ -924,8 +942,14 @@ int pci_enable_wake(struct pci_dev *dev, pci_power_t state, int enable)
value >>= ffs(PCI_PM_CAP_PME_MASK) - 1; /* First bit of mask */ value >>= ffs(PCI_PM_CAP_PME_MASK) - 1; /* First bit of mask */
/* Check if it can generate PME# from requested state. */ /* Check if it can generate PME# from requested state. */
if (!value || !(value & (1 << state))) if (!value || !(value & (1 << state))) {
/* if it can't, revert what the platform hook changed,
* always reporting the base "EINVAL, can't PME#" error
*/
if (enable)
call_platform_enable_wakeup(&dev->dev, 0);
return enable ? -EINVAL : 0; return enable ? -EINVAL : 0;
}
pci_read_config_word(dev, pm + PCI_PM_CTRL, &value); pci_read_config_word(dev, pm + PCI_PM_CTRL, &value);
@ -936,7 +960,7 @@ int pci_enable_wake(struct pci_dev *dev, pci_power_t state, int enable)
value &= ~PCI_PM_CTRL_PME_ENABLE; value &= ~PCI_PM_CTRL_PME_ENABLE;
pci_write_config_word(dev, pm + PCI_PM_CTRL, value); pci_write_config_word(dev, pm + PCI_PM_CTRL, value);
return 0; return 0;
} }

View File

@ -311,7 +311,6 @@ done:
return NULL; return NULL;
found: found:
down_write(&dev->dev.bus->subsys.rwsem);
dev->card_link = clink; dev->card_link = clink;
dev->dev.driver = &drv->link.driver; dev->dev.driver = &drv->link.driver;
if (pnp_bus_type.probe(&dev->dev)) if (pnp_bus_type.probe(&dev->dev))
@ -319,14 +318,11 @@ found:
if (device_bind_driver(&dev->dev)) if (device_bind_driver(&dev->dev))
goto err_out; goto err_out;
up_write(&dev->dev.bus->subsys.rwsem);
return dev; return dev;
err_out: err_out:
dev->dev.driver = NULL; dev->dev.driver = NULL;
dev->card_link = NULL; dev->card_link = NULL;
up_write(&dev->dev.bus->subsys.rwsem);
return NULL; return NULL;
} }
@ -340,11 +336,9 @@ void pnp_release_card_device(struct pnp_dev * dev)
struct pnp_card_driver * drv = dev->card_link->driver; struct pnp_card_driver * drv = dev->card_link->driver;
if (!drv) if (!drv)
return; return;
down_write(&dev->dev.bus->subsys.rwsem);
drv->link.remove = &card_remove; drv->link.remove = &card_remove;
device_release_driver(&dev->dev); device_release_driver(&dev->dev);
drv->link.remove = &card_remove_first; drv->link.remove = &card_remove_first;
up_write(&dev->dev.bus->subsys.rwsem);
} }
/* /*

View File

@ -871,6 +871,12 @@ io_subchannel_register(struct work_struct *work)
} }
goto out; goto out;
} }
/*
* Now we know this subchannel will stay, we can throw
* our delayed uevent.
*/
sch->dev.uevent_suppress = 0;
kobject_uevent(&sch->dev.kobj, KOBJ_ADD);
/* make it known to the system */ /* make it known to the system */
ret = ccw_device_register(cdev); ret = ccw_device_register(cdev);
if (ret) { if (ret) {

View File

@ -423,27 +423,25 @@ static int ap_uevent (struct device *dev, char **envp, int num_envp,
char *buffer, int buffer_size) char *buffer, int buffer_size)
{ {
struct ap_device *ap_dev = to_ap_dev(dev); struct ap_device *ap_dev = to_ap_dev(dev);
int length; int retval = 0, length = 0, i = 0;
if (!ap_dev) if (!ap_dev)
return -ENODEV; return -ENODEV;
/* Set up DEV_TYPE environment variable. */ /* Set up DEV_TYPE environment variable. */
envp[0] = buffer; retval = add_uevent_var(envp, num_envp, &i,
length = scnprintf(buffer, buffer_size, "DEV_TYPE=%04X", buffer, buffer_size, &length,
ap_dev->device_type); "DEV_TYPE=%04X", ap_dev->device_type);
if (buffer_size - length <= 0) if (retval)
return -ENOMEM; return retval;
buffer += length;
buffer_size -= length;
/* Add MODALIAS= */ /* Add MODALIAS= */
envp[1] = buffer; retval = add_uevent_var(envp, num_envp, &i,
length = scnprintf(buffer, buffer_size, "MODALIAS=ap:t%02X", buffer, buffer_size, &length,
ap_dev->device_type); "MODALIAS=ap:t%02X", ap_dev->device_type);
if (buffer_size - length <= 0)
return -ENOMEM; envp[i] = NULL;
envp[2] = NULL; return retval;
return 0;
} }
static struct bus_type ap_bus_type = { static struct bus_type ap_bus_type = {

View File

@ -37,7 +37,6 @@ qeth_procfile_seq_start(struct seq_file *s, loff_t *offset)
struct device *dev = NULL; struct device *dev = NULL;
loff_t nr = 0; loff_t nr = 0;
down_read(&qeth_ccwgroup_driver.driver.bus->subsys.rwsem);
if (*offset == 0) if (*offset == 0)
return SEQ_START_TOKEN; return SEQ_START_TOKEN;
while (1) { while (1) {
@ -53,7 +52,6 @@ qeth_procfile_seq_start(struct seq_file *s, loff_t *offset)
static void static void
qeth_procfile_seq_stop(struct seq_file *s, void* it) qeth_procfile_seq_stop(struct seq_file *s, void* it)
{ {
up_read(&qeth_ccwgroup_driver.driver.bus->subsys.rwsem);
} }
static void * static void *

View File

@ -435,7 +435,7 @@ struct Scsi_Host *scsi_host_lookup(unsigned short hostnum)
struct class_device *cdev; struct class_device *cdev;
struct Scsi_Host *shost = ERR_PTR(-ENXIO), *p; struct Scsi_Host *shost = ERR_PTR(-ENXIO), *p;
down_read(&class->subsys.rwsem); down(&class->sem);
list_for_each_entry(cdev, &class->children, node) { list_for_each_entry(cdev, &class->children, node) {
p = class_to_shost(cdev); p = class_to_shost(cdev);
if (p->host_no == hostnum) { if (p->host_no == hostnum) {
@ -443,7 +443,7 @@ struct Scsi_Host *scsi_host_lookup(unsigned short hostnum)
break; break;
} }
} }
up_read(&class->subsys.rwsem); up(&class->sem);
return shost; return shost;
} }

View File

@ -246,7 +246,6 @@ static char *usb_dump_interface_descriptor(char *start, char *end,
if (start > end) if (start > end)
return start; return start;
down_read(&usb_bus_type.subsys.rwsem);
if (iface) { if (iface) {
driver_name = (iface->dev.driver driver_name = (iface->dev.driver
? iface->dev.driver->name ? iface->dev.driver->name
@ -263,7 +262,6 @@ static char *usb_dump_interface_descriptor(char *start, char *end,
desc->bInterfaceSubClass, desc->bInterfaceSubClass,
desc->bInterfaceProtocol, desc->bInterfaceProtocol,
driver_name); driver_name);
up_read(&usb_bus_type.subsys.rwsem);
return start; return start;
} }

View File

@ -421,14 +421,11 @@ static int claimintf(struct dev_state *ps, unsigned int ifnum)
if (test_bit(ifnum, &ps->ifclaimed)) if (test_bit(ifnum, &ps->ifclaimed))
return 0; return 0;
/* lock against other changes to driver bindings */
down_write(&usb_bus_type.subsys.rwsem);
intf = usb_ifnum_to_if(dev, ifnum); intf = usb_ifnum_to_if(dev, ifnum);
if (!intf) if (!intf)
err = -ENOENT; err = -ENOENT;
else else
err = usb_driver_claim_interface(&usbfs_driver, intf, ps); err = usb_driver_claim_interface(&usbfs_driver, intf, ps);
up_write(&usb_bus_type.subsys.rwsem);
if (err == 0) if (err == 0)
set_bit(ifnum, &ps->ifclaimed); set_bit(ifnum, &ps->ifclaimed);
return err; return err;
@ -444,8 +441,6 @@ static int releaseintf(struct dev_state *ps, unsigned int ifnum)
if (ifnum >= 8*sizeof(ps->ifclaimed)) if (ifnum >= 8*sizeof(ps->ifclaimed))
return err; return err;
dev = ps->dev; dev = ps->dev;
/* lock against other changes to driver bindings */
down_write(&usb_bus_type.subsys.rwsem);
intf = usb_ifnum_to_if(dev, ifnum); intf = usb_ifnum_to_if(dev, ifnum);
if (!intf) if (!intf)
err = -ENOENT; err = -ENOENT;
@ -453,7 +448,6 @@ static int releaseintf(struct dev_state *ps, unsigned int ifnum)
usb_driver_release_interface(&usbfs_driver, intf); usb_driver_release_interface(&usbfs_driver, intf);
err = 0; err = 0;
} }
up_write(&usb_bus_type.subsys.rwsem);
return err; return err;
} }
@ -813,7 +807,6 @@ static int proc_getdriver(struct dev_state *ps, void __user *arg)
if (copy_from_user(&gd, arg, sizeof(gd))) if (copy_from_user(&gd, arg, sizeof(gd)))
return -EFAULT; return -EFAULT;
down_read(&usb_bus_type.subsys.rwsem);
intf = usb_ifnum_to_if(ps->dev, gd.interface); intf = usb_ifnum_to_if(ps->dev, gd.interface);
if (!intf || !intf->dev.driver) if (!intf || !intf->dev.driver)
ret = -ENODATA; ret = -ENODATA;
@ -822,7 +815,6 @@ static int proc_getdriver(struct dev_state *ps, void __user *arg)
sizeof(gd.driver)); sizeof(gd.driver));
ret = (copy_to_user(arg, &gd, sizeof(gd)) ? -EFAULT : 0); ret = (copy_to_user(arg, &gd, sizeof(gd)) ? -EFAULT : 0);
} }
up_read(&usb_bus_type.subsys.rwsem);
return ret; return ret;
} }
@ -1351,15 +1343,12 @@ static int proc_ioctl(struct dev_state *ps, struct usbdevfs_ioctl *ctl)
/* disconnect kernel driver from interface */ /* disconnect kernel driver from interface */
case USBDEVFS_DISCONNECT: case USBDEVFS_DISCONNECT:
down_write(&usb_bus_type.subsys.rwsem);
if (intf->dev.driver) { if (intf->dev.driver) {
driver = to_usb_driver(intf->dev.driver); driver = to_usb_driver(intf->dev.driver);
dev_dbg (&intf->dev, "disconnect by usbfs\n"); dev_dbg (&intf->dev, "disconnect by usbfs\n");
usb_driver_release_interface(driver, intf); usb_driver_release_interface(driver, intf);
} else } else
retval = -ENODATA; retval = -ENODATA;
up_write(&usb_bus_type.subsys.rwsem);
break; break;
/* let kernel drivers try to (re)bind to the interface */ /* let kernel drivers try to (re)bind to the interface */
@ -1371,7 +1360,6 @@ static int proc_ioctl(struct dev_state *ps, struct usbdevfs_ioctl *ctl)
/* talk directly to the interface's driver */ /* talk directly to the interface's driver */
default: default:
down_read(&usb_bus_type.subsys.rwsem);
if (intf->dev.driver) if (intf->dev.driver)
driver = to_usb_driver(intf->dev.driver); driver = to_usb_driver(intf->dev.driver);
if (driver == NULL || driver->ioctl == NULL) { if (driver == NULL || driver->ioctl == NULL) {
@ -1381,7 +1369,6 @@ static int proc_ioctl(struct dev_state *ps, struct usbdevfs_ioctl *ctl)
if (retval == -ENOIOCTLCMD) if (retval == -ENOIOCTLCMD)
retval = -ENOTTY; retval = -ENOTTY;
} }
up_read(&usb_bus_type.subsys.rwsem);
} }
/* cleanup and return */ /* cleanup and return */

View File

@ -287,9 +287,9 @@ static int usb_unbind_interface(struct device *dev)
* way to bind to an interface is to return the private data from * way to bind to an interface is to return the private data from
* the driver's probe() method. * the driver's probe() method.
* *
* Callers must own the device lock and the driver model's usb_bus_type.subsys * Callers must own the device lock, so driver probe() entries don't need
* writelock. So driver probe() entries don't need extra locking, * extra locking, but other call contexts may need to explicitly claim that
* but other call contexts may need to explicitly claim those locks. * lock.
*/ */
int usb_driver_claim_interface(struct usb_driver *driver, int usb_driver_claim_interface(struct usb_driver *driver,
struct usb_interface *iface, void* priv) struct usb_interface *iface, void* priv)
@ -330,9 +330,9 @@ EXPORT_SYMBOL(usb_driver_claim_interface);
* also causes the driver disconnect() method to be called. * also causes the driver disconnect() method to be called.
* *
* This call is synchronous, and may not be used in an interrupt context. * This call is synchronous, and may not be used in an interrupt context.
* Callers must own the device lock and the driver model's usb_bus_type.subsys * Callers must own the device lock, so driver disconnect() entries don't
* writelock. So driver disconnect() entries don't need extra locking, * need extra locking, but other call contexts may need to explicitly claim
* but other call contexts may need to explicitly claim those locks. * that lock.
*/ */
void usb_driver_release_interface(struct usb_driver *driver, void usb_driver_release_interface(struct usb_driver *driver,
struct usb_interface *iface) struct usb_interface *iface)

View File

@ -119,8 +119,7 @@ MODULE_PARM_DESC(use_both_schemes,
"first one fails"); "first one fails");
#ifdef DEBUG static inline char *portspeed(int portstatus)
static inline char *portspeed (int portstatus)
{ {
if (portstatus & (1 << USB_PORT_FEAT_HIGHSPEED)) if (portstatus & (1 << USB_PORT_FEAT_HIGHSPEED))
return "480 Mb/s"; return "480 Mb/s";
@ -129,7 +128,6 @@ static inline char *portspeed (int portstatus)
else else
return "12 Mb/s"; return "12 Mb/s";
} }
#endif
/* Note that hdev or one of its children must be locked! */ /* Note that hdev or one of its children must be locked! */
static inline struct usb_hub *hdev_to_hub(struct usb_device *hdev) static inline struct usb_hub *hdev_to_hub(struct usb_device *hdev)

View File

@ -1349,7 +1349,7 @@ static void release_interface(struct device *dev)
* *
* This call is synchronous. The calling context must be able to sleep, * This call is synchronous. The calling context must be able to sleep,
* must own the device lock, and must not hold the driver model's USB * must own the device lock, and must not hold the driver model's USB
* bus rwsem; usb device driver probe() methods cannot use this routine. * bus mutex; usb device driver probe() methods cannot use this routine.
* *
* Returns zero on success, or else the status code returned by the * Returns zero on success, or else the status code returned by the
* underlying call that failed. On successful completion, each interface * underlying call that failed. On successful completion, each interface

View File

@ -486,9 +486,6 @@ static int ohci_run (struct ohci_hcd *ohci)
* or if bus glue did the same (e.g. for PCI add-in cards with * or if bus glue did the same (e.g. for PCI add-in cards with
* PCI PM support). * PCI PM support).
*/ */
ohci_dbg (ohci, "resetting from state '%s', control = 0x%x\n",
hcfs2string (ohci->hc_control & OHCI_CTRL_HCFS),
ohci_readl (ohci, &ohci->regs->control));
if ((ohci->hc_control & OHCI_CTRL_RWC) != 0 if ((ohci->hc_control & OHCI_CTRL_RWC) != 0
&& !device_may_wakeup(hcd->self.controller)) && !device_may_wakeup(hcd->self.controller))
device_init_wakeup(hcd->self.controller, 1); device_init_wakeup(hcd->self.controller, 1);
@ -744,9 +741,6 @@ static void ohci_stop (struct usb_hcd *hcd)
{ {
struct ohci_hcd *ohci = hcd_to_ohci (hcd); struct ohci_hcd *ohci = hcd_to_ohci (hcd);
ohci_dbg (ohci, "stop %s controller (state 0x%02x)\n",
hcfs2string (ohci->hc_control & OHCI_CTRL_HCFS),
hcd->state);
ohci_dump (ohci, 1); ohci_dump (ohci, 1);
flush_scheduled_work(); flush_scheduled_work();

View File

@ -179,6 +179,48 @@ struct dentry *debugfs_create_u32(const char *name, mode_t mode,
} }
EXPORT_SYMBOL_GPL(debugfs_create_u32); EXPORT_SYMBOL_GPL(debugfs_create_u32);
static void debugfs_u64_set(void *data, u64 val)
{
*(u64 *)data = val;
}
static u64 debugfs_u64_get(void *data)
{
return *(u64 *)data;
}
DEFINE_SIMPLE_ATTRIBUTE(fops_u64, debugfs_u64_get, debugfs_u64_set, "%llu\n");
/**
* debugfs_create_u64 - create a debugfs file that is used to read and write an unsigned 64-bit value
* @name: a pointer to a string containing the name of the file to create.
* @mode: the permission that the file should have
* @parent: a pointer to the parent dentry for this file. This should be a
* directory dentry if set. If this parameter is %NULL, then the
* file will be created in the root of the debugfs filesystem.
* @value: a pointer to the variable that the file should read to and write
* from.
*
* This function creates a file in debugfs with the given name that
* contains the value of the variable @value. If the @mode variable is so
* set, it can be read from, and written to.
*
* This function will return a pointer to a dentry if it succeeds. This
* pointer must be passed to the debugfs_remove() function when the file is
* to be removed (no automatic cleanup happens if your module is unloaded,
* you are responsible here.) If an error occurs, %NULL will be returned.
*
* If debugfs is not enabled in the kernel, the value -%ENODEV will be
* returned. It is not wise to check for this value, but rather, check for
* %NULL or !%NULL instead as to eliminate the need for #ifdef in the calling
* code.
*/
struct dentry *debugfs_create_u64(const char *name, mode_t mode,
struct dentry *parent, u64 *value)
{
return debugfs_create_file(name, mode, parent, value, &fops_u64);
}
EXPORT_SYMBOL_GPL(debugfs_create_u64);
static ssize_t read_file_bool(struct file *file, char __user *user_buf, static ssize_t read_file_bool(struct file *file, char __user *user_buf,
size_t count, loff_t *ppos) size_t count, loff_t *ppos)
{ {

View File

@ -1243,22 +1243,13 @@ int __user_path_lookup_open(const char __user *name, unsigned int lookup_flags,
return err; return err;
} }
/* static inline struct dentry *__lookup_hash_kern(struct qstr *name, struct dentry *base, struct nameidata *nd)
* Restricted form of lookup. Doesn't follow links, single-component only,
* needs parent already locked. Doesn't follow mounts.
* SMP-safe.
*/
static struct dentry * __lookup_hash(struct qstr *name, struct dentry * base, struct nameidata *nd)
{ {
struct dentry * dentry; struct dentry *dentry;
struct inode *inode; struct inode *inode;
int err; int err;
inode = base->d_inode; inode = base->d_inode;
err = permission(inode, MAY_EXEC, nd);
dentry = ERR_PTR(err);
if (err)
goto out;
/* /*
* See if the low-level filesystem might want * See if the low-level filesystem might want
@ -1287,35 +1278,76 @@ out:
return dentry; return dentry;
} }
/*
* Restricted form of lookup. Doesn't follow links, single-component only,
* needs parent already locked. Doesn't follow mounts.
* SMP-safe.
*/
static inline struct dentry * __lookup_hash(struct qstr *name, struct dentry *base, struct nameidata *nd)
{
struct dentry *dentry;
struct inode *inode;
int err;
inode = base->d_inode;
err = permission(inode, MAY_EXEC, nd);
dentry = ERR_PTR(err);
if (err)
goto out;
dentry = __lookup_hash_kern(name, base, nd);
out:
return dentry;
}
static struct dentry *lookup_hash(struct nameidata *nd) static struct dentry *lookup_hash(struct nameidata *nd)
{ {
return __lookup_hash(&nd->last, nd->dentry, nd); return __lookup_hash(&nd->last, nd->dentry, nd);
} }
/* SMP-safe */ /* SMP-safe */
struct dentry * lookup_one_len(const char * name, struct dentry * base, int len) static inline int __lookup_one_len(const char *name, struct qstr *this, struct dentry *base, int len)
{ {
unsigned long hash; unsigned long hash;
struct qstr this;
unsigned int c; unsigned int c;
this.name = name; this->name = name;
this.len = len; this->len = len;
if (!len) if (!len)
goto access; return -EACCES;
hash = init_name_hash(); hash = init_name_hash();
while (len--) { while (len--) {
c = *(const unsigned char *)name++; c = *(const unsigned char *)name++;
if (c == '/' || c == '\0') if (c == '/' || c == '\0')
goto access; return -EACCES;
hash = partial_name_hash(c, hash); hash = partial_name_hash(c, hash);
} }
this.hash = end_name_hash(hash); this->hash = end_name_hash(hash);
return 0;
}
struct dentry *lookup_one_len(const char *name, struct dentry *base, int len)
{
int err;
struct qstr this;
err = __lookup_one_len(name, &this, base, len);
if (err)
return ERR_PTR(err);
return __lookup_hash(&this, base, NULL); return __lookup_hash(&this, base, NULL);
access: }
return ERR_PTR(-EACCES);
struct dentry *lookup_one_len_kern(const char *name, struct dentry *base, int len)
{
int err;
struct qstr this;
err = __lookup_one_len(name, &this, base, len);
if (err)
return ERR_PTR(err);
return __lookup_hash_kern(&this, base, NULL);
} }
/* /*

View File

@ -725,16 +725,6 @@ static int test_bdev_super(struct super_block *s, void *data)
return (void *)s->s_bdev == data; return (void *)s->s_bdev == data;
} }
static void bdev_uevent(struct block_device *bdev, enum kobject_action action)
{
if (bdev->bd_disk) {
if (bdev->bd_part)
kobject_uevent(&bdev->bd_part->kobj, action);
else
kobject_uevent(&bdev->bd_disk->kobj, action);
}
}
int get_sb_bdev(struct file_system_type *fs_type, int get_sb_bdev(struct file_system_type *fs_type,
int flags, const char *dev_name, void *data, int flags, const char *dev_name, void *data,
int (*fill_super)(struct super_block *, void *, int), int (*fill_super)(struct super_block *, void *, int),
@ -782,7 +772,6 @@ int get_sb_bdev(struct file_system_type *fs_type,
} }
s->s_flags |= MS_ACTIVE; s->s_flags |= MS_ACTIVE;
bdev_uevent(bdev, KOBJ_MOUNT);
} }
return simple_set_mnt(mnt, s); return simple_set_mnt(mnt, s);
@ -801,7 +790,6 @@ void kill_block_super(struct super_block *sb)
{ {
struct block_device *bdev = sb->s_bdev; struct block_device *bdev = sb->s_bdev;
bdev_uevent(bdev, KOBJ_UMOUNT);
generic_shutdown_super(sb); generic_shutdown_super(sb);
sync_blockdev(bdev); sync_blockdev(bdev);
close_bdev_excl(bdev); close_bdev_excl(bdev);

View File

@ -59,7 +59,7 @@ read(struct file * file, char __user * userbuf, size_t count, loff_t * off)
if (copy_to_user(userbuf, buffer, count)) if (copy_to_user(userbuf, buffer, count))
return -EFAULT; return -EFAULT;
pr_debug("offs = %lld, *off = %lld, count = %zd\n", offs, *off, count); pr_debug("offs = %lld, *off = %lld, count = %d\n", offs, *off, count);
*off = offs + count; *off = offs + count;

View File

@ -633,6 +633,7 @@ struct sysfs_schedule_callback_struct {
struct kobject *kobj; struct kobject *kobj;
void (*func)(void *); void (*func)(void *);
void *data; void *data;
struct module *owner;
struct work_struct work; struct work_struct work;
}; };
@ -643,6 +644,7 @@ static void sysfs_schedule_callback_work(struct work_struct *work)
(ss->func)(ss->data); (ss->func)(ss->data);
kobject_put(ss->kobj); kobject_put(ss->kobj);
module_put(ss->owner);
kfree(ss); kfree(ss);
} }
@ -651,6 +653,7 @@ static void sysfs_schedule_callback_work(struct work_struct *work)
* @kobj: object we're acting for. * @kobj: object we're acting for.
* @func: callback function to invoke later. * @func: callback function to invoke later.
* @data: argument to pass to @func. * @data: argument to pass to @func.
* @owner: module owning the callback code
* *
* sysfs attribute methods must not unregister themselves or their parent * sysfs attribute methods must not unregister themselves or their parent
* kobject (which would amount to the same thing). Attempts to do so will * kobject (which would amount to the same thing). Attempts to do so will
@ -663,20 +666,25 @@ static void sysfs_schedule_callback_work(struct work_struct *work)
* until @func returns. * until @func returns.
* *
* Returns 0 if the request was submitted, -ENOMEM if storage could not * Returns 0 if the request was submitted, -ENOMEM if storage could not
* be allocated. * be allocated, -ENODEV if a reference to @owner isn't available.
*/ */
int sysfs_schedule_callback(struct kobject *kobj, void (*func)(void *), int sysfs_schedule_callback(struct kobject *kobj, void (*func)(void *),
void *data) void *data, struct module *owner)
{ {
struct sysfs_schedule_callback_struct *ss; struct sysfs_schedule_callback_struct *ss;
if (!try_module_get(owner))
return -ENODEV;
ss = kmalloc(sizeof(*ss), GFP_KERNEL); ss = kmalloc(sizeof(*ss), GFP_KERNEL);
if (!ss) if (!ss) {
module_put(owner);
return -ENOMEM; return -ENOMEM;
}
kobject_get(kobj); kobject_get(kobj);
ss->kobj = kobj; ss->kobj = kobj;
ss->func = func; ss->func = func;
ss->data = data; ss->data = data;
ss->owner = owner;
INIT_WORK(&ss->work, sysfs_schedule_callback_work); INIT_WORK(&ss->work, sysfs_schedule_callback_work);
schedule_work(&ss->work); schedule_work(&ss->work);
return 0; return 0;

View File

@ -70,9 +70,11 @@ void sysfs_remove_group(struct kobject * kobj,
{ {
struct dentry * dir; struct dentry * dir;
if (grp->name) if (grp->name) {
dir = lookup_one_len(grp->name, kobj->dentry, dir = lookup_one_len_kern(grp->name, kobj->dentry,
strlen(grp->name)); strlen(grp->name));
BUG_ON(IS_ERR(dir));
}
else else
dir = dget(kobj->dentry); dir = dget(kobj->dentry);

View File

@ -44,6 +44,8 @@ struct dentry *debugfs_create_u16(const char *name, mode_t mode,
struct dentry *parent, u16 *value); struct dentry *parent, u16 *value);
struct dentry *debugfs_create_u32(const char *name, mode_t mode, struct dentry *debugfs_create_u32(const char *name, mode_t mode,
struct dentry *parent, u32 *value); struct dentry *parent, u32 *value);
struct dentry *debugfs_create_u64(const char *name, mode_t mode,
struct dentry *parent, u64 *value);
struct dentry *debugfs_create_bool(const char *name, mode_t mode, struct dentry *debugfs_create_bool(const char *name, mode_t mode,
struct dentry *parent, u32 *value); struct dentry *parent, u32 *value);
@ -104,6 +106,13 @@ static inline struct dentry *debugfs_create_u32(const char *name, mode_t mode,
return ERR_PTR(-ENODEV); return ERR_PTR(-ENODEV);
} }
static inline struct dentry *debugfs_create_u64(const char *name, mode_t mode,
struct dentry *parent,
u64 *value)
{
return ERR_PTR(-ENODEV);
}
static inline struct dentry *debugfs_create_bool(const char *name, mode_t mode, static inline struct dentry *debugfs_create_bool(const char *name, mode_t mode,
struct dentry *parent, struct dentry *parent,
u32 *value) u32 *value)

View File

@ -34,9 +34,24 @@ struct device;
struct device_driver; struct device_driver;
struct class; struct class;
struct class_device; struct class_device;
struct bus_type;
struct bus_attribute {
struct attribute attr;
ssize_t (*show)(struct bus_type *, char * buf);
ssize_t (*store)(struct bus_type *, const char * buf, size_t count);
};
#define BUS_ATTR(_name,_mode,_show,_store) \
struct bus_attribute bus_attr_##_name = __ATTR(_name,_mode,_show,_store)
extern int __must_check bus_create_file(struct bus_type *,
struct bus_attribute *);
extern void bus_remove_file(struct bus_type *, struct bus_attribute *);
struct bus_type { struct bus_type {
const char * name; const char * name;
struct module * owner;
struct subsystem subsys; struct subsystem subsys;
struct kset drivers; struct kset drivers;
@ -49,6 +64,8 @@ struct bus_type {
struct bus_attribute * bus_attrs; struct bus_attribute * bus_attrs;
struct device_attribute * dev_attrs; struct device_attribute * dev_attrs;
struct driver_attribute * drv_attrs; struct driver_attribute * drv_attrs;
struct bus_attribute drivers_autoprobe_attr;
struct bus_attribute drivers_probe_attr;
int (*match)(struct device * dev, struct device_driver * drv); int (*match)(struct device * dev, struct device_driver * drv);
int (*uevent)(struct device *dev, char **envp, int (*uevent)(struct device *dev, char **envp,
@ -61,6 +78,9 @@ struct bus_type {
int (*suspend_late)(struct device * dev, pm_message_t state); int (*suspend_late)(struct device * dev, pm_message_t state);
int (*resume_early)(struct device * dev); int (*resume_early)(struct device * dev);
int (*resume)(struct device * dev); int (*resume)(struct device * dev);
unsigned int drivers_autoprobe:1;
unsigned int multithread_probe:1;
}; };
extern int __must_check bus_register(struct bus_type * bus); extern int __must_check bus_register(struct bus_type * bus);
@ -102,26 +122,10 @@ extern int bus_unregister_notifier(struct bus_type *bus,
#define BUS_NOTIFY_UNBIND_DRIVER 0x00000004 /* driver about to be #define BUS_NOTIFY_UNBIND_DRIVER 0x00000004 /* driver about to be
unbound */ unbound */
/* sysfs interface for exporting bus attributes */
struct bus_attribute {
struct attribute attr;
ssize_t (*show)(struct bus_type *, char * buf);
ssize_t (*store)(struct bus_type *, const char * buf, size_t count);
};
#define BUS_ATTR(_name,_mode,_show,_store) \
struct bus_attribute bus_attr_##_name = __ATTR(_name,_mode,_show,_store)
extern int __must_check bus_create_file(struct bus_type *,
struct bus_attribute *);
extern void bus_remove_file(struct bus_type *, struct bus_attribute *);
struct device_driver { struct device_driver {
const char * name; const char * name;
struct bus_type * bus; struct bus_type * bus;
struct completion unloaded;
struct kobject kobj; struct kobject kobj;
struct klist klist_devices; struct klist klist_devices;
struct klist_node knode_bus; struct klist_node knode_bus;
@ -135,8 +139,6 @@ struct device_driver {
void (*shutdown) (struct device * dev); void (*shutdown) (struct device * dev);
int (*suspend) (struct device * dev, pm_message_t state); int (*suspend) (struct device * dev, pm_message_t state);
int (*resume) (struct device * dev); int (*resume) (struct device * dev);
unsigned int multithread_probe:1;
}; };
@ -181,10 +183,9 @@ struct class {
struct list_head children; struct list_head children;
struct list_head devices; struct list_head devices;
struct list_head interfaces; struct list_head interfaces;
struct kset class_dirs;
struct semaphore sem; /* locks both the children and interfaces lists */ struct semaphore sem; /* locks both the children and interfaces lists */
struct kobject *virtual_dir;
struct class_attribute * class_attrs; struct class_attribute * class_attrs;
struct class_device_attribute * class_dev_attrs; struct class_device_attribute * class_dev_attrs;
struct device_attribute * dev_attrs; struct device_attribute * dev_attrs;
@ -328,11 +329,23 @@ extern struct class_device *class_device_create(struct class *cls,
__attribute__((format(printf,5,6))); __attribute__((format(printf,5,6)));
extern void class_device_destroy(struct class *cls, dev_t devt); extern void class_device_destroy(struct class *cls, dev_t devt);
/*
* The type of device, "struct device" is embedded in. A class
* or bus can contain devices of different types
* like "partitions" and "disks", "mouse" and "event".
* This identifies the device type and carries type-specific
* information, equivalent to the kobj_type of a kobject.
* If "name" is specified, the uevent will contain it in
* the DEVTYPE variable.
*/
struct device_type { struct device_type {
struct device_attribute *attrs; const char *name;
struct attribute_group **groups;
int (*uevent)(struct device *dev, char **envp, int num_envp, int (*uevent)(struct device *dev, char **envp, int num_envp,
char *buffer, int buffer_size); char *buffer, int buffer_size);
void (*release)(struct device *dev); void (*release)(struct device *dev);
int (*suspend)(struct device * dev, pm_message_t state);
int (*resume)(struct device * dev);
}; };
/* interface for exporting device attributes */ /* interface for exporting device attributes */
@ -354,8 +367,12 @@ extern int __must_check device_create_bin_file(struct device *dev,
struct bin_attribute *attr); struct bin_attribute *attr);
extern void device_remove_bin_file(struct device *dev, extern void device_remove_bin_file(struct device *dev,
struct bin_attribute *attr); struct bin_attribute *attr);
extern int device_schedule_callback(struct device *dev, extern int device_schedule_callback_owner(struct device *dev,
void (*func)(struct device *)); void (*func)(struct device *), struct module *owner);
/* This is a macro to avoid include problems with THIS_MODULE */
#define device_schedule_callback(dev, func) \
device_schedule_callback_owner(dev, func, THIS_MODULE)
/* device resource management */ /* device resource management */
typedef void (*dr_release_t)(struct device *dev, void *res); typedef void (*dr_release_t)(struct device *dev, void *res);
@ -554,7 +571,11 @@ extern const char *dev_driver_string(struct device *dev);
#define dev_dbg(dev, format, arg...) \ #define dev_dbg(dev, format, arg...) \
dev_printk(KERN_DEBUG , dev , format , ## arg) dev_printk(KERN_DEBUG , dev , format , ## arg)
#else #else
#define dev_dbg(dev, format, arg...) do { (void)(dev); } while (0) static inline int __attribute__ ((format (printf, 2, 3)))
dev_dbg(struct device * dev, const char * fmt, ...)
{
return 0;
}
#endif #endif
#define dev_err(dev, format, arg...) \ #define dev_err(dev, format, arg...) \

View File

@ -22,7 +22,6 @@
#include <linux/sysfs.h> #include <linux/sysfs.h>
#include <linux/compiler.h> #include <linux/compiler.h>
#include <linux/spinlock.h> #include <linux/spinlock.h>
#include <linux/rwsem.h>
#include <linux/kref.h> #include <linux/kref.h>
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/wait.h> #include <linux/wait.h>
@ -43,11 +42,9 @@ enum kobject_action {
KOBJ_ADD = (__force kobject_action_t) 0x01, /* exclusive to core */ KOBJ_ADD = (__force kobject_action_t) 0x01, /* exclusive to core */
KOBJ_REMOVE = (__force kobject_action_t) 0x02, /* exclusive to core */ KOBJ_REMOVE = (__force kobject_action_t) 0x02, /* exclusive to core */
KOBJ_CHANGE = (__force kobject_action_t) 0x03, /* device state change */ KOBJ_CHANGE = (__force kobject_action_t) 0x03, /* device state change */
KOBJ_MOUNT = (__force kobject_action_t) 0x04, /* mount event for block devices (broken) */ KOBJ_OFFLINE = (__force kobject_action_t) 0x04, /* device offline */
KOBJ_UMOUNT = (__force kobject_action_t) 0x05, /* umount event for block devices (broken) */ KOBJ_ONLINE = (__force kobject_action_t) 0x05, /* device online */
KOBJ_OFFLINE = (__force kobject_action_t) 0x06, /* device offline */ KOBJ_MOVE = (__force kobject_action_t) 0x06, /* device move */
KOBJ_ONLINE = (__force kobject_action_t) 0x07, /* device online */
KOBJ_MOVE = (__force kobject_action_t) 0x08, /* device move */
}; };
struct kobject { struct kobject {
@ -89,6 +86,8 @@ extern void kobject_unregister(struct kobject *);
extern struct kobject * kobject_get(struct kobject *); extern struct kobject * kobject_get(struct kobject *);
extern void kobject_put(struct kobject *); extern void kobject_put(struct kobject *);
extern struct kobject *kobject_kset_add_dir(struct kset *kset,
struct kobject *, const char *);
extern struct kobject *kobject_add_dir(struct kobject *, const char *); extern struct kobject *kobject_add_dir(struct kobject *, const char *);
extern char * kobject_get_path(struct kobject *, gfp_t); extern char * kobject_get_path(struct kobject *, gfp_t);
@ -175,7 +174,6 @@ extern struct kobject * kset_find_obj(struct kset *, const char *);
struct subsystem { struct subsystem {
struct kset kset; struct kset kset;
struct rw_semaphore rwsem;
}; };
#define decl_subsys(_name,_type,_uevent_ops) \ #define decl_subsys(_name,_type,_uevent_ops) \

View File

@ -82,6 +82,7 @@ extern struct file *nameidata_to_filp(struct nameidata *nd, int flags);
extern void release_open_intent(struct nameidata *); extern void release_open_intent(struct nameidata *);
extern struct dentry * lookup_one_len(const char *, struct dentry *, int); extern struct dentry * lookup_one_len(const char *, struct dentry *, int);
extern struct dentry *lookup_one_len_kern(const char *, struct dentry *, int);
extern int follow_down(struct vfsmount **, struct dentry **); extern int follow_down(struct vfsmount **, struct dentry **);
extern int follow_up(struct vfsmount **, struct dentry **); extern int follow_up(struct vfsmount **, struct dentry **);

View File

@ -361,8 +361,6 @@ struct pci_driver {
struct pci_error_handlers *err_handler; struct pci_error_handlers *err_handler;
struct device_driver driver; struct device_driver driver;
struct pci_dynids dynids; struct pci_dynids dynids;
int multithread_probe;
}; };
#define to_pci_driver(drv) container_of(drv,struct pci_driver, driver) #define to_pci_driver(drv) container_of(drv,struct pci_driver, driver)

View File

@ -166,6 +166,24 @@ extern struct pm_ops *pm_ops;
extern int pm_suspend(suspend_state_t state); extern int pm_suspend(suspend_state_t state);
/**
* arch_suspend_disable_irqs - disable IRQs for suspend
*
* Disables IRQs (in the default case). This is a weak symbol in the common
* code and thus allows architectures to override it if more needs to be
* done. Not called for suspend to disk.
*/
extern void arch_suspend_disable_irqs(void);
/**
* arch_suspend_enable_irqs - enable IRQs after suspend
*
* Enables IRQs (in the default case). This is a weak symbol in the common
* code and thus allows architectures to override it if more needs to be
* done. Not called for suspend to disk.
*/
extern void arch_suspend_enable_irqs(void);
/* /*
* Device power management * Device power management
*/ */
@ -273,6 +291,20 @@ extern void __suspend_report_result(const char *function, void *fn, int ret);
__suspend_report_result(__FUNCTION__, fn, ret); \ __suspend_report_result(__FUNCTION__, fn, ret); \
} while (0) } while (0)
/*
* Platform hook to activate device wakeup capability, if that's not already
* handled by enable_irq_wake() etc.
* Returns zero on success, else negative errno
*/
extern int (*platform_enable_wakeup)(struct device *dev, int is_on);
static inline int call_platform_enable_wakeup(struct device *dev, int is_on)
{
if (platform_enable_wakeup)
return (*platform_enable_wakeup)(dev, is_on);
return 0;
}
#else /* !CONFIG_PM */ #else /* !CONFIG_PM */
static inline int device_suspend(pm_message_t state) static inline int device_suspend(pm_message_t state)
@ -294,6 +326,11 @@ static inline void dpm_runtime_resume(struct device * dev)
#define suspend_report_result(fn, ret) do { } while (0) #define suspend_report_result(fn, ret) do { } while (0)
static inline int call_platform_enable_wakeup(struct device *dev, int is_on)
{
return -EIO;
}
#endif #endif
/* changes to device_may_wakeup take effect on the next pm state change. /* changes to device_may_wakeup take effect on the next pm state change.

View File

@ -80,7 +80,7 @@ struct sysfs_ops {
#ifdef CONFIG_SYSFS #ifdef CONFIG_SYSFS
extern int sysfs_schedule_callback(struct kobject *kobj, extern int sysfs_schedule_callback(struct kobject *kobj,
void (*func)(void *), void *data); void (*func)(void *), void *data, struct module *owner);
extern int __must_check extern int __must_check
sysfs_create_dir(struct kobject *, struct dentry *); sysfs_create_dir(struct kobject *, struct dentry *);
@ -137,7 +137,7 @@ extern int __must_check sysfs_init(void);
#else /* CONFIG_SYSFS */ #else /* CONFIG_SYSFS */
static inline int sysfs_schedule_callback(struct kobject *kobj, static inline int sysfs_schedule_callback(struct kobject *kobj,
void (*func)(void *), void *data) void (*func)(void *), void *data, struct module *owner)
{ {
return -ENOSYS; return -ENOSYS;
} }

View File

@ -1148,8 +1148,10 @@ int mod_sysfs_setup(struct module *mod,
goto out; goto out;
mod->holders_dir = kobject_add_dir(&mod->mkobj.kobj, "holders"); mod->holders_dir = kobject_add_dir(&mod->mkobj.kobj, "holders");
if (!mod->holders_dir) if (!mod->holders_dir) {
err = -ENOMEM;
goto out_unreg; goto out_unreg;
}
err = module_param_sysfs_setup(mod, kparam, num_params); err = module_param_sysfs_setup(mod, kparam, num_params);
if (err) if (err)

View File

@ -111,13 +111,24 @@ static int suspend_prepare(suspend_state_t state)
return error; return error;
} }
/* default implementation */
void __attribute__ ((weak)) arch_suspend_disable_irqs(void)
{
local_irq_disable();
}
/* default implementation */
void __attribute__ ((weak)) arch_suspend_enable_irqs(void)
{
local_irq_enable();
}
int suspend_enter(suspend_state_t state) int suspend_enter(suspend_state_t state)
{ {
int error = 0; int error = 0;
unsigned long flags;
local_irq_save(flags); arch_suspend_disable_irqs();
BUG_ON(!irqs_disabled());
if ((error = device_power_down(PMSG_SUSPEND))) { if ((error = device_power_down(PMSG_SUSPEND))) {
printk(KERN_ERR "Some devices failed to power down\n"); printk(KERN_ERR "Some devices failed to power down\n");
@ -126,7 +137,8 @@ int suspend_enter(suspend_state_t state)
error = pm_ops->enter(state); error = pm_ops->enter(state);
device_power_up(); device_power_up();
Done: Done:
local_irq_restore(flags); arch_suspend_enable_irqs();
BUG_ON(irqs_disabled());
return error; return error;
} }

View File

@ -157,7 +157,7 @@ static void unlink(struct kobject * kobj)
} }
/** /**
* kobject_add - add an object to the hierarchy. * kobject_shadow_add - add an object to the hierarchy.
* @kobj: object. * @kobj: object.
* @shadow_parent: sysfs directory to add to. * @shadow_parent: sysfs directory to add to.
*/ */
@ -174,6 +174,7 @@ int kobject_shadow_add(struct kobject * kobj, struct dentry *shadow_parent)
if (!*kobj->k_name) { if (!*kobj->k_name) {
pr_debug("kobject attempted to be registered with no name!\n"); pr_debug("kobject attempted to be registered with no name!\n");
WARN_ON(1); WARN_ON(1);
kobject_put(kobj);
return -EINVAL; return -EINVAL;
} }
parent = kobject_get(kobj->parent); parent = kobject_get(kobj->parent);
@ -190,8 +191,8 @@ int kobject_shadow_add(struct kobject * kobj, struct dentry *shadow_parent)
list_add_tail(&kobj->entry,&kobj->kset->list); list_add_tail(&kobj->entry,&kobj->kset->list);
spin_unlock(&kobj->kset->list_lock); spin_unlock(&kobj->kset->list_lock);
kobj->parent = parent;
} }
kobj->parent = parent;
error = create_dir(kobj, shadow_parent); error = create_dir(kobj, shadow_parent);
if (error) { if (error) {
@ -311,13 +312,43 @@ EXPORT_SYMBOL(kobject_set_name);
int kobject_rename(struct kobject * kobj, const char *new_name) int kobject_rename(struct kobject * kobj, const char *new_name)
{ {
int error = 0; int error = 0;
const char *devpath = NULL;
char *devpath_string = NULL;
char *envp[2];
kobj = kobject_get(kobj); kobj = kobject_get(kobj);
if (!kobj) if (!kobj)
return -EINVAL; return -EINVAL;
if (!kobj->parent) if (!kobj->parent)
return -EINVAL; return -EINVAL;
devpath = kobject_get_path(kobj, GFP_KERNEL);
if (!devpath) {
error = -ENOMEM;
goto out;
}
devpath_string = kmalloc(strlen(devpath) + 15, GFP_KERNEL);
if (!devpath_string) {
error = -ENOMEM;
goto out;
}
sprintf(devpath_string, "DEVPATH_OLD=%s", devpath);
envp[0] = devpath_string;
envp[1] = NULL;
/* Note : if we want to send the new name alone, not the full path,
* we could probably use kobject_name(kobj); */
error = sysfs_rename_dir(kobj, kobj->parent->dentry, new_name); error = sysfs_rename_dir(kobj, kobj->parent->dentry, new_name);
/* This function is mostly/only used for network interface.
* Some hotplug package track interfaces by their name and
* therefore want to know when the name is changed by the user. */
if (!error)
kobject_uevent_env(kobj, KOBJ_MOVE, envp);
out:
kfree(devpath_string);
kfree(devpath);
kobject_put(kobj); kobject_put(kobj);
return error; return error;
@ -488,13 +519,15 @@ static struct kobj_type dir_ktype = {
}; };
/** /**
* kobject_add_dir - add sub directory of object. * kobject_kset_add_dir - add sub directory of object.
* @kset: kset the directory is belongs to.
* @parent: object in which a directory is created. * @parent: object in which a directory is created.
* @name: directory name. * @name: directory name.
* *
* Add a plain directory object as child of given object. * Add a plain directory object as child of given object.
*/ */
struct kobject *kobject_add_dir(struct kobject *parent, const char *name) struct kobject *kobject_kset_add_dir(struct kset *kset,
struct kobject *parent, const char *name)
{ {
struct kobject *k; struct kobject *k;
int ret; int ret;
@ -506,13 +539,14 @@ struct kobject *kobject_add_dir(struct kobject *parent, const char *name)
if (!k) if (!k)
return NULL; return NULL;
k->kset = kset;
k->parent = parent; k->parent = parent;
k->ktype = &dir_ktype; k->ktype = &dir_ktype;
kobject_set_name(k, name); kobject_set_name(k, name);
ret = kobject_register(k); ret = kobject_register(k);
if (ret < 0) { if (ret < 0) {
printk(KERN_WARNING "kobject_add_dir: " printk(KERN_WARNING "%s: kobject_register error: %d\n",
"kobject_register error: %d\n", ret); __func__, ret);
kobject_del(k); kobject_del(k);
return NULL; return NULL;
} }
@ -520,6 +554,18 @@ struct kobject *kobject_add_dir(struct kobject *parent, const char *name)
return k; return k;
} }
/**
* kobject_add_dir - add sub directory of object.
* @parent: object in which a directory is created.
* @name: directory name.
*
* Add a plain directory object as child of given object.
*/
struct kobject *kobject_add_dir(struct kobject *parent, const char *name)
{
return kobject_kset_add_dir(NULL, parent, name);
}
/** /**
* kset_init - initialize a kset for use * kset_init - initialize a kset for use
* @k: kset * @k: kset
@ -613,7 +659,6 @@ struct kobject * kset_find_obj(struct kset * kset, const char * name)
void subsystem_init(struct subsystem * s) void subsystem_init(struct subsystem * s)
{ {
init_rwsem(&s->rwsem);
kset_init(&s->kset); kset_init(&s->kset);
} }
@ -622,8 +667,7 @@ void subsystem_init(struct subsystem * s)
* @s: the subsystem we're registering. * @s: the subsystem we're registering.
* *
* Once we register the subsystem, we want to make sure that * Once we register the subsystem, we want to make sure that
* the kset points back to this subsystem for correct usage of * the kset points back to this subsystem.
* the rwsem.
*/ */
int subsystem_register(struct subsystem * s) int subsystem_register(struct subsystem * s)

View File

@ -42,10 +42,6 @@ static char *action_to_string(enum kobject_action action)
return "remove"; return "remove";
case KOBJ_CHANGE: case KOBJ_CHANGE:
return "change"; return "change";
case KOBJ_MOUNT:
return "mount";
case KOBJ_UMOUNT:
return "umount";
case KOBJ_OFFLINE: case KOBJ_OFFLINE:
return "offline"; return "offline";
case KOBJ_ONLINE: case KOBJ_ONLINE:
@ -95,10 +91,8 @@ int kobject_uevent_env(struct kobject *kobj, enum kobject_action action,
/* search the kset we belong to */ /* search the kset we belong to */
top_kobj = kobj; top_kobj = kobj;
if (!top_kobj->kset && top_kobj->parent) { while (!top_kobj->kset && top_kobj->parent) {
do { top_kobj = top_kobj->parent;
top_kobj = top_kobj->parent;
} while (!top_kobj->kset && top_kobj->parent);
} }
if (!top_kobj->kset) { if (!top_kobj->kset) {
pr_debug("kobject attempted to send uevent without kset!\n"); pr_debug("kobject attempted to send uevent without kset!\n");
@ -115,6 +109,16 @@ int kobject_uevent_env(struct kobject *kobj, enum kobject_action action,
return 0; return 0;
} }
/* originating subsystem */
if (uevent_ops && uevent_ops->name)
subsystem = uevent_ops->name(kset, kobj);
else
subsystem = kobject_name(&kset->kobj);
if (!subsystem) {
pr_debug("unset subsytem caused the event to drop!\n");
return 0;
}
/* environment index */ /* environment index */
envp = kzalloc(NUM_ENVP * sizeof (char *), GFP_KERNEL); envp = kzalloc(NUM_ENVP * sizeof (char *), GFP_KERNEL);
if (!envp) if (!envp)
@ -134,12 +138,6 @@ int kobject_uevent_env(struct kobject *kobj, enum kobject_action action,
goto exit; goto exit;
} }
/* originating subsystem */
if (uevent_ops && uevent_ops->name)
subsystem = uevent_ops->name(kset, kobj);
else
subsystem = kobject_name(&kset->kobj);
/* event environemnt for helper process only */ /* event environemnt for helper process only */
envp[i++] = "HOME=/"; envp[i++] = "HOME=/";
envp[i++] = "PATH=/sbin:/bin:/usr/sbin:/usr/bin"; envp[i++] = "PATH=/sbin:/bin:/usr/sbin:/usr/bin";

View File

@ -21,6 +21,7 @@
void kref_init(struct kref *kref) void kref_init(struct kref *kref)
{ {
atomic_set(&kref->refcount,1); atomic_set(&kref->refcount,1);
smp_mb();
} }
/** /**
@ -31,6 +32,7 @@ void kref_get(struct kref *kref)
{ {
WARN_ON(!atomic_read(&kref->refcount)); WARN_ON(!atomic_read(&kref->refcount));
atomic_inc(&kref->refcount); atomic_inc(&kref->refcount);
smp_mb__after_atomic_inc();
} }
/** /**

View File

@ -412,20 +412,25 @@ static int netdev_uevent(struct device *d, char **envp,
int num_envp, char *buf, int size) int num_envp, char *buf, int size)
{ {
struct net_device *dev = to_net_dev(d); struct net_device *dev = to_net_dev(d);
int i = 0; int retval, len = 0, i = 0;
int n;
/* pass interface to uevent. */ /* pass interface to uevent. */
envp[i++] = buf; retval = add_uevent_var(envp, num_envp, &i,
n = snprintf(buf, size, "INTERFACE=%s", dev->name) + 1; buf, size, &len,
buf += n; "INTERFACE=%s", dev->name);
size -= n; if (retval)
goto exit;
if ((size <= 0) || (i >= num_envp)) /* pass ifindex to uevent.
return -ENOMEM; * ifindex is useful as it won't change (interface name may change)
* and is what RtNetlink uses natively. */
retval = add_uevent_var(envp, num_envp, &i,
buf, size, &len,
"IFINDEX=%d", dev->ifindex);
exit:
envp[i] = NULL; envp[i] = NULL;
return 0; return retval;
} }
#endif #endif

View File

@ -61,9 +61,9 @@ static int soundbus_uevent(struct device *dev, char **envp, int num_envp,
{ {
struct soundbus_dev * soundbus_dev; struct soundbus_dev * soundbus_dev;
struct of_device * of; struct of_device * of;
char *scratch, *compat, *compat2; char *compat;
int i = 0; int retval = 0, i = 0, length = 0;
int length, cplen, cplen2, seen = 0; int cplen, seen = 0;
if (!dev) if (!dev)
return -ENODEV; return -ENODEV;
@ -75,63 +75,47 @@ static int soundbus_uevent(struct device *dev, char **envp, int num_envp,
of = &soundbus_dev->ofdev; of = &soundbus_dev->ofdev;
/* stuff we want to pass to /sbin/hotplug */ /* stuff we want to pass to /sbin/hotplug */
envp[i++] = scratch = buffer; retval = add_uevent_var(envp, num_envp, &i,
length = scnprintf (scratch, buffer_size, "OF_NAME=%s", of->node->name); buffer, buffer_size, &length,
++length; "OF_NAME=%s", of->node->name);
buffer_size -= length; if (retval)
if ((buffer_size <= 0) || (i >= num_envp)) return retval;
return -ENOMEM;
scratch += length;
envp[i++] = scratch; retval = add_uevent_var(envp, num_envp, &i,
length = scnprintf (scratch, buffer_size, "OF_TYPE=%s", of->node->type); buffer, buffer_size, &length,
++length; "OF_TYPE=%s", of->node->type);
buffer_size -= length; if (retval)
if ((buffer_size <= 0) || (i >= num_envp)) return retval;
return -ENOMEM;
scratch += length;
/* Since the compatible field can contain pretty much anything /* Since the compatible field can contain pretty much anything
* it's not really legal to split it out with commas. We split it * it's not really legal to split it out with commas. We split it
* up using a number of environment variables instead. */ * up using a number of environment variables instead. */
compat = (char *) get_property(of->node, "compatible", &cplen); compat = (char *) get_property(of->node, "compatible", &cplen);
compat2 = compat;
cplen2= cplen;
while (compat && cplen > 0) { while (compat && cplen > 0) {
envp[i++] = scratch; int tmp = length;
length = scnprintf (scratch, buffer_size, retval = add_uevent_var(envp, num_envp, &i,
"OF_COMPATIBLE_%d=%s", seen, compat); buffer, buffer_size, &length,
++length; "OF_COMPATIBLE_%d=%s", seen, compat);
buffer_size -= length; if (retval)
if ((buffer_size <= 0) || (i >= num_envp)) return retval;
return -ENOMEM; compat += length - tmp;
scratch += length; cplen -= length - tmp;
length = strlen (compat) + 1; seen += 1;
compat += length;
cplen -= length;
seen++;
} }
envp[i++] = scratch; retval = add_uevent_var(envp, num_envp, &i,
length = scnprintf (scratch, buffer_size, "OF_COMPATIBLE_N=%d", seen); buffer, buffer_size, &length,
++length; "OF_COMPATIBLE_N=%d", seen);
buffer_size -= length; if (retval)
if ((buffer_size <= 0) || (i >= num_envp)) return retval;
return -ENOMEM; retval = add_uevent_var(envp, num_envp, &i,
scratch += length; buffer, buffer_size, &length,
"MODALIAS=%s", soundbus_dev->modalias);
envp[i++] = scratch;
length = scnprintf (scratch, buffer_size, "MODALIAS=%s",
soundbus_dev->modalias);
buffer_size -= length;
if ((buffer_size <= 0) || (i >= num_envp))
return -ENOMEM;
envp[i] = NULL; envp[i] = NULL;
return 0; return retval;
} }
static int soundbus_device_remove(struct device *dev) static int soundbus_device_remove(struct device *dev)