mm: memcg/percpu: account extra objcg space to memory cgroups
Similar to slab memory allocator, for each accounted percpu object there is an extra space which is used to store obj_cgroup membership. Charge it too. [akpm@linux-foundation.org: fix layout] Link: https://lkml.kernel.org/r/20211126040606.97836-1-zhengqi.arch@bytedance.com Signed-off-by: Qi Zheng <zhengqi.arch@bytedance.com> Acked-by: Dennis Zhou <dennis@kernel.org> Cc: Tejun Heo <tj@kernel.org> Cc: Christoph Lameter <cl@linux.com> Cc: Muchun Song <songmuchun@bytedance.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
bf181c5825
commit
8c57c07741
@ -113,6 +113,24 @@ static inline int pcpu_chunk_map_bits(struct pcpu_chunk *chunk)
|
|||||||
return pcpu_nr_pages_to_map_bits(chunk->nr_pages);
|
return pcpu_nr_pages_to_map_bits(chunk->nr_pages);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_MEMCG_KMEM
|
||||||
|
/**
|
||||||
|
* pcpu_obj_full_size - helper to calculate size of each accounted object
|
||||||
|
* @size: size of area to allocate in bytes
|
||||||
|
*
|
||||||
|
* For each accounted object there is an extra space which is used to store
|
||||||
|
* obj_cgroup membership. Charge it too.
|
||||||
|
*/
|
||||||
|
static inline size_t pcpu_obj_full_size(size_t size)
|
||||||
|
{
|
||||||
|
size_t extra_size;
|
||||||
|
|
||||||
|
extra_size = size / PCPU_MIN_ALLOC_SIZE * sizeof(struct obj_cgroup *);
|
||||||
|
|
||||||
|
return size * num_possible_cpus() + extra_size;
|
||||||
|
}
|
||||||
|
#endif /* CONFIG_MEMCG_KMEM */
|
||||||
|
|
||||||
#ifdef CONFIG_PERCPU_STATS
|
#ifdef CONFIG_PERCPU_STATS
|
||||||
|
|
||||||
#include <linux/spinlock.h>
|
#include <linux/spinlock.h>
|
||||||
|
10
mm/percpu.c
10
mm/percpu.c
@ -1635,7 +1635,7 @@ static bool pcpu_memcg_pre_alloc_hook(size_t size, gfp_t gfp,
|
|||||||
if (!objcg)
|
if (!objcg)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (obj_cgroup_charge(objcg, gfp, size * num_possible_cpus())) {
|
if (obj_cgroup_charge(objcg, gfp, pcpu_obj_full_size(size))) {
|
||||||
obj_cgroup_put(objcg);
|
obj_cgroup_put(objcg);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1656,10 +1656,10 @@ static void pcpu_memcg_post_alloc_hook(struct obj_cgroup *objcg,
|
|||||||
|
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
mod_memcg_state(obj_cgroup_memcg(objcg), MEMCG_PERCPU_B,
|
mod_memcg_state(obj_cgroup_memcg(objcg), MEMCG_PERCPU_B,
|
||||||
size * num_possible_cpus());
|
pcpu_obj_full_size(size));
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
} else {
|
} else {
|
||||||
obj_cgroup_uncharge(objcg, size * num_possible_cpus());
|
obj_cgroup_uncharge(objcg, pcpu_obj_full_size(size));
|
||||||
obj_cgroup_put(objcg);
|
obj_cgroup_put(objcg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1676,11 +1676,11 @@ static void pcpu_memcg_free_hook(struct pcpu_chunk *chunk, int off, size_t size)
|
|||||||
return;
|
return;
|
||||||
chunk->obj_cgroups[off >> PCPU_MIN_ALLOC_SHIFT] = NULL;
|
chunk->obj_cgroups[off >> PCPU_MIN_ALLOC_SHIFT] = NULL;
|
||||||
|
|
||||||
obj_cgroup_uncharge(objcg, size * num_possible_cpus());
|
obj_cgroup_uncharge(objcg, pcpu_obj_full_size(size));
|
||||||
|
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
mod_memcg_state(obj_cgroup_memcg(objcg), MEMCG_PERCPU_B,
|
mod_memcg_state(obj_cgroup_memcg(objcg), MEMCG_PERCPU_B,
|
||||||
-(size * num_possible_cpus()));
|
-pcpu_obj_full_size(size));
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
|
|
||||||
obj_cgroup_put(objcg);
|
obj_cgroup_put(objcg);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user