mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-01 10:42:11 +00:00
kobject: fix kset_find_obj() race with concurrent last kobject_put()
Anatol Pomozov identified a race condition that hits module unloading and re-loading. To quote Anatol: "This is a race codition that exists between kset_find_obj() and kobject_put(). kset_find_obj() might return kobject that has refcount equal to 0 if this kobject is freeing by kobject_put() in other thread. Here is timeline for the crash in case if kset_find_obj() searches for an object tht nobody holds and other thread is doing kobject_put() on the same kobject: THREAD A (calls kset_find_obj()) THREAD B (calls kobject_put()) splin_lock() atomic_dec_return(kobj->kref), counter gets zero here ... starts kobject cleanup .... spin_lock() // WAIT thread A in kobj_kset_leave() iterate over kset->list atomic_inc(kobj->kref) (counter becomes 1) spin_unlock() spin_lock() // taken // it does not know that thread A increased counter so it remove obj from list spin_unlock() vfree(module) // frees module object with containing kobj // kobj points to freed memory area!! kobject_put(kobj) // OOPS!!!! The race above happens because module.c tries to use kset_find_obj() when somebody unloads module. The module.c code was introduced in commit 6494a93d55fa" Anatol supplied a patch specific for module.c that worked around the problem by simply not using kset_find_obj() at all, but rather than make a local band-aid, this just fixes kset_find_obj() to be thread-safe using the proper model of refusing the get a new reference if the refcount has already dropped to zero. See examples of this proper refcount handling not only in the kref documentation, but in various other equivalent uses of this pattern by grepping for atomic_inc_not_zero(). [ Side note: the module race does indicate that module loading and unloading is not properly serialized wrt sysfs information using the module mutex. That may require further thought, but this is the correct fix at the kobject layer regardless. ] Reported-analyzed-and-tested-by: Anatol Pomozov <anatol.pomozov@gmail.com> Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Cc: Al Viro <viro@zeniv.linux.org.uk> Cc: stable@vger.kernel.org Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
1de14c3c5c
commit
a49b7e82ca
@ -529,6 +529,13 @@ struct kobject *kobject_get(struct kobject *kobj)
|
|||||||
return kobj;
|
return kobj;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static struct kobject *kobject_get_unless_zero(struct kobject *kobj)
|
||||||
|
{
|
||||||
|
if (!kref_get_unless_zero(&kobj->kref))
|
||||||
|
kobj = NULL;
|
||||||
|
return kobj;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* kobject_cleanup - free kobject resources.
|
* kobject_cleanup - free kobject resources.
|
||||||
* @kobj: object to cleanup
|
* @kobj: object to cleanup
|
||||||
@ -751,7 +758,7 @@ struct kobject *kset_find_obj(struct kset *kset, const char *name)
|
|||||||
|
|
||||||
list_for_each_entry(k, &kset->list, entry) {
|
list_for_each_entry(k, &kset->list, entry) {
|
||||||
if (kobject_name(k) && !strcmp(kobject_name(k), name)) {
|
if (kobject_name(k) && !strcmp(kobject_name(k), name)) {
|
||||||
ret = kobject_get(k);
|
ret = kobject_get_unless_zero(k);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user