mm: memcontrol: introduce memcg_reparent_ops
authorMuchun Song <songmuchun@bytedance.com>
Tue, 21 Jun 2022 12:56:55 +0000 (20:56 +0800)
committerLiam R. Howlett <Liam.Howlett@oracle.com>
Wed, 20 Jul 2022 00:15:11 +0000 (20:15 -0400)
In the previous patch, we know how to make the lruvec lock safe when LRU
pages are reparented.  We should do something like following.

    memcg_reparent_objcgs(memcg)
        1) lock
        // lruvec belongs to memcg and lruvec_parent belongs to parent memcg.
        spin_lock(&lruvec->lru_lock);
        spin_lock(&lruvec_parent->lru_lock);

        2) relocate from current memcg to its parent
        // Move all the pages from the lruvec list to the parent lruvec list.

        3) unlock
        spin_unlock(&lruvec_parent->lru_lock);
        spin_unlock(&lruvec->lru_lock);

Apart from the page lruvec lock, the deferred split queue lock (THP only)
also needs to do something similar.  So we extract the necessary three
steps in the memcg_reparent_objcgs().

    memcg_reparent_objcgs(memcg)
        1) lock
        memcg_reparent_ops->lock(memcg, parent);

        2) relocate
        memcg_reparent_ops->relocate(memcg, reparent);

        3) unlock
        memcg_reparent_ops->unlock(memcg, reparent);

Now there are two different locks (e.g.  lruvec lock and deferred split
queue lock) need to use this infrastructure.  In the next patch, we will
use those APIs to make those locks safe when the LRU pages reparented.

Link: https://lkml.kernel.org/r/20220621125658.64935-9-songmuchun@bytedance.com
Signed-off-by: Muchun Song <songmuchun@bytedance.com>
Cc: Johannes Weiner <hannes@cmpxchg.org>
Cc: Michal Hocko <mhocko@kernel.org>
Cc: Michal Koutný <mkoutny@suse.com>
Cc: Roman Gushchin <roman.gushchin@linux.dev>
Cc: Shakeel Butt <shakeelb@google.com>
Cc: Waiman Long <longman@redhat.com>
Cc: Xiongchun Duan <duanxiongchun@bytedance.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
include/linux/memcontrol.h
mm/memcontrol.c

index a8bd4bb3950236d0060be02396e62cdac7f56c1e..63dbdef60cbdae7591d6df57c9366a391529fadd 100644 (file)
@@ -353,6 +353,26 @@ struct mem_cgroup {
        struct mem_cgroup_per_node *nodeinfo[];
 };
 
+struct memcg_reparent_ops {
+       /*
+        * Note that interrupt is disabled before calling those callbacks,
+        * so the interrupt should remain disabled when leaving those callbacks.
+        */
+       void (*lock)(struct mem_cgroup *src, struct mem_cgroup *dst);
+       void (*relocate)(struct mem_cgroup *src, struct mem_cgroup *dst);
+       void (*unlock)(struct mem_cgroup *src, struct mem_cgroup *dst);
+};
+
+#define DEFINE_MEMCG_REPARENT_OPS(name)                                        \
+       const struct memcg_reparent_ops memcg_##name##_reparent_ops = { \
+               .lock           = name##_reparent_lock,                 \
+               .relocate       = name##_reparent_relocate,             \
+               .unlock         = name##_reparent_unlock,               \
+       }
+
+#define DECLARE_MEMCG_REPARENT_OPS(name)                               \
+       extern const struct memcg_reparent_ops memcg_##name##_reparent_ops
+
 /*
  * size of first charge trial. "32" comes from vmscan.c's magic value.
  * TODO: maybe necessary to use big numbers in big irons.
index 346a954e190e75ac418f4578c29f16f6c2a306c5..6ef3a264054e8c84e9b4138b287a1c7ab3d03b90 100644 (file)
@@ -335,24 +335,60 @@ static struct obj_cgroup *obj_cgroup_alloc(void)
        return objcg;
 }
 
-static void memcg_reparent_objcgs(struct mem_cgroup *memcg)
+static void objcg_reparent_lock(struct mem_cgroup *src, struct mem_cgroup *dst)
+{
+       spin_lock(&objcg_lock);
+}
+
+static void objcg_reparent_relocate(struct mem_cgroup *src, struct mem_cgroup *dst)
 {
        struct obj_cgroup *objcg, *iter;
-       struct mem_cgroup *parent = parent_mem_cgroup(memcg);
 
-       objcg = rcu_replace_pointer(memcg->objcg, NULL, true);
+       objcg = rcu_replace_pointer(src->objcg, NULL, true);
+       /* 1) Ready to reparent active objcg. */
+       list_add(&objcg->list, &src->objcg_list);
+       /* 2) Reparent active objcg and already reparented objcgs to dst. */
+       list_for_each_entry(iter, &src->objcg_list, list)
+               WRITE_ONCE(iter->memcg, dst);
+       /* 3) Move already reparented objcgs to the dst's list */
+       list_splice(&src->objcg_list, &dst->objcg_list);
+}
+
+static void objcg_reparent_unlock(struct mem_cgroup *src, struct mem_cgroup *dst)
+{
+       spin_unlock(&objcg_lock);
+}
 
-       spin_lock_irq(&objcg_lock);
+static DEFINE_MEMCG_REPARENT_OPS(objcg);
 
-       /* 1) Ready to reparent active objcg. */
-       list_add(&objcg->list, &memcg->objcg_list);
-       /* 2) Reparent active objcg and already reparented objcgs to parent. */
-       list_for_each_entry(iter, &memcg->objcg_list, list)
-               WRITE_ONCE(iter->memcg, parent);
-       /* 3) Move already reparented objcgs to the parent's list */
-       list_splice(&memcg->objcg_list, &parent->objcg_list);
-
-       spin_unlock_irq(&objcg_lock);
+static const struct memcg_reparent_ops *memcg_reparent_ops[] = {
+       &memcg_objcg_reparent_ops,
+};
+
+#define DEFINE_MEMCG_REPARENT_FUNC(phase)                              \
+       static void memcg_reparent_##phase(struct mem_cgroup *src,      \
+                                          struct mem_cgroup *dst)      \
+       {                                                               \
+               int i;                                                  \
+                                                                       \
+               for (i = 0; i < ARRAY_SIZE(memcg_reparent_ops); i++)    \
+                       memcg_reparent_ops[i]->phase(src, dst);         \
+       }
+
+DEFINE_MEMCG_REPARENT_FUNC(lock)
+DEFINE_MEMCG_REPARENT_FUNC(relocate)
+DEFINE_MEMCG_REPARENT_FUNC(unlock)
+
+static void memcg_reparent_objcgs(struct mem_cgroup *src)
+{
+       struct mem_cgroup *dst = parent_mem_cgroup(src);
+       struct obj_cgroup *objcg = rcu_dereference_protected(src->objcg, true);
+
+       local_irq_disable();
+       memcg_reparent_lock(src, dst);
+       memcg_reparent_relocate(src, dst);
+       memcg_reparent_unlock(src, dst);
+       local_irq_enable();
 
        percpu_ref_kill(&objcg->refcnt);
 }