* [PATCH 1/2] make shrinker_sem an rwsem
@ 2004-07-28 11:02 Nick Piggin
0 siblings, 0 replies; only message in thread
From: Nick Piggin @ 2004-07-28 11:02 UTC (permalink / raw)
To: Andrew Morton; +Cc: Linux Memory Management
[-- Attachment #1: Type: text/plain, Size: 134 bytes --]
Making shrinker_sem an rwsem allows tasks in shrink_slab to be preempted
or cond_resched'ed without slab reclaim grinding to a halt.
[-- Attachment #2: vm-shrink-slab-fix.patch --]
[-- Type: text/x-patch, Size: 4113 bytes --]
Use an rwsem to protect the shrinker list instead of a regular semaphore.
Modifications to the list are now done under the write lock, shrink_slab
takes the read lock, and access to shrinker->nr becomes racy (which is no
different to how we scan zones). The shrinker->shrinker function also becomes
concurrent.
Previously, having the slab scanner get preempted or scheduling while holding
the semaphore would cause other tasks to skip putting pressure on the slab.
Also, make shrink_icache_memory return -1 if it can't do anything in order to
hold pressure on this cache and prevent useless looping in shrink_slab.
Signed-off-by: Nick Piggin <nickpiggin@yahoo.com.au>
---
linux-2.6-npiggin/fs/inode.c | 5 +++--
linux-2.6-npiggin/mm/vmscan.c | 39 +++++++++++++++++++++++----------------
2 files changed, 26 insertions(+), 18 deletions(-)
diff -puN mm/vmscan.c~vm-shrink-slab-fix mm/vmscan.c
--- linux-2.6/mm/vmscan.c~vm-shrink-slab-fix 2004-07-28 20:52:20.000000000 +1000
+++ linux-2.6-npiggin/mm/vmscan.c 2004-07-28 21:02:05.000000000 +1000
@@ -32,6 +32,7 @@
#include <linux/topology.h>
#include <linux/cpu.h>
#include <linux/notifier.h>
+#include <linux/rwsem.h>
#include <asm/tlbflush.h>
#include <asm/div64.h>
@@ -122,7 +123,7 @@ int vm_swappiness = 60;
static long total_memory;
static LIST_HEAD(shrinker_list);
-static DECLARE_MUTEX(shrinker_sem);
+static DECLARE_RWSEM(shrinker_rwsem);
/*
* Add a shrinker callback to be called from the vm
@@ -136,9 +137,9 @@ struct shrinker *set_shrinker(int seeks,
shrinker->shrinker = theshrinker;
shrinker->seeks = seeks;
shrinker->nr = 0;
- down(&shrinker_sem);
+ down_write(&shrinker_rwsem);
list_add(&shrinker->list, &shrinker_list);
- up(&shrinker_sem);
+ up_write(&shrinker_rwsem);
}
return shrinker;
}
@@ -149,13 +150,13 @@ EXPORT_SYMBOL(set_shrinker);
*/
void remove_shrinker(struct shrinker *shrinker)
{
- down(&shrinker_sem);
+ down_write(&shrinker_rwsem);
list_del(&shrinker->list);
- up(&shrinker_sem);
+ up_write(&shrinker_rwsem);
kfree(shrinker);
}
EXPORT_SYMBOL(remove_shrinker);
-
+
#define SHRINK_BATCH 128
/*
* Call the shrink functions to age shrinkable caches
@@ -175,12 +176,16 @@ static int shrink_slab(unsigned long sca
struct shrinker *shrinker;
long pages;
- if (down_trylock(&shrinker_sem))
+ if (scanned == 0)
+ return;
+
+ if (!down_read_trylock(&shrinker_rwsem))
return 0;
pages = nr_used_zone_pages();
list_for_each_entry(shrinker, &shrinker_list, list) {
unsigned long long delta;
+ unsigned long total_scan;
delta = (4 * scanned) / shrinker->seeks;
delta *= (*shrinker->shrinker)(0, gfp_mask);
@@ -189,23 +194,25 @@ static int shrink_slab(unsigned long sca
if (shrinker->nr < 0)
shrinker->nr = LONG_MAX; /* It wrapped! */
- if (shrinker->nr <= SHRINK_BATCH)
- continue;
- while (shrinker->nr) {
- long this_scan = shrinker->nr;
+ total_scan = shrinker->nr;
+ shrinker->nr = 0;
+
+ while (total_scan >= SHRINK_BATCH) {
+ long this_scan = SHRINK_BATCH;
int shrink_ret;
- if (this_scan > 128)
- this_scan = 128;
shrink_ret = (*shrinker->shrinker)(this_scan, gfp_mask);
- mod_page_state(slabs_scanned, this_scan);
- shrinker->nr -= this_scan;
if (shrink_ret == -1)
break;
+ mod_page_state(slabs_scanned, this_scan);
+ total_scan -= this_scan;
+
cond_resched();
}
+
+ shrinker->nr += total_scan;
}
- up(&shrinker_sem);
+ up_read(&shrinker_rwsem);
return 0;
}
diff -puN fs/inode.c~vm-shrink-slab-fix fs/inode.c
--- linux-2.6/fs/inode.c~vm-shrink-slab-fix 2004-07-28 20:52:20.000000000 +1000
+++ linux-2.6-npiggin/fs/inode.c 2004-07-28 20:52:20.000000000 +1000
@@ -485,8 +485,9 @@ static int shrink_icache_memory(int nr,
* and we don't want to recurse into the FS that called us
* in clear_inode() and friends..
*/
- if (gfp_mask & __GFP_FS)
- prune_icache(nr);
+ if (!(gfp_mask & __GFP_FS))
+ return -1;
+ prune_icache(nr);
}
return (inodes_stat.nr_unused / 100) * sysctl_vfs_cache_pressure;
}
_
^ permalink raw reply [flat|nested] only message in thread
only message in thread, other threads:[~2004-07-28 11:02 UTC | newest]
Thread overview: (only message) (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2004-07-28 11:02 [PATCH 1/2] make shrinker_sem an rwsem Nick Piggin
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox