Commit fafd688e authored by Peter W Morreale's avatar Peter W Morreale Committed by Linus Torvalds

mm: add /proc controls for pdflush threads

Add /proc entries to give the admin the ability to control the minimum and
maximum number of pdflush threads.  This allows finer control of pdflush
on both large and small machines.

The rationale is simply one size does not fit all.  Admins on large and/or
small systems may want to tune the min/max pdflush thread count to best
suit their needs.  Right now the min/max is hardcoded to 2/8.  While
probably a fair estimate for smaller machines, large machines with large
numbers of CPUs and large numbers of filesystems/block devices may benefit
from larger numbers of threads working on different block devices.

Even if the background flushing algorithm is radically changed, it is
still likely that multiple threads will be involved and admins would still
desire finer control on the min/max other than to have to recompile the
kernel.

The patch adds '/proc/sys/vm/nr_pdflush_threads_min' and
'/proc/sys/vm/nr_pdflush_threads_max' with r/w permissions.

The minimum value for nr_pdflush_threads_min is 1 and the maximum value is
the current value of nr_pdflush_threads_max.  This minimum is required
since additional thread creation is performed in a pdflush thread itself.

The minimum value for nr_pdflush_threads_max is the current value of
nr_pdflush_threads_min and the maximum value can be 1000.

Documentation/sysctl/vm.txt is also updated.

[akpm@linux-foundation.org: fix comment, fix whitespace, use __read_mostly]
Signed-off-by: default avatarPeter W Morreale <pmorreale@novell.com>
Reviewed-by: default avatarRik van Riel <riel@redhat.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent a56ed663
...@@ -39,6 +39,8 @@ Currently, these files are in /proc/sys/vm: ...@@ -39,6 +39,8 @@ Currently, these files are in /proc/sys/vm:
- nr_hugepages - nr_hugepages
- nr_overcommit_hugepages - nr_overcommit_hugepages
- nr_pdflush_threads - nr_pdflush_threads
- nr_pdflush_threads_min
- nr_pdflush_threads_max
- nr_trim_pages (only if CONFIG_MMU=n) - nr_trim_pages (only if CONFIG_MMU=n)
- numa_zonelist_order - numa_zonelist_order
- oom_dump_tasks - oom_dump_tasks
...@@ -463,6 +465,32 @@ The default value is 0. ...@@ -463,6 +465,32 @@ The default value is 0.
============================================================== ==============================================================
nr_pdflush_threads_min
This value controls the minimum number of pdflush threads.
At boot time, the kernel will create and maintain 'nr_pdflush_threads_min'
threads for the kernel's lifetime.
The default value is 2. The minimum value you can specify is 1, and
the maximum value is the current setting of 'nr_pdflush_threads_max'.
See 'nr_pdflush_threads_max' below for more information.
==============================================================
nr_pdflush_threads_max
This value controls the maximum number of pdflush threads that can be
created. The pdflush algorithm will create a new pdflush thread (up to
this maximum) if no pdflush threads have been available for >= 1 second.
The default value is 8. The minimum value you can specify is the
current value of 'nr_pdflush_threads_min' and the
maximum is 1000.
==============================================================
overcommit_memory: overcommit_memory:
This value contains a flag that enables memory overcommitment. This value contains a flag that enables memory overcommitment.
......
...@@ -168,6 +168,8 @@ void writeback_set_ratelimit(void); ...@@ -168,6 +168,8 @@ void writeback_set_ratelimit(void);
/* pdflush.c */ /* pdflush.c */
extern int nr_pdflush_threads; /* Global so it can be exported to sysctl extern int nr_pdflush_threads; /* Global so it can be exported to sysctl
read-only. */ read-only. */
extern int nr_pdflush_threads_max; /* Global so it can be exported to sysctl */
extern int nr_pdflush_threads_min; /* Global so it can be exported to sysctl */
#endif /* WRITEBACK_H */ #endif /* WRITEBACK_H */
...@@ -101,6 +101,7 @@ static int __maybe_unused one = 1; ...@@ -101,6 +101,7 @@ static int __maybe_unused one = 1;
static int __maybe_unused two = 2; static int __maybe_unused two = 2;
static unsigned long one_ul = 1; static unsigned long one_ul = 1;
static int one_hundred = 100; static int one_hundred = 100;
static int one_thousand = 1000;
/* this is needed for the proc_dointvec_minmax for [fs_]overflow UID and GID */ /* this is needed for the proc_dointvec_minmax for [fs_]overflow UID and GID */
static int maxolduid = 65535; static int maxolduid = 65535;
...@@ -1026,6 +1027,28 @@ static struct ctl_table vm_table[] = { ...@@ -1026,6 +1027,28 @@ static struct ctl_table vm_table[] = {
.mode = 0444 /* read-only*/, .mode = 0444 /* read-only*/,
.proc_handler = &proc_dointvec, .proc_handler = &proc_dointvec,
}, },
{
.ctl_name = CTL_UNNUMBERED,
.procname = "nr_pdflush_threads_min",
.data = &nr_pdflush_threads_min,
.maxlen = sizeof nr_pdflush_threads_min,
.mode = 0644 /* read-write */,
.proc_handler = &proc_dointvec_minmax,
.strategy = &sysctl_intvec,
.extra1 = &one,
.extra2 = &nr_pdflush_threads_max,
},
{
.ctl_name = CTL_UNNUMBERED,
.procname = "nr_pdflush_threads_max",
.data = &nr_pdflush_threads_max,
.maxlen = sizeof nr_pdflush_threads_max,
.mode = 0644 /* read-write */,
.proc_handler = &proc_dointvec_minmax,
.strategy = &sysctl_intvec,
.extra1 = &nr_pdflush_threads_min,
.extra2 = &one_thousand,
},
{ {
.ctl_name = VM_SWAPPINESS, .ctl_name = VM_SWAPPINESS,
.procname = "swappiness", .procname = "swappiness",
......
...@@ -57,6 +57,14 @@ static DEFINE_SPINLOCK(pdflush_lock); ...@@ -57,6 +57,14 @@ static DEFINE_SPINLOCK(pdflush_lock);
*/ */
int nr_pdflush_threads = 0; int nr_pdflush_threads = 0;
/*
* The max/min number of pdflush threads. R/W by sysctl at
* /proc/sys/vm/nr_pdflush_threads_max/min
*/
int nr_pdflush_threads_max __read_mostly = MAX_PDFLUSH_THREADS;
int nr_pdflush_threads_min __read_mostly = MIN_PDFLUSH_THREADS;
/* /*
* The time at which the pdflush thread pool last went empty * The time at which the pdflush thread pool last went empty
*/ */
...@@ -68,7 +76,7 @@ static unsigned long last_empty_jifs; ...@@ -68,7 +76,7 @@ static unsigned long last_empty_jifs;
* Thread pool management algorithm: * Thread pool management algorithm:
* *
* - The minimum and maximum number of pdflush instances are bound * - The minimum and maximum number of pdflush instances are bound
* by MIN_PDFLUSH_THREADS and MAX_PDFLUSH_THREADS. * by nr_pdflush_threads_min and nr_pdflush_threads_max.
* *
* - If there have been no idle pdflush instances for 1 second, create * - If there have been no idle pdflush instances for 1 second, create
* a new one. * a new one.
...@@ -134,8 +142,8 @@ static int __pdflush(struct pdflush_work *my_work) ...@@ -134,8 +142,8 @@ static int __pdflush(struct pdflush_work *my_work)
* To throttle creation, we reset last_empty_jifs. * To throttle creation, we reset last_empty_jifs.
*/ */
if (time_after(jiffies, last_empty_jifs + 1 * HZ)) { if (time_after(jiffies, last_empty_jifs + 1 * HZ)) {
if (list_empty(&pdflush_list)) { if (list_empty(&pdflush_list) &&
if (nr_pdflush_threads < MAX_PDFLUSH_THREADS) { nr_pdflush_threads < nr_pdflush_threads_max) {
last_empty_jifs = jiffies; last_empty_jifs = jiffies;
nr_pdflush_threads++; nr_pdflush_threads++;
spin_unlock_irq(&pdflush_lock); spin_unlock_irq(&pdflush_lock);
...@@ -143,7 +151,6 @@ static int __pdflush(struct pdflush_work *my_work) ...@@ -143,7 +151,6 @@ static int __pdflush(struct pdflush_work *my_work)
spin_lock_irq(&pdflush_lock); spin_lock_irq(&pdflush_lock);
} }
} }
}
my_work->fn = NULL; my_work->fn = NULL;
...@@ -153,7 +160,7 @@ static int __pdflush(struct pdflush_work *my_work) ...@@ -153,7 +160,7 @@ static int __pdflush(struct pdflush_work *my_work)
*/ */
if (list_empty(&pdflush_list)) if (list_empty(&pdflush_list))
continue; continue;
if (nr_pdflush_threads <= MIN_PDFLUSH_THREADS) if (nr_pdflush_threads <= nr_pdflush_threads_min)
continue; continue;
pdf = list_entry(pdflush_list.prev, struct pdflush_work, list); pdf = list_entry(pdflush_list.prev, struct pdflush_work, list);
if (time_after(jiffies, pdf->when_i_went_to_sleep + 1 * HZ)) { if (time_after(jiffies, pdf->when_i_went_to_sleep + 1 * HZ)) {
...@@ -259,9 +266,9 @@ static int __init pdflush_init(void) ...@@ -259,9 +266,9 @@ static int __init pdflush_init(void)
* Pre-set nr_pdflush_threads... If we fail to create, * Pre-set nr_pdflush_threads... If we fail to create,
* the count will be decremented. * the count will be decremented.
*/ */
nr_pdflush_threads = MIN_PDFLUSH_THREADS; nr_pdflush_threads = nr_pdflush_threads_min;
for (i = 0; i < MIN_PDFLUSH_THREADS; i++) for (i = 0; i < nr_pdflush_threads_min; i++)
start_one_pdflush_thread(); start_one_pdflush_thread();
return 0; return 0;
} }
......
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