| 
									
										
										
										
											2006-10-19 23:28:32 -07:00
										 |  |  | #ifndef __INCLUDE_LINUX_OOM_H
 | 
					
						
							|  |  |  | #define __INCLUDE_LINUX_OOM_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:53 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												oom: badness heuristic rewrite
This a complete rewrite of the oom killer's badness() heuristic which is
used to determine which task to kill in oom conditions.  The goal is to
make it as simple and predictable as possible so the results are better
understood and we end up killing the task which will lead to the most
memory freeing while still respecting the fine-tuning from userspace.
Instead of basing the heuristic on mm->total_vm for each task, the task's
rss and swap space is used instead.  This is a better indication of the
amount of memory that will be freeable if the oom killed task is chosen
and subsequently exits.  This helps specifically in cases where KDE or
GNOME is chosen for oom kill on desktop systems instead of a memory
hogging task.
The baseline for the heuristic is a proportion of memory that each task is
currently using in memory plus swap compared to the amount of "allowable"
memory.  "Allowable," in this sense, means the system-wide resources for
unconstrained oom conditions, the set of mempolicy nodes, the mems
attached to current's cpuset, or a memory controller's limit.  The
proportion is given on a scale of 0 (never kill) to 1000 (always kill),
roughly meaning that if a task has a badness() score of 500 that the task
consumes approximately 50% of allowable memory resident in RAM or in swap
space.
The proportion is always relative to the amount of "allowable" memory and
not the total amount of RAM systemwide so that mempolicies and cpusets may
operate in isolation; they shall not need to know the true size of the
machine on which they are running if they are bound to a specific set of
nodes or mems, respectively.
Root tasks are given 3% extra memory just like __vm_enough_memory()
provides in LSMs.  In the event of two tasks consuming similar amounts of
memory, it is generally better to save root's task.
Because of the change in the badness() heuristic's baseline, it is also
necessary to introduce a new user interface to tune it.  It's not possible
to redefine the meaning of /proc/pid/oom_adj with a new scale since the
ABI cannot be changed for backward compatability.  Instead, a new tunable,
/proc/pid/oom_score_adj, is added that ranges from -1000 to +1000.  It may
be used to polarize the heuristic such that certain tasks are never
considered for oom kill while others may always be considered.  The value
is added directly into the badness() score so a value of -500, for
example, means to discount 50% of its memory consumption in comparison to
other tasks either on the system, bound to the mempolicy, in the cpuset,
or sharing the same memory controller.
/proc/pid/oom_adj is changed so that its meaning is rescaled into the
units used by /proc/pid/oom_score_adj, and vice versa.  Changing one of
these per-task tunables will rescale the value of the other to an
equivalent meaning.  Although /proc/pid/oom_adj was originally defined as
a bitshift on the badness score, it now shares the same linear growth as
/proc/pid/oom_score_adj but with different granularity.  This is required
so the ABI is not broken with userspace applications and allows oom_adj to
be deprecated for future removal.
Signed-off-by: David Rientjes <rientjes@google.com>
Cc: Nick Piggin <npiggin@suse.de>
Cc: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Cc: KOSAKI Motohiro <kosaki.motohiro@jp.fujitsu.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: Balbir Singh <balbir@in.ibm.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
											
										 
											2010-08-09 17:19:46 -07:00
										 |  |  | #include <linux/sched.h>
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:59 -07:00
										 |  |  | #include <linux/types.h>
 | 
					
						
							| 
									
										
										
										
											2009-12-15 16:45:33 -08:00
										 |  |  | #include <linux/nodemask.h>
 | 
					
						
							| 
									
										
										
										
											2012-10-13 10:46:48 +01:00
										 |  |  | #include <uapi/linux/oom.h>
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:59 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct zonelist; | 
					
						
							|  |  |  | struct notifier_block; | 
					
						
							| 
									
										
										
										
											2010-08-09 17:19:43 -07:00
										 |  |  | struct mem_cgroup; | 
					
						
							|  |  |  | struct task_struct; | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:59 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:53 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Types of limitations to the nodes from which allocations may occur | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | enum oom_constraint { | 
					
						
							|  |  |  | 	CONSTRAINT_NONE, | 
					
						
							|  |  |  | 	CONSTRAINT_CPUSET, | 
					
						
							|  |  |  | 	CONSTRAINT_MEMORY_POLICY, | 
					
						
							| 
									
										
										
										
											2010-08-09 17:18:54 -07:00
										 |  |  | 	CONSTRAINT_MEMCG, | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:53 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												mm, memcg: introduce own oom handler to iterate only over its own threads
The global oom killer is serialized by the per-zonelist
try_set_zonelist_oom() which is used in the page allocator.  Concurrent
oom kills are thus a rare event and only occur in systems using
mempolicies and with a large number of nodes.
Memory controller oom kills, however, can frequently be concurrent since
there is no serialization once the oom killer is called for oom conditions
in several different memcgs in parallel.
This creates a massive contention on tasklist_lock since the oom killer
requires the readside for the tasklist iteration.  If several memcgs are
calling the oom killer, this lock can be held for a substantial amount of
time, especially if threads continue to enter it as other threads are
exiting.
Since the exit path grabs the writeside of the lock with irqs disabled in
a few different places, this can cause a soft lockup on cpus as a result
of tasklist_lock starvation.
The kernel lacks unfair writelocks, and successful calls to the oom killer
usually result in at least one thread entering the exit path, so an
alternative solution is needed.
This patch introduces a seperate oom handler for memcgs so that they do
not require tasklist_lock for as much time.  Instead, it iterates only
over the threads attached to the oom memcg and grabs a reference to the
selected thread before calling oom_kill_process() to ensure it doesn't
prematurely exit.
This still requires tasklist_lock for the tasklist dump, iterating
children of the selected process, and killing all other threads on the
system sharing the same memory as the selected victim.  So while this
isn't a complete solution to tasklist_lock starvation, it significantly
reduces the amount of time that it is held.
Acked-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Acked-by: Michal Hocko <mhocko@suse.cz>
Signed-off-by: David Rientjes <rientjes@google.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: KOSAKI Motohiro <kosaki.motohiro@jp.fujitsu.com>
Reviewed-by: Sha Zhengju <handai.szj@taobao.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
											
										 
											2012-07-31 16:43:44 -07:00
										 |  |  | enum oom_scan_t { | 
					
						
							|  |  |  | 	OOM_SCAN_OK,		/* scan thread and find its badness */ | 
					
						
							|  |  |  | 	OOM_SCAN_CONTINUE,	/* do not consider thread for oom kill */ | 
					
						
							|  |  |  | 	OOM_SCAN_ABORT,		/* abort the iteration and return */ | 
					
						
							|  |  |  | 	OOM_SCAN_SELECT,	/* always select this thread first */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 16:02:56 -08:00
										 |  |  | /* Thread is the potential origin of an oom condition; kill first on oom */ | 
					
						
							|  |  |  | #define OOM_FLAG_ORIGIN		((__force oom_flags_t)0x1)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void set_current_oom_origin(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	current->signal->oom_flags |= OOM_FLAG_ORIGIN; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void clear_current_oom_origin(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	current->signal->oom_flags &= ~OOM_FLAG_ORIGIN; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline bool oom_task_origin(const struct task_struct *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return !!(p->signal->oom_flags & OOM_FLAG_ORIGIN); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-05-24 17:11:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 15:06:47 -07:00
										 |  |  | extern unsigned long oom_badness(struct task_struct *p, | 
					
						
							|  |  |  | 		struct mem_cgroup *memcg, const nodemask_t *nodemask, | 
					
						
							|  |  |  | 		unsigned long totalpages); | 
					
						
							| 
									
										
											  
											
												mm, memcg: introduce own oom handler to iterate only over its own threads
The global oom killer is serialized by the per-zonelist
try_set_zonelist_oom() which is used in the page allocator.  Concurrent
oom kills are thus a rare event and only occur in systems using
mempolicies and with a large number of nodes.
Memory controller oom kills, however, can frequently be concurrent since
there is no serialization once the oom killer is called for oom conditions
in several different memcgs in parallel.
This creates a massive contention on tasklist_lock since the oom killer
requires the readside for the tasklist iteration.  If several memcgs are
calling the oom killer, this lock can be held for a substantial amount of
time, especially if threads continue to enter it as other threads are
exiting.
Since the exit path grabs the writeside of the lock with irqs disabled in
a few different places, this can cause a soft lockup on cpus as a result
of tasklist_lock starvation.
The kernel lacks unfair writelocks, and successful calls to the oom killer
usually result in at least one thread entering the exit path, so an
alternative solution is needed.
This patch introduces a seperate oom handler for memcgs so that they do
not require tasklist_lock for as much time.  Instead, it iterates only
over the threads attached to the oom memcg and grabs a reference to the
selected thread before calling oom_kill_process() to ensure it doesn't
prematurely exit.
This still requires tasklist_lock for the tasklist dump, iterating
children of the selected process, and killing all other threads on the
system sharing the same memory as the selected victim.  So while this
isn't a complete solution to tasklist_lock starvation, it significantly
reduces the amount of time that it is held.
Acked-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Acked-by: Michal Hocko <mhocko@suse.cz>
Signed-off-by: David Rientjes <rientjes@google.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: KOSAKI Motohiro <kosaki.motohiro@jp.fujitsu.com>
Reviewed-by: Sha Zhengju <handai.szj@taobao.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
											
										 
											2012-07-31 16:43:44 -07:00
										 |  |  | extern void oom_kill_process(struct task_struct *p, gfp_t gfp_mask, int order, | 
					
						
							|  |  |  | 			     unsigned int points, unsigned long totalpages, | 
					
						
							|  |  |  | 			     struct mem_cgroup *memcg, nodemask_t *nodemask, | 
					
						
							|  |  |  | 			     const char *message); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 17:18:57 -07:00
										 |  |  | extern int try_set_zonelist_oom(struct zonelist *zonelist, gfp_t gfp_flags); | 
					
						
							| 
									
										
										
										
											2008-04-28 02:12:17 -07:00
										 |  |  | extern void clear_zonelist_oom(struct zonelist *zonelist, gfp_t gfp_flags); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:55 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-31 16:43:48 -07:00
										 |  |  | extern void check_panic_on_oom(enum oom_constraint constraint, gfp_t gfp_mask, | 
					
						
							|  |  |  | 			       int order, const nodemask_t *nodemask); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												mm, memcg: introduce own oom handler to iterate only over its own threads
The global oom killer is serialized by the per-zonelist
try_set_zonelist_oom() which is used in the page allocator.  Concurrent
oom kills are thus a rare event and only occur in systems using
mempolicies and with a large number of nodes.
Memory controller oom kills, however, can frequently be concurrent since
there is no serialization once the oom killer is called for oom conditions
in several different memcgs in parallel.
This creates a massive contention on tasklist_lock since the oom killer
requires the readside for the tasklist iteration.  If several memcgs are
calling the oom killer, this lock can be held for a substantial amount of
time, especially if threads continue to enter it as other threads are
exiting.
Since the exit path grabs the writeside of the lock with irqs disabled in
a few different places, this can cause a soft lockup on cpus as a result
of tasklist_lock starvation.
The kernel lacks unfair writelocks, and successful calls to the oom killer
usually result in at least one thread entering the exit path, so an
alternative solution is needed.
This patch introduces a seperate oom handler for memcgs so that they do
not require tasklist_lock for as much time.  Instead, it iterates only
over the threads attached to the oom memcg and grabs a reference to the
selected thread before calling oom_kill_process() to ensure it doesn't
prematurely exit.
This still requires tasklist_lock for the tasklist dump, iterating
children of the selected process, and killing all other threads on the
system sharing the same memory as the selected victim.  So while this
isn't a complete solution to tasklist_lock starvation, it significantly
reduces the amount of time that it is held.
Acked-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Acked-by: Michal Hocko <mhocko@suse.cz>
Signed-off-by: David Rientjes <rientjes@google.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: KOSAKI Motohiro <kosaki.motohiro@jp.fujitsu.com>
Reviewed-by: Sha Zhengju <handai.szj@taobao.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
											
										 
											2012-07-31 16:43:44 -07:00
										 |  |  | extern enum oom_scan_t oom_scan_process_thread(struct task_struct *task, | 
					
						
							|  |  |  | 		unsigned long totalpages, const nodemask_t *nodemask, | 
					
						
							|  |  |  | 		bool force_kill); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-15 16:45:33 -08:00
										 |  |  | extern void out_of_memory(struct zonelist *zonelist, gfp_t gfp_mask, | 
					
						
							| 
									
										
										
										
											2012-03-21 16:34:04 -07:00
										 |  |  | 		int order, nodemask_t *mask, bool force_kill); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:53 -07:00
										 |  |  | extern int register_oom_notifier(struct notifier_block *nb); | 
					
						
							|  |  |  | extern int unregister_oom_notifier(struct notifier_block *nb); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-21 17:03:09 -07:00
										 |  |  | extern bool oom_killer_disabled; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void oom_killer_disable(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	oom_killer_disabled = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void oom_killer_enable(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	oom_killer_disabled = false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-08-09 17:18:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-12 15:07:22 -08:00
										 |  |  | static inline bool oom_gfp_allowed(gfp_t gfp_mask) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (gfp_mask & __GFP_FS) && !(gfp_mask & __GFP_NORETRY); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 18:03:00 -07:00
										 |  |  | extern struct task_struct *find_lock_task_mm(struct task_struct *p); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 17:18:56 -07:00
										 |  |  | /* sysctls */ | 
					
						
							|  |  |  | extern int sysctl_oom_dump_tasks; | 
					
						
							|  |  |  | extern int sysctl_oom_kill_allocating_task; | 
					
						
							|  |  |  | extern int sysctl_panic_on_oom; | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:53 -07:00
										 |  |  | #endif /* _INCLUDE_LINUX_OOM_H */
 |