diff options
Diffstat (limited to 'mm/vmscan.c')
-rw-r--r-- | mm/vmscan.c | 39 |
1 files changed, 27 insertions, 12 deletions
diff --git a/mm/vmscan.c b/mm/vmscan.c index 72babac71dea..eeacb0d695c3 100644 --- a/mm/vmscan.c +++ b/mm/vmscan.c | |||
@@ -34,6 +34,7 @@ | |||
34 | #include <linux/notifier.h> | 34 | #include <linux/notifier.h> |
35 | #include <linux/rwsem.h> | 35 | #include <linux/rwsem.h> |
36 | #include <linux/delay.h> | 36 | #include <linux/delay.h> |
37 | #include <linux/kthread.h> | ||
37 | 38 | ||
38 | #include <asm/tlbflush.h> | 39 | #include <asm/tlbflush.h> |
39 | #include <asm/div64.h> | 40 | #include <asm/div64.h> |
@@ -1223,7 +1224,6 @@ static int kswapd(void *p) | |||
1223 | }; | 1224 | }; |
1224 | cpumask_t cpumask; | 1225 | cpumask_t cpumask; |
1225 | 1226 | ||
1226 | daemonize("kswapd%d", pgdat->node_id); | ||
1227 | cpumask = node_to_cpumask(pgdat->node_id); | 1227 | cpumask = node_to_cpumask(pgdat->node_id); |
1228 | if (!cpus_empty(cpumask)) | 1228 | if (!cpus_empty(cpumask)) |
1229 | set_cpus_allowed(tsk, cpumask); | 1229 | set_cpus_allowed(tsk, cpumask); |
@@ -1450,7 +1450,7 @@ out: | |||
1450 | not required for correctness. So if the last cpu in a node goes | 1450 | not required for correctness. So if the last cpu in a node goes |
1451 | away, we get changed to run anywhere: as the first one comes back, | 1451 | away, we get changed to run anywhere: as the first one comes back, |
1452 | restore their cpu bindings. */ | 1452 | restore their cpu bindings. */ |
1453 | static int cpu_callback(struct notifier_block *nfb, | 1453 | static int __devinit cpu_callback(struct notifier_block *nfb, |
1454 | unsigned long action, void *hcpu) | 1454 | unsigned long action, void *hcpu) |
1455 | { | 1455 | { |
1456 | pg_data_t *pgdat; | 1456 | pg_data_t *pgdat; |
@@ -1468,20 +1468,35 @@ static int cpu_callback(struct notifier_block *nfb, | |||
1468 | } | 1468 | } |
1469 | #endif /* CONFIG_HOTPLUG_CPU */ | 1469 | #endif /* CONFIG_HOTPLUG_CPU */ |
1470 | 1470 | ||
1471 | /* | ||
1472 | * This kswapd start function will be called by init and node-hot-add. | ||
1473 | * On node-hot-add, kswapd will moved to proper cpus if cpus are hot-added. | ||
1474 | */ | ||
1475 | int kswapd_run(int nid) | ||
1476 | { | ||
1477 | pg_data_t *pgdat = NODE_DATA(nid); | ||
1478 | int ret = 0; | ||
1479 | |||
1480 | if (pgdat->kswapd) | ||
1481 | return 0; | ||
1482 | |||
1483 | pgdat->kswapd = kthread_run(kswapd, pgdat, "kswapd%d", nid); | ||
1484 | if (IS_ERR(pgdat->kswapd)) { | ||
1485 | /* failure at boot is fatal */ | ||
1486 | BUG_ON(system_state == SYSTEM_BOOTING); | ||
1487 | printk("Failed to start kswapd on node %d\n",nid); | ||
1488 | ret = -1; | ||
1489 | } | ||
1490 | return ret; | ||
1491 | } | ||
1492 | |||
1471 | static int __init kswapd_init(void) | 1493 | static int __init kswapd_init(void) |
1472 | { | 1494 | { |
1473 | pg_data_t *pgdat; | 1495 | int nid; |
1474 | 1496 | ||
1475 | swap_setup(); | 1497 | swap_setup(); |
1476 | for_each_online_pgdat(pgdat) { | 1498 | for_each_online_node(nid) |
1477 | pid_t pid; | 1499 | kswapd_run(nid); |
1478 | |||
1479 | pid = kernel_thread(kswapd, pgdat, CLONE_KERNEL); | ||
1480 | BUG_ON(pid < 0); | ||
1481 | read_lock(&tasklist_lock); | ||
1482 | pgdat->kswapd = find_task_by_pid(pid); | ||
1483 | read_unlock(&tasklist_lock); | ||
1484 | } | ||
1485 | hotcpu_notifier(cpu_callback, 0); | 1500 | hotcpu_notifier(cpu_callback, 0); |
1486 | return 0; | 1501 | return 0; |
1487 | } | 1502 | } |