Commit 0430f78b authored by Petr Mladek's avatar Petr Mladek Committed by Jiri Kosina

livepatch: Consolidate klp_free functions

The code for freeing livepatch structures is a bit scattered and tricky:

  + direct calls to klp_free_*_limited() and kobject_put() are
    used to release partially initialized objects

  + klp_free_patch() removes the patch from the public list
    and releases all objects except for patch->kobj

  + object_put(&patch->kobj) and the related wait_for_completion()
    are called directly outside klp_mutex; this code is duplicated;

Now, we are going to remove the registration stage to simplify the API
and the code. This would require handling more situations in
klp_enable_patch() error paths.

More importantly, we are going to add a feature called atomic replace.
It will need to dynamically create func and object structures. We will
want to reuse the existing init() and free() functions. This would
create even more error path scenarios.

This patch implements more straightforward free functions:

  + checks kobj_added flag instead of @limit[*]

  + initializes patch->list early so that the check for empty list
    always works

  + The action(s) that has to be done outside klp_mutex are done
    in separate klp_free_patch_finish() function. It waits only
    when patch->kobj was really released via the _start() part.

The patch does not change the existing behavior.

[*] We need our own flag to track that the kobject was successfully
    added to the hierarchy.  Note that kobj.state_initialized only
    indicates that kobject has been initialized, not whether is has
    been added (and needs to be removed on cleanup).
