mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
Merge git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/driver-2.6
* git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/driver-2.6: driver core: kmalloc() failure check in driver_probe_device Driver core: bus: remove indentation level Driver core: Don't ignore error returns from probing Driver core: Don't leak 'old_class_name' in drivers/base/core.c::device_rename() driver core fixes: sysfs_create_group() retval in topology.c driver core fixes: device_create_file() retval check in dmapool.c driver core fixes: device_add() cleanup on error driver core fixes: bus_add_device() cleanup on error driver core fixes: bus_add_attrs() retval check driver core fixes: sysfs_create_link() retval check in class.c sysfs: update obsolete comment in sysfs_update_file sysfs: remove duplicated dput in sysfs_update_file HOWTO: bug report addition Fix dev_printk() is now GPL-only Driver core: plug device probe memory leak Documentation: feature-removal-schedule typo
This commit is contained in:
commit
65740356cc
9 changed files with 121 additions and 63 deletions
|
@ -395,6 +395,26 @@ bugme-janitor mailing list (every change in the bugzilla is mailed here)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Managing bug reports
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
One of the best ways to put into practice your hacking skills is by fixing
|
||||||
|
bugs reported by other people. Not only you will help to make the kernel
|
||||||
|
more stable, you'll learn to fix real world problems and you will improve
|
||||||
|
your skills, and other developers will be aware of your presence. Fixing
|
||||||
|
bugs is one of the best ways to get merits among other developers, because
|
||||||
|
not many people like wasting time fixing other people's bugs.
|
||||||
|
|
||||||
|
To work in the already reported bug reports, go to http://bugzilla.kernel.org.
|
||||||
|
If you want to be advised of the future bug reports, you can subscribe to the
|
||||||
|
bugme-new mailing list (only new bug reports are mailed here) or to the
|
||||||
|
bugme-janitor mailing list (every change in the bugzilla is mailed here)
|
||||||
|
|
||||||
|
http://lists.osdl.org/mailman/listinfo/bugme-new
|
||||||
|
http://lists.osdl.org/mailman/listinfo/bugme-janitors
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Mailing lists
|
Mailing lists
|
||||||
-------------
|
-------------
|
||||||
|
|
||||||
|
|
|
@ -255,7 +255,7 @@ Who: Stephen Hemminger <shemminger@osdl.org>
|
||||||
|
|
||||||
|
|
||||||
What: PHYSDEVPATH, PHYSDEVBUS, PHYSDEVDRIVER in the uevent environment
|
What: PHYSDEVPATH, PHYSDEVBUS, PHYSDEVDRIVER in the uevent environment
|
||||||
When: Oktober 2008
|
When: October 2008
|
||||||
Why: The stacking of class devices makes these values misleading and
|
Why: The stacking of class devices makes these values misleading and
|
||||||
inconsistent.
|
inconsistent.
|
||||||
Class devices should not carry any of these properties, and bus
|
Class devices should not carry any of these properties, and bus
|
||||||
|
|
|
@ -372,19 +372,30 @@ int bus_add_device(struct device * dev)
|
||||||
pr_debug("bus %s: add device %s\n", bus->name, dev->bus_id);
|
pr_debug("bus %s: add device %s\n", bus->name, dev->bus_id);
|
||||||
error = device_add_attrs(bus, dev);
|
error = device_add_attrs(bus, dev);
|
||||||
if (error)
|
if (error)
|
||||||
goto out;
|
goto out_put;
|
||||||
error = sysfs_create_link(&bus->devices.kobj,
|
error = sysfs_create_link(&bus->devices.kobj,
|
||||||
&dev->kobj, dev->bus_id);
|
&dev->kobj, dev->bus_id);
|
||||||
if (error)
|
if (error)
|
||||||
goto out;
|
goto out_id;
|
||||||
error = sysfs_create_link(&dev->kobj,
|
error = sysfs_create_link(&dev->kobj,
|
||||||
&dev->bus->subsys.kset.kobj, "subsystem");
|
&dev->bus->subsys.kset.kobj, "subsystem");
|
||||||
if (error)
|
if (error)
|
||||||
goto out;
|
goto out_subsys;
|
||||||
error = sysfs_create_link(&dev->kobj,
|
error = sysfs_create_link(&dev->kobj,
|
||||||
&dev->bus->subsys.kset.kobj, "bus");
|
&dev->bus->subsys.kset.kobj, "bus");
|
||||||
|
if (error)
|
||||||
|
goto out_deprecated;
|
||||||
}
|
}
|
||||||
out:
|
return 0;
|
||||||
|
|
||||||
|
out_deprecated:
|
||||||
|
sysfs_remove_link(&dev->kobj, "subsystem");
|
||||||
|
out_subsys:
|
||||||
|
sysfs_remove_link(&bus->devices.kobj, dev->bus_id);
|
||||||
|
out_id:
|
||||||
|
device_remove_attrs(bus, dev);
|
||||||
|
out_put:
|
||||||
|
put_bus(dev->bus);
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -428,8 +439,10 @@ void bus_remove_device(struct device * dev)
|
||||||
sysfs_remove_link(&dev->kobj, "bus");
|
sysfs_remove_link(&dev->kobj, "bus");
|
||||||
sysfs_remove_link(&dev->bus->devices.kobj, dev->bus_id);
|
sysfs_remove_link(&dev->bus->devices.kobj, dev->bus_id);
|
||||||
device_remove_attrs(dev->bus, dev);
|
device_remove_attrs(dev->bus, dev);
|
||||||
dev->is_registered = 0;
|
if (dev->is_registered) {
|
||||||
klist_del(&dev->knode_bus);
|
dev->is_registered = 0;
|
||||||
|
klist_del(&dev->knode_bus);
|
||||||
|
}
|
||||||
pr_debug("bus %s: remove device %s\n", dev->bus->name, dev->bus_id);
|
pr_debug("bus %s: remove device %s\n", dev->bus->name, dev->bus_id);
|
||||||
device_release_driver(dev);
|
device_release_driver(dev);
|
||||||
put_bus(dev->bus);
|
put_bus(dev->bus);
|
||||||
|
@ -505,34 +518,36 @@ int bus_add_driver(struct device_driver *drv)
|
||||||
struct bus_type * bus = get_bus(drv->bus);
|
struct bus_type * bus = get_bus(drv->bus);
|
||||||
int error = 0;
|
int error = 0;
|
||||||
|
|
||||||
if (bus) {
|
if (!bus)
|
||||||
pr_debug("bus %s: add driver %s\n", bus->name, drv->name);
|
return 0;
|
||||||
error = kobject_set_name(&drv->kobj, "%s", drv->name);
|
|
||||||
if (error)
|
|
||||||
goto out_put_bus;
|
|
||||||
drv->kobj.kset = &bus->drivers;
|
|
||||||
if ((error = kobject_register(&drv->kobj)))
|
|
||||||
goto out_put_bus;
|
|
||||||
|
|
||||||
error = driver_attach(drv);
|
pr_debug("bus %s: add driver %s\n", bus->name, drv->name);
|
||||||
if (error)
|
error = kobject_set_name(&drv->kobj, "%s", drv->name);
|
||||||
goto out_unregister;
|
if (error)
|
||||||
klist_add_tail(&drv->knode_bus, &bus->klist_drivers);
|
goto out_put_bus;
|
||||||
module_add_driver(drv->owner, drv);
|
drv->kobj.kset = &bus->drivers;
|
||||||
|
if ((error = kobject_register(&drv->kobj)))
|
||||||
|
goto out_put_bus;
|
||||||
|
|
||||||
error = driver_add_attrs(bus, drv);
|
error = driver_attach(drv);
|
||||||
if (error) {
|
if (error)
|
||||||
/* How the hell do we get out of this pickle? Give up */
|
goto out_unregister;
|
||||||
printk(KERN_ERR "%s: driver_add_attrs(%s) failed\n",
|
klist_add_tail(&drv->knode_bus, &bus->klist_drivers);
|
||||||
__FUNCTION__, drv->name);
|
module_add_driver(drv->owner, drv);
|
||||||
}
|
|
||||||
error = add_bind_files(drv);
|
error = driver_add_attrs(bus, drv);
|
||||||
if (error) {
|
if (error) {
|
||||||
/* Ditto */
|
/* How the hell do we get out of this pickle? Give up */
|
||||||
printk(KERN_ERR "%s: add_bind_files(%s) failed\n",
|
printk(KERN_ERR "%s: driver_add_attrs(%s) failed\n",
|
||||||
__FUNCTION__, drv->name);
|
__FUNCTION__, drv->name);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
error = add_bind_files(drv);
|
||||||
|
if (error) {
|
||||||
|
/* Ditto */
|
||||||
|
printk(KERN_ERR "%s: add_bind_files(%s) failed\n",
|
||||||
|
__FUNCTION__, drv->name);
|
||||||
|
}
|
||||||
|
|
||||||
return error;
|
return error;
|
||||||
out_unregister:
|
out_unregister:
|
||||||
kobject_unregister(&drv->kobj);
|
kobject_unregister(&drv->kobj);
|
||||||
|
@ -552,16 +567,17 @@ out_put_bus:
|
||||||
|
|
||||||
void bus_remove_driver(struct device_driver * drv)
|
void bus_remove_driver(struct device_driver * drv)
|
||||||
{
|
{
|
||||||
if (drv->bus) {
|
if (!drv->bus)
|
||||||
remove_bind_files(drv);
|
return;
|
||||||
driver_remove_attrs(drv->bus, drv);
|
|
||||||
klist_remove(&drv->knode_bus);
|
remove_bind_files(drv);
|
||||||
pr_debug("bus %s: remove driver %s\n", drv->bus->name, drv->name);
|
driver_remove_attrs(drv->bus, drv);
|
||||||
driver_detach(drv);
|
klist_remove(&drv->knode_bus);
|
||||||
module_remove_driver(drv);
|
pr_debug("bus %s: remove driver %s\n", drv->bus->name, drv->name);
|
||||||
kobject_unregister(&drv->kobj);
|
driver_detach(drv);
|
||||||
put_bus(drv->bus);
|
module_remove_driver(drv);
|
||||||
}
|
kobject_unregister(&drv->kobj);
|
||||||
|
put_bus(drv->bus);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -732,11 +748,15 @@ 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_add_attrs(bus);
|
retval = bus_add_attrs(bus);
|
||||||
|
if (retval)
|
||||||
|
goto bus_attrs_fail;
|
||||||
|
|
||||||
pr_debug("bus type '%s' registered\n", bus->name);
|
pr_debug("bus type '%s' registered\n", bus->name);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
bus_attrs_fail:
|
||||||
|
kset_unregister(&bus->drivers);
|
||||||
bus_drivers_fail:
|
bus_drivers_fail:
|
||||||
kset_unregister(&bus->devices);
|
kset_unregister(&bus->devices);
|
||||||
bus_devices_fail:
|
bus_devices_fail:
|
||||||
|
|
|
@ -562,7 +562,10 @@ int class_device_add(struct class_device *class_dev)
|
||||||
goto out2;
|
goto out2;
|
||||||
|
|
||||||
/* add the needed attributes to this device */
|
/* add the needed attributes to this device */
|
||||||
sysfs_create_link(&class_dev->kobj, &parent_class->subsys.kset.kobj, "subsystem");
|
error = sysfs_create_link(&class_dev->kobj,
|
||||||
|
&parent_class->subsys.kset.kobj, "subsystem");
|
||||||
|
if (error)
|
||||||
|
goto out3;
|
||||||
class_dev->uevent_attr.attr.name = "uevent";
|
class_dev->uevent_attr.attr.name = "uevent";
|
||||||
class_dev->uevent_attr.attr.mode = S_IWUSR;
|
class_dev->uevent_attr.attr.mode = S_IWUSR;
|
||||||
class_dev->uevent_attr.attr.owner = parent_class->owner;
|
class_dev->uevent_attr.attr.owner = parent_class->owner;
|
||||||
|
|
|
@ -44,7 +44,7 @@ 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 : "");
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(dev_driver_string);
|
EXPORT_SYMBOL(dev_driver_string);
|
||||||
|
|
||||||
#define to_dev(obj) container_of(obj, struct device, kobj)
|
#define to_dev(obj) container_of(obj, struct device, kobj)
|
||||||
#define to_dev_attr(_attr) container_of(_attr, struct device_attribute, attr)
|
#define to_dev_attr(_attr) container_of(_attr, struct device_attribute, attr)
|
||||||
|
@ -433,14 +433,16 @@ int device_add(struct device *dev)
|
||||||
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;
|
||||||
device_create_file(dev, &dev->uevent_attr);
|
error = device_create_file(dev, &dev->uevent_attr);
|
||||||
|
if (error)
|
||||||
|
goto attrError;
|
||||||
|
|
||||||
if (MAJOR(dev->devt)) {
|
if (MAJOR(dev->devt)) {
|
||||||
struct device_attribute *attr;
|
struct device_attribute *attr;
|
||||||
attr = kzalloc(sizeof(*attr), GFP_KERNEL);
|
attr = kzalloc(sizeof(*attr), GFP_KERNEL);
|
||||||
if (!attr) {
|
if (!attr) {
|
||||||
error = -ENOMEM;
|
error = -ENOMEM;
|
||||||
goto PMError;
|
goto ueventattrError;
|
||||||
}
|
}
|
||||||
attr->attr.name = "dev";
|
attr->attr.name = "dev";
|
||||||
attr->attr.mode = S_IRUGO;
|
attr->attr.mode = S_IRUGO;
|
||||||
|
@ -450,7 +452,7 @@ int device_add(struct device *dev)
|
||||||
error = device_create_file(dev, attr);
|
error = device_create_file(dev, attr);
|
||||||
if (error) {
|
if (error) {
|
||||||
kfree(attr);
|
kfree(attr);
|
||||||
goto attrError;
|
goto ueventattrError;
|
||||||
}
|
}
|
||||||
|
|
||||||
dev->devt_attr = attr;
|
dev->devt_attr = attr;
|
||||||
|
@ -477,7 +479,8 @@ int device_add(struct device *dev)
|
||||||
if ((error = bus_add_device(dev)))
|
if ((error = bus_add_device(dev)))
|
||||||
goto BusError;
|
goto BusError;
|
||||||
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);
|
||||||
|
|
||||||
|
@ -496,6 +499,8 @@ int device_add(struct device *dev)
|
||||||
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:
|
||||||
|
@ -507,6 +512,8 @@ int device_add(struct device *dev)
|
||||||
device_remove_file(dev, dev->devt_attr);
|
device_remove_file(dev, dev->devt_attr);
|
||||||
kfree(dev->devt_attr);
|
kfree(dev->devt_attr);
|
||||||
}
|
}
|
||||||
|
ueventattrError:
|
||||||
|
device_remove_file(dev, &dev->uevent_attr);
|
||||||
attrError:
|
attrError:
|
||||||
kobject_uevent(&dev->kobj, KOBJ_REMOVE);
|
kobject_uevent(&dev->kobj, KOBJ_REMOVE);
|
||||||
kobject_del(&dev->kobj);
|
kobject_del(&dev->kobj);
|
||||||
|
@ -805,8 +812,10 @@ int device_rename(struct device *dev, char *new_name)
|
||||||
|
|
||||||
if (dev->class) {
|
if (dev->class) {
|
||||||
old_symlink_name = kmalloc(BUS_ID_SIZE, GFP_KERNEL);
|
old_symlink_name = kmalloc(BUS_ID_SIZE, GFP_KERNEL);
|
||||||
if (!old_symlink_name)
|
if (!old_symlink_name) {
|
||||||
return -ENOMEM;
|
error = -ENOMEM;
|
||||||
|
goto out_free_old_class;
|
||||||
|
}
|
||||||
strlcpy(old_symlink_name, dev->bus_id, BUS_ID_SIZE);
|
strlcpy(old_symlink_name, dev->bus_id, BUS_ID_SIZE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -830,9 +839,10 @@ int device_rename(struct device *dev, char *new_name)
|
||||||
}
|
}
|
||||||
put_device(dev);
|
put_device(dev);
|
||||||
|
|
||||||
kfree(old_class_name);
|
|
||||||
kfree(new_class_name);
|
kfree(new_class_name);
|
||||||
kfree(old_symlink_name);
|
kfree(old_symlink_name);
|
||||||
|
out_free_old_class:
|
||||||
|
kfree(old_class_name);
|
||||||
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
|
@ -171,6 +171,8 @@ int driver_probe_device(struct device_driver * drv, struct device * dev)
|
||||||
drv->bus->name, dev->bus_id, drv->name);
|
drv->bus->name, dev->bus_id, drv->name);
|
||||||
|
|
||||||
data = kmalloc(sizeof(*data), GFP_KERNEL);
|
data = kmalloc(sizeof(*data), GFP_KERNEL);
|
||||||
|
if (!data)
|
||||||
|
return -ENOMEM;
|
||||||
data->drv = drv;
|
data->drv = drv;
|
||||||
data->dev = dev;
|
data->dev = dev;
|
||||||
|
|
||||||
|
@ -178,7 +180,7 @@ int driver_probe_device(struct device_driver * drv, struct device * dev)
|
||||||
probe_task = kthread_run(really_probe, data,
|
probe_task = kthread_run(really_probe, data,
|
||||||
"probe-%s", dev->bus_id);
|
"probe-%s", dev->bus_id);
|
||||||
if (IS_ERR(probe_task))
|
if (IS_ERR(probe_task))
|
||||||
ret = PTR_ERR(probe_task);
|
ret = really_probe(data);
|
||||||
} else
|
} else
|
||||||
ret = really_probe(data);
|
ret = really_probe(data);
|
||||||
|
|
||||||
|
|
|
@ -141,11 +141,20 @@ dma_pool_create (const char *name, struct device *dev,
|
||||||
init_waitqueue_head (&retval->waitq);
|
init_waitqueue_head (&retval->waitq);
|
||||||
|
|
||||||
if (dev) {
|
if (dev) {
|
||||||
|
int ret;
|
||||||
|
|
||||||
down (&pools_lock);
|
down (&pools_lock);
|
||||||
if (list_empty (&dev->dma_pools))
|
if (list_empty (&dev->dma_pools))
|
||||||
device_create_file (dev, &dev_attr_pools);
|
ret = device_create_file (dev, &dev_attr_pools);
|
||||||
|
else
|
||||||
|
ret = 0;
|
||||||
/* note: not currently insisting "name" be unique */
|
/* note: not currently insisting "name" be unique */
|
||||||
list_add (&retval->pools, &dev->dma_pools);
|
if (!ret)
|
||||||
|
list_add (&retval->pools, &dev->dma_pools);
|
||||||
|
else {
|
||||||
|
kfree(retval);
|
||||||
|
retval = NULL;
|
||||||
|
}
|
||||||
up (&pools_lock);
|
up (&pools_lock);
|
||||||
} else
|
} else
|
||||||
INIT_LIST_HEAD (&retval->pools);
|
INIT_LIST_HEAD (&retval->pools);
|
||||||
|
|
|
@ -97,8 +97,7 @@ static struct attribute_group topology_attr_group = {
|
||||||
/* Add/Remove cpu_topology interface for CPU device */
|
/* Add/Remove cpu_topology interface for CPU device */
|
||||||
static int __cpuinit topology_add_dev(struct sys_device * sys_dev)
|
static int __cpuinit topology_add_dev(struct sys_device * sys_dev)
|
||||||
{
|
{
|
||||||
sysfs_create_group(&sys_dev->kobj, &topology_attr_group);
|
return sysfs_create_group(&sys_dev->kobj, &topology_attr_group);
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int __cpuinit topology_remove_dev(struct sys_device * sys_dev)
|
static int __cpuinit topology_remove_dev(struct sys_device * sys_dev)
|
||||||
|
|
|
@ -483,17 +483,12 @@ int sysfs_update_file(struct kobject * kobj, const struct attribute * attr)
|
||||||
(victim->d_parent->d_inode == dir->d_inode)) {
|
(victim->d_parent->d_inode == dir->d_inode)) {
|
||||||
victim->d_inode->i_mtime = CURRENT_TIME;
|
victim->d_inode->i_mtime = CURRENT_TIME;
|
||||||
fsnotify_modify(victim);
|
fsnotify_modify(victim);
|
||||||
|
|
||||||
/**
|
|
||||||
* Drop reference from initial sysfs_get_dentry().
|
|
||||||
*/
|
|
||||||
dput(victim);
|
|
||||||
res = 0;
|
res = 0;
|
||||||
} else
|
} else
|
||||||
d_drop(victim);
|
d_drop(victim);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Drop the reference acquired from sysfs_get_dentry() above.
|
* Drop the reference acquired from lookup_one_len() above.
|
||||||
*/
|
*/
|
||||||
dput(victim);
|
dput(victim);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue