mm/zswap: Convert dst-mem to hotplug state machine
authorSebastian Andrzej Siewior <bigeasy@linutronix.de>
Sat, 26 Nov 2016 23:13:39 +0000 (00:13 +0100)
committerThomas Gleixner <tglx@linutronix.de>
Thu, 1 Dec 2016 23:52:36 +0000 (00:52 +0100)
Install the callbacks via the state machine and let the core invoke
the callbacks on the already online CPUs.

Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Cc: linux-mm@kvack.org
Cc: Seth Jennings <sjenning@redhat.com>
Cc: rt@linutronix.de
Link: http://lkml.kernel.org/r/20161126231350.10321-12-bigeasy@linutronix.de
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
include/linux/cpuhotplug.h
mm/zswap.c

index 9f29dd9960881614175456f2dfe48e95547de74a..62f51a4e867690c472397dd2a26d1cd5ac4a5071 100644 (file)
@@ -65,6 +65,7 @@ enum cpuhp_state {
        CPUHP_ARM_BL_PREPARE,
        CPUHP_TRACE_RB_PREPARE,
        CPUHP_MM_ZS_PREPARE,
+       CPUHP_MM_ZSWP_MEM_PREPARE,
        CPUHP_TIMERS_DEAD,
        CPUHP_NOTF_ERR_INJ_PREPARE,
        CPUHP_MIPS_SOC_PREPARE,
index 275b22cc8df4352c8210948e4381aa4bce83dc08..b13aa5706348289162522f06eb1510ccd11a9ff6 100644 (file)
@@ -352,70 +352,28 @@ static struct zswap_entry *zswap_entry_find_get(struct rb_root *root,
 **********************************/
 static DEFINE_PER_CPU(u8 *, zswap_dstmem);
 
-static int __zswap_cpu_dstmem_notifier(unsigned long action, unsigned long cpu)
+static int zswap_dstmem_prepare(unsigned int cpu)
 {
        u8 *dst;
 
-       switch (action) {
-       case CPU_UP_PREPARE:
-               dst = kmalloc_node(PAGE_SIZE * 2, GFP_KERNEL, cpu_to_node(cpu));
-               if (!dst) {
-                       pr_err("can't allocate compressor buffer\n");
-                       return NOTIFY_BAD;
-               }
-               per_cpu(zswap_dstmem, cpu) = dst;
-               break;
-       case CPU_DEAD:
-       case CPU_UP_CANCELED:
-               dst = per_cpu(zswap_dstmem, cpu);
-               kfree(dst);
-               per_cpu(zswap_dstmem, cpu) = NULL;
-               break;
-       default:
-               break;
+       dst = kmalloc_node(PAGE_SIZE * 2, GFP_KERNEL, cpu_to_node(cpu));
+       if (!dst) {
+               pr_err("can't allocate compressor buffer\n");
+               return -ENOMEM;
        }
-       return NOTIFY_OK;
+       per_cpu(zswap_dstmem, cpu) = dst;
+       return 0;
 }
 
-static int zswap_cpu_dstmem_notifier(struct notifier_block *nb,
-                                    unsigned long action, void *pcpu)
+static int zswap_dstmem_dead(unsigned int cpu)
 {
-       return __zswap_cpu_dstmem_notifier(action, (unsigned long)pcpu);
-}
-
-static struct notifier_block zswap_dstmem_notifier = {
-       .notifier_call =        zswap_cpu_dstmem_notifier,
-};
+       u8 *dst;
 
-static int __init zswap_cpu_dstmem_init(void)
-{
-       unsigned long cpu;
+       dst = per_cpu(zswap_dstmem, cpu);
+       kfree(dst);
+       per_cpu(zswap_dstmem, cpu) = NULL;
 
-       cpu_notifier_register_begin();
-       for_each_online_cpu(cpu)
-               if (__zswap_cpu_dstmem_notifier(CPU_UP_PREPARE, cpu) ==
-                   NOTIFY_BAD)
-                       goto cleanup;
-       __register_cpu_notifier(&zswap_dstmem_notifier);
-       cpu_notifier_register_done();
        return 0;
-
-cleanup:
-       for_each_online_cpu(cpu)
-               __zswap_cpu_dstmem_notifier(CPU_UP_CANCELED, cpu);
-       cpu_notifier_register_done();
-       return -ENOMEM;
-}
-
-static void zswap_cpu_dstmem_destroy(void)
-{
-       unsigned long cpu;
-
-       cpu_notifier_register_begin();
-       for_each_online_cpu(cpu)
-               __zswap_cpu_dstmem_notifier(CPU_UP_CANCELED, cpu);
-       __unregister_cpu_notifier(&zswap_dstmem_notifier);
-       cpu_notifier_register_done();
 }
 
 static int __zswap_cpu_comp_notifier(struct zswap_pool *pool,
@@ -1238,6 +1196,7 @@ static void __exit zswap_debugfs_exit(void) { }
 static int __init init_zswap(void)
 {
        struct zswap_pool *pool;
+       int ret;
 
        zswap_init_started = true;
 
@@ -1246,7 +1205,9 @@ static int __init init_zswap(void)
                goto cache_fail;
        }
 
-       if (zswap_cpu_dstmem_init()) {
+       ret = cpuhp_setup_state(CPUHP_MM_ZSWP_MEM_PREPARE, "mm/zswap:prepare",
+                               zswap_dstmem_prepare, zswap_dstmem_dead);
+       if (ret) {
                pr_err("dstmem alloc failed\n");
                goto dstmem_fail;
        }
@@ -1267,7 +1228,7 @@ static int __init init_zswap(void)
        return 0;
 
 pool_fail:
-       zswap_cpu_dstmem_destroy();
+       cpuhp_remove_state(CPUHP_MM_ZSWP_MEM_PREPARE);
 dstmem_fail:
        zswap_entry_cache_destroy();
 cache_fail: