Commit 86a4c6d9 authored by Andrea Arcangeli's avatar Andrea Arcangeli Committed by Linus Torvalds

[PATCH] mm: fix several oom killer bugs

Fix several oom killer bugs, most important avoid spurious oom kills
badness algorithm tweaked by Thomas Gleixner to deal with fork bombs

This is the core of the oom-killer fixes I developed partly taking
the idea from Thomas's patches of getting feedback from the exit
path, plus I moved the oom killer into page_alloc.c as it should to
be able to check the watermarks before killing more stuff.  This
also tweaks the badness to take thread bombs more into account (that
change to badness is from Thomas, from my part I'd rather rewrite
badness from scratch instead, but that's an orthgonal issue ;).
With this applied the oom killer is very sane, no more 5 sec waits
and suprious oom kills.
Signed-off-by: default avatarAndrea Arcangeli <andrea@suse.de>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 96fa06f4
...@@ -614,6 +614,11 @@ struct task_struct { ...@@ -614,6 +614,11 @@ struct task_struct {
struct key *process_keyring; /* keyring private to this process (CLONE_THREAD) */ struct key *process_keyring; /* keyring private to this process (CLONE_THREAD) */
struct key *thread_keyring; /* keyring private to this thread */ struct key *thread_keyring; /* keyring private to this thread */
#endif #endif
/*
* All archs should support atomic ops with
* 1 byte granularity.
*/
unsigned char memdie;
/* /*
* Must be changed atomically so it shouldn't be * Must be changed atomically so it shouldn't be
* be a shareable bitflag. * be a shareable bitflag.
...@@ -736,8 +741,7 @@ do { if (atomic_dec_and_test(&(tsk)->usage)) __put_task_struct(tsk); } while(0) ...@@ -736,8 +741,7 @@ do { if (atomic_dec_and_test(&(tsk)->usage)) __put_task_struct(tsk); } while(0)
#define PF_DUMPCORE 0x00000200 /* dumped core */ #define PF_DUMPCORE 0x00000200 /* dumped core */
#define PF_SIGNALED 0x00000400 /* killed by a signal */ #define PF_SIGNALED 0x00000400 /* killed by a signal */
#define PF_MEMALLOC 0x00000800 /* Allocating memory */ #define PF_MEMALLOC 0x00000800 /* Allocating memory */
#define PF_MEMDIE 0x00001000 /* Killed for out-of-memory */ #define PF_FLUSHER 0x00001000 /* responsible for disk writeback */
#define PF_FLUSHER 0x00002000 /* responsible for disk writeback */
#define PF_FREEZE 0x00004000 /* this task is being frozen for suspend now */ #define PF_FREEZE 0x00004000 /* this task is being frozen for suspend now */
#define PF_NOFREEZE 0x00008000 /* this thread should not be frozen */ #define PF_NOFREEZE 0x00008000 /* this thread should not be frozen */
......
...@@ -45,17 +45,29 @@ ...@@ -45,17 +45,29 @@
unsigned long badness(struct task_struct *p, unsigned long uptime) unsigned long badness(struct task_struct *p, unsigned long uptime)
{ {
unsigned long points, cpu_time, run_time, s; unsigned long points, cpu_time, run_time, s;
struct list_head *tsk;
if (!p->mm) if (!p->mm)
return 0; return 0;
if (p->flags & PF_MEMDIE)
return 0;
/* /*
* The memory size of the process is the basis for the badness. * The memory size of the process is the basis for the badness.
*/ */
points = p->mm->total_vm; points = p->mm->total_vm;
/*
* Processes which fork a lot of child processes are likely
* a good choice. We add the vmsize of the childs if they
* have an own mm. This prevents forking servers to flood the
* machine with an endless amount of childs
*/
list_for_each(tsk, &p->children) {
struct task_struct *chld;
chld = list_entry(tsk, struct task_struct, sibling);
if (chld->mm != p->mm && chld->mm)
points += chld->mm->total_vm;
}
/* /*
* CPU time is in tens of seconds and run time is in thousands * CPU time is in tens of seconds and run time is in thousands
* of seconds. There is no particular reason for this other than * of seconds. There is no particular reason for this other than
...@@ -132,14 +144,24 @@ static struct task_struct * select_bad_process(void) ...@@ -132,14 +144,24 @@ static struct task_struct * select_bad_process(void)
do_posix_clock_monotonic_gettime(&uptime); do_posix_clock_monotonic_gettime(&uptime);
do_each_thread(g, p) do_each_thread(g, p)
if (p->pid) { /* skip the init task with pid == 1 */
unsigned long points = badness(p, uptime.tv_sec); if (p->pid > 1) {
if (points > maxpoints) { unsigned long points;
/*
* This is in the process of releasing memory so wait it
* to finish before killing some other task by mistake.
*/
if ((p->memdie || (p->flags & PF_EXITING)) && !(p->flags & PF_DEAD))
return ERR_PTR(-1UL);
if (p->flags & PF_SWAPOFF)
return p;
points = badness(p, uptime.tv_sec);
if (points > maxpoints || !chosen) {
chosen = p; chosen = p;
maxpoints = points; maxpoints = points;
} }
if (p->flags & PF_SWAPOFF)
return p;
} }
while_each_thread(g, p); while_each_thread(g, p);
return chosen; return chosen;
...@@ -152,6 +174,12 @@ static struct task_struct * select_bad_process(void) ...@@ -152,6 +174,12 @@ static struct task_struct * select_bad_process(void)
*/ */
static void __oom_kill_task(task_t *p) static void __oom_kill_task(task_t *p)
{ {
if (p->pid == 1) {
WARN_ON(1);
printk(KERN_WARNING "tried to kill init!\n");
return;
}
task_lock(p); task_lock(p);
if (!p->mm || p->mm == &init_mm) { if (!p->mm || p->mm == &init_mm) {
WARN_ON(1); WARN_ON(1);
...@@ -168,25 +196,53 @@ static void __oom_kill_task(task_t *p) ...@@ -168,25 +196,53 @@ static void __oom_kill_task(task_t *p)
* exit() and clear out its resources quickly... * exit() and clear out its resources quickly...
*/ */
p->time_slice = HZ; p->time_slice = HZ;
p->flags |= PF_MEMALLOC | PF_MEMDIE; p->memdie = 1;
/* This process has hardware access, be more careful. */
if (cap_t(p->cap_effective) & CAP_TO_MASK(CAP_SYS_RAWIO)) {
force_sig(SIGTERM, p);
} else {
force_sig(SIGKILL, p); force_sig(SIGKILL, p);
}
} }
static struct mm_struct *oom_kill_task(task_t *p) static struct mm_struct *oom_kill_task(task_t *p)
{ {
struct mm_struct *mm = get_task_mm(p); struct mm_struct *mm = get_task_mm(p);
if (!mm || mm == &init_mm) task_t * g, * q;
if (!mm)
return NULL;
if (mm == &init_mm) {
mmput(mm);
return NULL; return NULL;
}
__oom_kill_task(p); __oom_kill_task(p);
/*
* kill all processes that share the ->mm (i.e. all threads),
* but are in a different thread group
*/
do_each_thread(g, q)
if (q->mm == mm && q->tgid != p->tgid)
__oom_kill_task(q);
while_each_thread(g, q);
return mm; return mm;
} }
static struct mm_struct *oom_kill_process(struct task_struct *p)
{
struct mm_struct *mm;
struct task_struct *c;
struct list_head *tsk;
/* Try to kill a child first */
list_for_each(tsk, &p->children) {
c = list_entry(tsk, struct task_struct, sibling);
if (c->mm == p->mm)
continue;
mm = oom_kill_task(c);
if (mm)
return mm;
}
return oom_kill_task(p);
}
/** /**
* oom_kill - kill the "best" process when we run out of memory * oom_kill - kill the "best" process when we run out of memory
...@@ -196,117 +252,40 @@ static struct mm_struct *oom_kill_task(task_t *p) ...@@ -196,117 +252,40 @@ static struct mm_struct *oom_kill_task(task_t *p)
* OR try to be smart about which process to kill. Note that we * OR try to be smart about which process to kill. Note that we
* don't have to be perfect here, we just have to be good. * don't have to be perfect here, we just have to be good.
*/ */
static void oom_kill(void) void out_of_memory(int gfp_mask)
{ {
struct mm_struct *mm; struct mm_struct *mm = NULL;
struct task_struct *g, *p, *q; task_t * p;
read_lock(&tasklist_lock); read_lock(&tasklist_lock);
retry: retry:
p = select_bad_process(); p = select_bad_process();
if (PTR_ERR(p) == -1UL)
goto out;
/* Found nothing?!?! Either we hang forever, or we panic. */ /* Found nothing?!?! Either we hang forever, or we panic. */
if (!p) { if (!p) {
read_unlock(&tasklist_lock);
show_free_areas(); show_free_areas();
panic("Out of memory and no killable processes...\n"); panic("Out of memory and no killable processes...\n");
} }
mm = oom_kill_task(p); printk("oom-killer: gfp_mask=0x%x\n", gfp_mask);
show_free_areas();
mm = oom_kill_process(p);
if (!mm) if (!mm)
goto retry; goto retry;
/*
* kill all processes that share the ->mm (i.e. all threads), out:
* but are in a different thread group
*/
do_each_thread(g, q)
if (q->mm == mm && q->tgid != p->tgid)
__oom_kill_task(q);
while_each_thread(g, q);
if (!p->mm)
printk(KERN_INFO "Fixed up OOM kill of mm-less task\n");
read_unlock(&tasklist_lock); read_unlock(&tasklist_lock);
if (mm)
mmput(mm); mmput(mm);
/* /*
* Make kswapd go out of the way, so "p" has a good chance of * Give "p" a good chance of killing itself before we
* killing itself before someone else gets the chance to ask * retry to allocate memory.
* for more memory.
*/
yield();
return;
}
/**
* out_of_memory - is the system out of memory?
*/
void out_of_memory(int gfp_mask)
{
/*
* oom_lock protects out_of_memory()'s static variables.
* It's a global lock; this is not performance-critical.
*/ */
static DEFINE_SPINLOCK(oom_lock); __set_current_state(TASK_INTERRUPTIBLE);
static unsigned long first, last, count, lastkill; schedule_timeout(1);
unsigned long now, since;
spin_lock(&oom_lock);
now = jiffies;
since = now - last;
last = now;
/*
* If it's been a long time since last failure,
* we're not oom.
*/
if (since > 5*HZ)
goto reset;
/*
* If we haven't tried for at least one second,
* we're not really oom.
*/
since = now - first;
if (since < HZ)
goto out_unlock;
/*
* If we have gotten only a few failures,
* we're not really oom.
*/
if (++count < 10)
goto out_unlock;
/*
* If we just killed a process, wait a while
* to give that task a chance to exit. This
* avoids killing multiple processes needlessly.
*/
since = now - lastkill;
if (since < HZ*5)
goto out_unlock;
/*
* Ok, really out of memory. Kill something.
*/
lastkill = now;
printk("oom-killer: gfp_mask=0x%x\n", gfp_mask);
show_free_areas();
/* oom_kill() sleeps */
spin_unlock(&oom_lock);
oom_kill();
spin_lock(&oom_lock);
reset:
/*
* We dropped the lock above, so check to be sure the variable
* first only ever increases to prevent false OOM's.
*/
if (time_after(now, first))
first = now;
count = 0;
out_unlock:
spin_unlock(&oom_lock);
} }
...@@ -704,6 +704,7 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order, ...@@ -704,6 +704,7 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order,
int classzone_idx; int classzone_idx;
int do_retry; int do_retry;
int can_try_harder; int can_try_harder;
int did_some_progress;
might_sleep_if(wait); might_sleep_if(wait);
...@@ -723,6 +724,7 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order, ...@@ -723,6 +724,7 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order,
classzone_idx = zone_idx(zones[0]); classzone_idx = zone_idx(zones[0]);
restart:
/* Go through the zonelist once, looking for a zone with enough free */ /* Go through the zonelist once, looking for a zone with enough free */
for (i = 0; (z = zones[i]) != NULL; i++) { for (i = 0; (z = zones[i]) != NULL; i++) {
...@@ -754,7 +756,7 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order, ...@@ -754,7 +756,7 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order,
} }
/* This allocation should allow future memory freeing. */ /* This allocation should allow future memory freeing. */
if ((p->flags & (PF_MEMALLOC | PF_MEMDIE)) && !in_interrupt()) { if (((p->flags & PF_MEMALLOC) || p->memdie) && !in_interrupt()) {
/* go through the zonelist yet again, ignoring mins */ /* go through the zonelist yet again, ignoring mins */
for (i = 0; (z = zones[i]) != NULL; i++) { for (i = 0; (z = zones[i]) != NULL; i++) {
page = buffered_rmqueue(z, order, gfp_mask); page = buffered_rmqueue(z, order, gfp_mask);
...@@ -769,17 +771,27 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order, ...@@ -769,17 +771,27 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order,
goto nopage; goto nopage;
rebalance: rebalance:
cond_resched();
/* We now go into synchronous reclaim */ /* We now go into synchronous reclaim */
p->flags |= PF_MEMALLOC; p->flags |= PF_MEMALLOC;
reclaim_state.reclaimed_slab = 0; reclaim_state.reclaimed_slab = 0;
p->reclaim_state = &reclaim_state; p->reclaim_state = &reclaim_state;
try_to_free_pages(zones, gfp_mask, order); did_some_progress = try_to_free_pages(zones, gfp_mask, order);
p->reclaim_state = NULL; p->reclaim_state = NULL;
p->flags &= ~PF_MEMALLOC; p->flags &= ~PF_MEMALLOC;
/* go through the zonelist yet one more time */ cond_resched();
if (likely(did_some_progress)) {
/*
* Go through the zonelist yet one more time, keep
* very high watermark here, this is only to catch
* a parallel oom killing, we must fail if we're still
* under heavy pressure.
*/
for (i = 0; (z = zones[i]) != NULL; i++) { for (i = 0; (z = zones[i]) != NULL; i++) {
if (!zone_watermark_ok(z, order, z->pages_min, if (!zone_watermark_ok(z, order, z->pages_min,
classzone_idx, can_try_harder, classzone_idx, can_try_harder,
...@@ -790,6 +802,26 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order, ...@@ -790,6 +802,26 @@ __alloc_pages(unsigned int gfp_mask, unsigned int order,
if (page) if (page)
goto got_pg; goto got_pg;
} }
} else if ((gfp_mask & __GFP_FS) && !(gfp_mask & __GFP_NORETRY)) {
/*
* Go through the zonelist yet one more time, keep
* very high watermark here, this is only to catch
* a parallel oom killing, we must fail if we're still
* under heavy pressure.
*/
for (i = 0; (z = zones[i]) != NULL; i++) {
if (!zone_watermark_ok(z, order, z->pages_high,
classzone_idx, 0, 0))
continue;
page = buffered_rmqueue(z, order, gfp_mask);
if (page)
goto got_pg;
}
out_of_memory(gfp_mask);
goto restart;
}
/* /*
* Don't let big-order allocations loop unless the caller explicitly * Don't let big-order allocations loop unless the caller explicitly
......
...@@ -59,6 +59,8 @@ void show_swap_cache_info(void) ...@@ -59,6 +59,8 @@ void show_swap_cache_info(void)
swap_cache_info.add_total, swap_cache_info.del_total, swap_cache_info.add_total, swap_cache_info.del_total,
swap_cache_info.find_success, swap_cache_info.find_total, swap_cache_info.find_success, swap_cache_info.find_total,
swap_cache_info.noent_race, swap_cache_info.exist_race); swap_cache_info.noent_race, swap_cache_info.exist_race);
printk("Free swap = %lukB\n", nr_swap_pages << (PAGE_SHIFT - 10));
printk("Total swap = %lukB\n", total_swap_pages << (PAGE_SHIFT - 10));
} }
/* /*
......
...@@ -937,8 +937,6 @@ int try_to_free_pages(struct zone **zones, ...@@ -937,8 +937,6 @@ int try_to_free_pages(struct zone **zones,
if (sc.nr_scanned && priority < DEF_PRIORITY - 2) if (sc.nr_scanned && priority < DEF_PRIORITY - 2)
blk_congestion_wait(WRITE, HZ/10); blk_congestion_wait(WRITE, HZ/10);
} }
if ((gfp_mask & __GFP_FS) && !(gfp_mask & __GFP_NORETRY))
out_of_memory(gfp_mask);
out: out:
for (i = 0; zones[i] != 0; i++) for (i = 0; zones[i] != 0; i++)
zones[i]->prev_priority = zones[i]->temp_priority; zones[i]->prev_priority = zones[i]->temp_priority;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment