提交 3965fc36 编写于 作者: V Vladimir Davydov 提交者: Linus Torvalds

slab: clean up kmem_cache_create_memcg() error handling

Currently kmem_cache_create_memcg() backoffs on failure inside
conditionals, without using gotos.  This results in the rollback code
duplication, which makes the function look cumbersome even though on
error we should only free the allocated cache.  Since in the next patch
I am going to add yet another rollback function call on error path
there, let's employ labels instead of conditionals for undoing any
changes on failure to keep things clean.
Signed-off-by: NVladimir Davydov <vdavydov@parallels.com>
Reviewed-by: NPekka Enberg <penberg@kernel.org>
Cc: Michal Hocko <mhocko@suse.cz>
Cc: Glauber Costa <glommer@gmail.com>
Cc: Johannes Weiner <hannes@cmpxchg.org>
Cc: Christoph Lameter <cl@linux.com>
Signed-off-by: NAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: NLinus Torvalds <torvalds@linux-foundation.org>
上级 309381fe
...@@ -171,13 +171,14 @@ kmem_cache_create_memcg(struct mem_cgroup *memcg, const char *name, size_t size, ...@@ -171,13 +171,14 @@ kmem_cache_create_memcg(struct mem_cgroup *memcg, const char *name, size_t size,
struct kmem_cache *parent_cache) struct kmem_cache *parent_cache)
{ {
struct kmem_cache *s = NULL; struct kmem_cache *s = NULL;
int err = 0; int err;
get_online_cpus(); get_online_cpus();
mutex_lock(&slab_mutex); mutex_lock(&slab_mutex);
if (!kmem_cache_sanity_check(memcg, name, size) == 0) err = kmem_cache_sanity_check(memcg, name, size);
goto out_locked; if (err)
goto out_unlock;
/* /*
* Some allocators will constraint the set of valid flags to a subset * Some allocators will constraint the set of valid flags to a subset
...@@ -189,45 +190,38 @@ kmem_cache_create_memcg(struct mem_cgroup *memcg, const char *name, size_t size, ...@@ -189,45 +190,38 @@ kmem_cache_create_memcg(struct mem_cgroup *memcg, const char *name, size_t size,
s = __kmem_cache_alias(memcg, name, size, align, flags, ctor); s = __kmem_cache_alias(memcg, name, size, align, flags, ctor);
if (s) if (s)
goto out_locked; goto out_unlock;
err = -ENOMEM;
s = kmem_cache_zalloc(kmem_cache, GFP_KERNEL); s = kmem_cache_zalloc(kmem_cache, GFP_KERNEL);
if (s) { if (!s)
s->object_size = s->size = size; goto out_unlock;
s->align = calculate_alignment(flags, align, size);
s->ctor = ctor;
if (memcg_register_cache(memcg, s, parent_cache)) { s->object_size = s->size = size;
kmem_cache_free(kmem_cache, s); s->align = calculate_alignment(flags, align, size);
err = -ENOMEM; s->ctor = ctor;
goto out_locked;
}
s->name = kstrdup(name, GFP_KERNEL); s->name = kstrdup(name, GFP_KERNEL);
if (!s->name) { if (!s->name)
kmem_cache_free(kmem_cache, s); goto out_free_cache;
err = -ENOMEM;
goto out_locked;
}
err = __kmem_cache_create(s, flags); err = memcg_register_cache(memcg, s, parent_cache);
if (!err) { if (err)
s->refcount = 1; goto out_free_cache;
list_add(&s->list, &slab_caches);
memcg_cache_list_add(memcg, s); err = __kmem_cache_create(s, flags);
} else { if (err)
kfree(s->name); goto out_free_cache;
kmem_cache_free(kmem_cache, s);
} s->refcount = 1;
} else list_add(&s->list, &slab_caches);
err = -ENOMEM; memcg_cache_list_add(memcg, s);
out_locked: out_unlock:
mutex_unlock(&slab_mutex); mutex_unlock(&slab_mutex);
put_online_cpus(); put_online_cpus();
if (err) { if (err) {
if (flags & SLAB_PANIC) if (flags & SLAB_PANIC)
panic("kmem_cache_create: Failed to create slab '%s'. Error %d\n", panic("kmem_cache_create: Failed to create slab '%s'. Error %d\n",
name, err); name, err);
...@@ -236,11 +230,14 @@ kmem_cache_create_memcg(struct mem_cgroup *memcg, const char *name, size_t size, ...@@ -236,11 +230,14 @@ kmem_cache_create_memcg(struct mem_cgroup *memcg, const char *name, size_t size,
name, err); name, err);
dump_stack(); dump_stack();
} }
return NULL; return NULL;
} }
return s; return s;
out_free_cache:
kfree(s->name);
kmem_cache_free(kmem_cache, s);
goto out_unlock;
} }
struct kmem_cache * struct kmem_cache *
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册