diff options
Diffstat (limited to 'mm/oom_kill.c')
-rw-r--r-- | mm/oom_kill.c | 49 |
1 files changed, 44 insertions, 5 deletions
diff --git a/mm/oom_kill.c b/mm/oom_kill.c index ef5084dbc793..4194b9db0104 100644 --- a/mm/oom_kill.c +++ b/mm/oom_kill.c | |||
@@ -29,6 +29,7 @@ | |||
29 | 29 | ||
30 | int sysctl_panic_on_oom; | 30 | int sysctl_panic_on_oom; |
31 | int sysctl_oom_kill_allocating_task; | 31 | int sysctl_oom_kill_allocating_task; |
32 | int sysctl_oom_dump_tasks; | ||
32 | static DEFINE_SPINLOCK(zone_scan_mutex); | 33 | static DEFINE_SPINLOCK(zone_scan_mutex); |
33 | /* #define DEBUG */ | 34 | /* #define DEBUG */ |
34 | 35 | ||
@@ -263,6 +264,41 @@ static struct task_struct *select_bad_process(unsigned long *ppoints, | |||
263 | } | 264 | } |
264 | 265 | ||
265 | /** | 266 | /** |
267 | * Dumps the current memory state of all system tasks, excluding kernel threads. | ||
268 | * State information includes task's pid, uid, tgid, vm size, rss, cpu, oom_adj | ||
269 | * score, and name. | ||
270 | * | ||
271 | * If the actual is non-NULL, only tasks that are a member of the mem_cgroup are | ||
272 | * shown. | ||
273 | * | ||
274 | * Call with tasklist_lock read-locked. | ||
275 | */ | ||
276 | static void dump_tasks(const struct mem_cgroup *mem) | ||
277 | { | ||
278 | struct task_struct *g, *p; | ||
279 | |||
280 | printk(KERN_INFO "[ pid ] uid tgid total_vm rss cpu oom_adj " | ||
281 | "name\n"); | ||
282 | do_each_thread(g, p) { | ||
283 | /* | ||
284 | * total_vm and rss sizes do not exist for tasks with a | ||
285 | * detached mm so there's no need to report them. | ||
286 | */ | ||
287 | if (!p->mm) | ||
288 | continue; | ||
289 | if (mem && !task_in_mem_cgroup(p, mem)) | ||
290 | continue; | ||
291 | |||
292 | task_lock(p); | ||
293 | printk(KERN_INFO "[%5d] %5d %5d %8lu %8lu %3d %3d %s\n", | ||
294 | p->pid, p->uid, p->tgid, p->mm->total_vm, | ||
295 | get_mm_rss(p->mm), (int)task_cpu(p), p->oomkilladj, | ||
296 | p->comm); | ||
297 | task_unlock(p); | ||
298 | } while_each_thread(g, p); | ||
299 | } | ||
300 | |||
301 | /** | ||
266 | * Send SIGKILL to the selected process irrespective of CAP_SYS_RAW_IO | 302 | * Send SIGKILL to the selected process irrespective of CAP_SYS_RAW_IO |
267 | * flag though it's unlikely that we select a process with CAP_SYS_RAW_IO | 303 | * flag though it's unlikely that we select a process with CAP_SYS_RAW_IO |
268 | * set. | 304 | * set. |
@@ -339,7 +375,8 @@ static int oom_kill_task(struct task_struct *p) | |||
339 | } | 375 | } |
340 | 376 | ||
341 | static int oom_kill_process(struct task_struct *p, gfp_t gfp_mask, int order, | 377 | static int oom_kill_process(struct task_struct *p, gfp_t gfp_mask, int order, |
342 | unsigned long points, const char *message) | 378 | unsigned long points, struct mem_cgroup *mem, |
379 | const char *message) | ||
343 | { | 380 | { |
344 | struct task_struct *c; | 381 | struct task_struct *c; |
345 | 382 | ||
@@ -349,6 +386,8 @@ static int oom_kill_process(struct task_struct *p, gfp_t gfp_mask, int order, | |||
349 | current->comm, gfp_mask, order, current->oomkilladj); | 386 | current->comm, gfp_mask, order, current->oomkilladj); |
350 | dump_stack(); | 387 | dump_stack(); |
351 | show_mem(); | 388 | show_mem(); |
389 | if (sysctl_oom_dump_tasks) | ||
390 | dump_tasks(mem); | ||
352 | } | 391 | } |
353 | 392 | ||
354 | /* | 393 | /* |
@@ -389,7 +428,7 @@ retry: | |||
389 | if (!p) | 428 | if (!p) |
390 | p = current; | 429 | p = current; |
391 | 430 | ||
392 | if (oom_kill_process(p, gfp_mask, 0, points, | 431 | if (oom_kill_process(p, gfp_mask, 0, points, mem, |
393 | "Memory cgroup out of memory")) | 432 | "Memory cgroup out of memory")) |
394 | goto retry; | 433 | goto retry; |
395 | out: | 434 | out: |
@@ -495,7 +534,7 @@ void out_of_memory(struct zonelist *zonelist, gfp_t gfp_mask, int order) | |||
495 | 534 | ||
496 | switch (constraint) { | 535 | switch (constraint) { |
497 | case CONSTRAINT_MEMORY_POLICY: | 536 | case CONSTRAINT_MEMORY_POLICY: |
498 | oom_kill_process(current, gfp_mask, order, points, | 537 | oom_kill_process(current, gfp_mask, order, points, NULL, |
499 | "No available memory (MPOL_BIND)"); | 538 | "No available memory (MPOL_BIND)"); |
500 | break; | 539 | break; |
501 | 540 | ||
@@ -505,7 +544,7 @@ void out_of_memory(struct zonelist *zonelist, gfp_t gfp_mask, int order) | |||
505 | /* Fall-through */ | 544 | /* Fall-through */ |
506 | case CONSTRAINT_CPUSET: | 545 | case CONSTRAINT_CPUSET: |
507 | if (sysctl_oom_kill_allocating_task) { | 546 | if (sysctl_oom_kill_allocating_task) { |
508 | oom_kill_process(current, gfp_mask, order, points, | 547 | oom_kill_process(current, gfp_mask, order, points, NULL, |
509 | "Out of memory (oom_kill_allocating_task)"); | 548 | "Out of memory (oom_kill_allocating_task)"); |
510 | break; | 549 | break; |
511 | } | 550 | } |
@@ -525,7 +564,7 @@ retry: | |||
525 | panic("Out of memory and no killable processes...\n"); | 564 | panic("Out of memory and no killable processes...\n"); |
526 | } | 565 | } |
527 | 566 | ||
528 | if (oom_kill_process(p, gfp_mask, order, points, | 567 | if (oom_kill_process(p, gfp_mask, order, points, NULL, |
529 | "Out of memory")) | 568 | "Out of memory")) |
530 | goto retry; | 569 | goto retry; |
531 | 570 | ||