Files
linux-apfs/fs/drop_caches.c
T

57 lines
1.3 KiB
C
Raw Normal View History

2006-01-08 01:00:39 -08:00
/*
* Implement the manual drop-all-pagecache function
*/
#include <linux/kernel.h>
#include <linux/mm.h>
#include <linux/fs.h>
#include <linux/writeback.h>
#include <linux/sysctl.h>
#include <linux/gfp.h>
/* A global variable is a bit ugly, but it keeps the code simple */
int sysctl_drop_caches;
2010-03-23 06:06:58 -04:00
static void drop_pagecache_sb(struct super_block *sb, void *unused)
2006-01-08 01:00:39 -08:00
{
struct inode *inode, *toput_inode = NULL;
2006-01-08 01:00:39 -08:00
spin_lock(&inode_lock);
list_for_each_entry(inode, &sb->s_inodes, i_sb_list) {
2010-06-02 17:38:30 -04:00
if (inode->i_state & (I_FREEING|I_WILL_FREE|I_NEW))
2006-01-08 01:00:39 -08:00
continue;
if (inode->i_mapping->nrpages == 0)
continue;
__iget(inode);
spin_unlock(&inode_lock);
invalidate_mapping_pages(inode->i_mapping, 0, -1);
iput(toput_inode);
toput_inode = inode;
spin_lock(&inode_lock);
2006-01-08 01:00:39 -08:00
}
spin_unlock(&inode_lock);
iput(toput_inode);
2006-01-08 01:00:39 -08:00
}
2008-04-29 00:58:57 -07:00
static void drop_slab(void)
2006-01-08 01:00:39 -08:00
{
int nr_objects;
do {
nr_objects = shrink_slab(1000, GFP_KERNEL, 1000);
} while (nr_objects > 10);
}
int drop_caches_sysctl_handler(ctl_table *table, int write,
void __user *buffer, size_t *length, loff_t *ppos)
2006-01-08 01:00:39 -08:00
{
proc_dointvec_minmax(table, write, buffer, length, ppos);
2006-01-08 01:00:39 -08:00
if (write) {
if (sysctl_drop_caches & 1)
2010-03-23 06:06:58 -04:00
iterate_supers(drop_pagecache_sb, NULL);
2006-01-08 01:00:39 -08:00
if (sysctl_drop_caches & 2)
drop_slab();
}
return 0;
}