Signed-off-by: default avatarPetr Mladek <pmladek@suse.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Miroslav Benes <mbenes@suse.cz>
Cc: Jessica Yu <jeyu@kernel.org>
Cc: Jiri Kosina <jikos@kernel.org>
Cc: Jason Baron <jbaron@akamai.com>
Acked-by: default avatarMiroslav Benes <mbenes@suse.cz>
Acked-by: default avatarJosh Poimboeuf <jpoimboe@redhat.com>
Signed-off-by: default avatarJiri Kosina <jkosina@suse.cz>
parent 26c3e98e
...@@ -45,6 +45,7 @@ ...@@ -45,6 +45,7 @@
* @stack_node: list node for klp_ops func_stack list * @stack_node: list node for klp_ops func_stack list
* @old_size: size of the old function * @old_size: size of the old function
* @new_size: size of the new function * @new_size: size of the new function
* @kobj_added: @kobj has been added and needs freeing
* @patched: the func has been added to the klp_ops list * @patched: the func has been added to the klp_ops list
* @transition: the func is currently being applied or reverted * @transition: the func is currently being applied or reverted
* *
...@@ -81,6 +82,7 @@ struct klp_func { ...@@ -81,6 +82,7 @@ struct klp_func {
struct kobject kobj; struct kobject kobj;
struct list_head stack_node; struct list_head stack_node;
unsigned long old_size, new_size; unsigned long old_size, new_size;
bool kobj_added;
bool patched; bool patched;
bool transition; bool transition;
}; };
...@@ -117,6 +119,7 @@ struct klp_callbacks { ...@@ -117,6 +119,7 @@ struct klp_callbacks {
* @kobj: kobject for sysfs resources * @kobj: kobject for sysfs resources
* @mod: kernel module associated with the patched object * @mod: kernel module associated with the patched object
* (NULL for vmlinux) * (NULL for vmlinux)
* @kobj_added: @kobj has been added and needs freeing
* @patched: the object's funcs have been added to the klp_ops list * @patched: the object's funcs have been added to the klp_ops list
*/ */
struct klp_object { struct klp_object {
...@@ -128,6 +131,7 @@ struct klp_object { ...@@ -128,6 +131,7 @@ struct klp_object {
/* internal */ /* internal */
struct kobject kobj; struct kobject kobj;
struct module *mod; struct module *mod;
bool kobj_added;
bool patched; bool patched;
}; };
...@@ -137,6 +141,7 @@ struct klp_object { ...@@ -137,6 +141,7 @@ struct klp_object {
* @objs: object entries for kernel objects to be patched * @objs: object entries for kernel objects to be patched
* @list: list node for global list of registered patches * @list: list node for global list of registered patches
* @kobj: kobject for sysfs resources * @kobj: kobject for sysfs resources
* @kobj_added: @kobj has been added and needs freeing
* @enabled: the patch is enabled (but operation may be incomplete) * @enabled: the patch is enabled (but operation may be incomplete)
* @finish: for waiting till it is safe to remove the patch module * @finish: for waiting till it is safe to remove the patch module
*/ */
...@@ -148,6 +153,7 @@ struct klp_patch { ...@@ -148,6 +153,7 @@ struct klp_patch {
/* internal */ /* internal */
struct list_head list; struct list_head list;
struct kobject kobj; struct kobject kobj;
bool kobj_added;
bool enabled; bool enabled;
struct completion finish; struct completion finish;
}; };
......
...@@ -465,17 +465,15 @@ static struct kobj_type klp_ktype_func = { ...@@ -465,17 +465,15 @@ static struct kobj_type klp_ktype_func = {
.sysfs_ops = &kobj_sysfs_ops, .sysfs_ops = &kobj_sysfs_ops,
}; };
/* static void klp_free_funcs(struct klp_object *obj)
* Free all functions' kobjects in the array up to some limit. When limit is
* NULL, all kobjects are freed.
*/
static void klp_free_funcs_limited(struct klp_object *obj,
struct klp_func *limit)
{ {
struct klp_func *func; struct klp_func *func;
for (func = obj->funcs; func->old_name && func != limit; func++) klp_for_each_func(obj, func) {
kobject_put(&func->kobj); /* Might be called from klp_init_patch() error path. */
if (func->kobj_added)
kobject_put(&func->kobj);
}
} }
/* Clean up when a patched object is unloaded */ /* Clean up when a patched object is unloaded */
...@@ -489,30 +487,60 @@ static void klp_free_object_loaded(struct klp_object *obj) ...@@ -489,30 +487,60 @@ static void klp_free_object_loaded(struct klp_object *obj)
func->old_func = NULL; func->old_func = NULL;
} }
/* static void klp_free_objects(struct klp_patch *patch)
* Free all objects' kobjects in the array up to some limit. When limit is
* NULL, all kobjects are freed.
*/
static void klp_free_objects_limited(struct klp_patch *patch,
struct klp_object *limit)
{ {
struct klp_object *obj; struct klp_object *obj;
for (obj = patch->objs; obj->funcs && obj != limit; obj++) { klp_for_each_object(patch, obj) {
klp_free_funcs_limited(obj, NULL); klp_free_funcs(obj);
kobject_put(&obj->kobj);
/* Might be called from klp_init_patch() error path. */
if (obj->kobj_added)
kobject_put(&obj->kobj);
} }
} }
static void klp_free_patch(struct klp_patch *patch) /*
* This function implements the free operations that can be called safely
* under klp_mutex.
*
* The operation must be completed by calling klp_free_patch_finish()
* outside klp_mutex.
*/
static void klp_free_patch_start(struct klp_patch *patch)
{ {
klp_free_objects_limited(patch, NULL);
if (!list_empty(&patch->list)) if (!list_empty(&patch->list))
list_del(&patch->list); list_del(&patch->list);
klp_free_objects(patch);
}
/*
* This function implements the free part that must be called outside
* klp_mutex.
*
* It must be called after klp_free_patch_start(). And it has to be
* the last function accessing the livepatch structures when the patch
* gets disabled.
*/
static void klp_free_patch_finish(struct klp_patch *patch)
{
/*
* Avoid deadlock with enabled_store() sysfs callback by
* calling this outside klp_mutex. It is safe because
* this is called when the patch gets disabled and it
* cannot get enabled again.
*/
if (patch->kobj_added) {
kobject_put(&patch->kobj);
wait_for_completion(&patch->finish);
}
} }
static int klp_init_func(struct klp_object *obj, struct klp_func *func) static int klp_init_func(struct klp_object *obj, struct klp_func *func)
{ {
int ret;
if (!func->old_name || !func->new_func) if (!func->old_name || !func->new_func)
return -EINVAL; return -EINVAL;
...@@ -528,9 +556,13 @@ static int klp_init_func(struct klp_object *obj, struct klp_func *func) ...@@ -528,9 +556,13 @@ static int klp_init_func(struct klp_object *obj, struct klp_func *func)
* object. If the user selects 0 for old_sympos, then 1 will be used * object. If the user selects 0 for old_sympos, then 1 will be used
* since a unique symbol will be the first occurrence. * since a unique symbol will be the first occurrence.
*/ */
return kobject_init_and_add(&func->kobj, &klp_ktype_func, ret = kobject_init_and_add(&func->kobj, &klp_ktype_func,
&obj->kobj, "%s,%lu", func->old_name, &obj->kobj, "%s,%lu", func->old_name,
func->old_sympos ? func->old_sympos : 1); func->old_sympos ? func->old_sympos : 1);
if (!ret)
func->kobj_added = true;
return ret;
} }
/* Arches may override this to finish any remaining arch-specific tasks */ /* Arches may override this to finish any remaining arch-specific tasks */
...@@ -589,9 +621,6 @@ static int klp_init_object(struct klp_patch *patch, struct klp_object *obj) ...@@ -589,9 +621,6 @@ static int klp_init_object(struct klp_patch *patch, struct klp_object *obj)
int ret; int ret;
const char *name; const char *name;
if (!obj->funcs)
return -EINVAL;
if (klp_is_module(obj) && strlen(obj->name) >= MODULE_NAME_LEN) if (klp_is_module(obj) && strlen(obj->name) >= MODULE_NAME_LEN)
return -EINVAL; return -EINVAL;
...@@ -605,46 +634,66 @@ static int klp_init_object(struct klp_patch *patch, struct klp_object *obj) ...@@ -605,46 +634,66 @@ static int klp_init_object(struct klp_patch *patch, struct klp_object *obj)
&patch->kobj, "%s", name); &patch->kobj, "%s", name);
if (ret) if (ret)
return ret; return ret;
obj->kobj_added = true;
klp_for_each_func(obj, func) { klp_for_each_func(obj, func) {
ret = klp_init_func(obj, func); ret = klp_init_func(obj, func);
if (ret) if (ret)
goto free; return ret;
} }
if (klp_is_object_loaded(obj)) { if (klp_is_object_loaded(obj))
ret = klp_init_object_loaded(patch, obj); ret = klp_init_object_loaded(patch, obj);
if (ret)
goto free;
}
return 0;
free:
klp_free_funcs_limited(obj, func);
kobject_put(&obj->kobj);
return ret; return ret;
} }
static int klp_init_patch(struct klp_patch *patch) static int klp_init_patch_early(struct klp_patch *patch)
{ {
struct klp_object *obj; struct klp_object *obj;
int ret; struct klp_func *func;
if (!patch->objs) if (!patch->objs)
return -EINVAL; return -EINVAL;
mutex_lock(&klp_mutex); INIT_LIST_HEAD(&patch->list);
patch->kobj_added = false;
patch->enabled = false; patch->enabled = false;
init_completion(&patch->finish); init_completion(&patch->finish);
klp_for_each_object(patch, obj) {
if (!obj->funcs)
return -EINVAL;
obj->kobj_added = false;
klp_for_each_func(obj, func)
func->kobj_added = false;
}
return 0;
}
static int klp_init_patch(struct klp_patch *patch)
{
struct klp_object *obj;
int ret;
mutex_lock(&klp_mutex);
ret = klp_init_patch_early(patch);
if (ret) {
mutex_unlock(&klp_mutex);
return ret;
}
ret = kobject_init_and_add(&patch->kobj, &klp_ktype_patch, ret = kobject_init_and_add(&patch->kobj, &klp_ktype_patch,
klp_root_kobj, "%s", patch->mod->name); klp_root_kobj, "%s", patch->mod->name);
if (ret) { if (ret) {
mutex_unlock(&klp_mutex); mutex_unlock(&klp_mutex);
return ret; return ret;
} }
patch->kobj_added = true;
klp_for_each_object(patch, obj) { klp_for_each_object(patch, obj) {
ret = klp_init_object(patch, obj); ret = klp_init_object(patch, obj);
...@@ -659,12 +708,11 @@ static int klp_init_patch(struct klp_patch *patch) ...@@ -659,12 +708,11 @@ static int klp_init_patch(struct klp_patch *patch)
return 0; return 0;
free: free:
klp_free_objects_limited(patch, obj); klp_free_patch_start(patch);
mutex_unlock(&klp_mutex); mutex_unlock(&klp_mutex);
kobject_put(&patch->kobj); klp_free_patch_finish(patch);
wait_for_completion(&patch->finish);
return ret; return ret;
} }
...@@ -693,12 +741,11 @@ int klp_unregister_patch(struct klp_patch *patch) ...@@ -693,12 +741,11 @@ int klp_unregister_patch(struct klp_patch *patch)
goto err; goto err;
} }
klp_free_patch(patch); klp_free_patch_start(patch);
mutex_unlock(&klp_mutex); mutex_unlock(&klp_mutex);
kobject_put(&patch->kobj); klp_free_patch_finish(patch);
wait_for_completion(&patch->finish);
return 0; return 0;
err: err:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment