dcache.c

来自「基于组件方式开发操作系统的OSKIT源代码」· C语言 代码 · 共 931 行 · 第 1/2 页

C
931
字号
/* * fs/dcache.c * * Complete reimplementation * (C) 1997 Thomas Schoebel-Theuer, * with heavy changes by Linus Torvalds *//* * Notes on the allocation strategy: * * The dcache is a master of the icache - whenever a dcache entry * exists, the inode will always exist. "iput()" is done either when * the dcache entry is deleted or garbage collected. */#include <linux/string.h>#include <linux/mm.h>#include <linux/fs.h>#include <linux/malloc.h>#include <linux/slab.h>#include <linux/init.h>#include <asm/uaccess.h>#define DCACHE_PARANOIA 1/* #define DCACHE_DEBUG 1 *//* For managing the dcache */extern unsigned long num_physpages, page_cache_size;extern int inodes_stat[];#define nr_inodes (inodes_stat[0])kmem_cache_t *dentry_cache; /* * This is the single most critical data structure when it comes * to the dcache: the hashtable for lookups. Somebody should try * to make this good - I've just made it work. * * This hash-function tries to avoid losing too many bits of hash * information, yet avoid using a prime hash-size or similar. */#define D_HASHBITS     10#define D_HASHSIZE     (1UL << D_HASHBITS)#define D_HASHMASK     (D_HASHSIZE-1)static struct list_head dentry_hashtable[D_HASHSIZE];static LIST_HEAD(dentry_unused);struct {	int nr_dentry;	int nr_unused;	int age_limit;		/* age in seconds */	int want_pages;		/* pages requested by system */	int dummy[2];} dentry_stat = {0, 0, 45, 0,};static inline void d_free(struct dentry *dentry){	if (dentry->d_op && dentry->d_op->d_release)		dentry->d_op->d_release(dentry);	if (dname_external(dentry)) 		kfree(dentry->d_name.name);	kmem_cache_free(dentry_cache, dentry); }/* * Release the dentry's inode, using the fileystem * d_iput() operation if defined. */static inline void dentry_iput(struct dentry * dentry){	struct inode *inode = dentry->d_inode;	if (inode) {		dentry->d_inode = NULL;		list_del(&dentry->d_alias);		INIT_LIST_HEAD(&dentry->d_alias);		if (dentry->d_op && dentry->d_op->d_iput)			dentry->d_op->d_iput(dentry, inode);		else			iput(inode);	}}/* * dput() * * This is complicated by the fact that we do not want to put * dentries that are no longer on any hash chain on the unused * list: we'd much rather just get rid of them immediately. * * However, that implies that we have to traverse the dentry * tree upwards to the parents which might _also_ now be * scheduled for deletion (it may have been only waiting for * its last child to go away). * * This tail recursion is done by hand as we don't want to depend * on the compiler to always get this right (gcc generally doesn't). * Real recursion would eat up our stack space. */void dput(struct dentry *dentry){	int count;	if (!dentry)		return;repeat:	count = dentry->d_count - 1;	if (count != 0)		goto out;	/*	 * Note that if d_op->d_delete blocks,	 * the dentry could go back in use.	 * Each fs will have to watch for this.	 */	if (dentry->d_op && dentry->d_op->d_delete) {		dentry->d_op->d_delete(dentry);		count = dentry->d_count - 1;		if (count != 0)			goto out;	}	if (!list_empty(&dentry->d_lru)) {		dentry_stat.nr_unused--;		list_del(&dentry->d_lru);	}	if (list_empty(&dentry->d_hash)) {		struct dentry * parent;		list_del(&dentry->d_child);		dentry_iput(dentry);		parent = dentry->d_parent;		d_free(dentry);		if (dentry == parent)			return;		dentry = parent;		goto repeat;	}	list_add(&dentry->d_lru, &dentry_unused);	dentry_stat.nr_unused++;	/*	 * Update the timestamp	 */	dentry->d_reftime = jiffies;out:	if (count >= 0) {		dentry->d_count = count;		return;	}	printk(KERN_CRIT "Negative d_count (%d) for %s/%s\n",		count,		dentry->d_parent->d_name.name,		dentry->d_name.name);	*(int *)0 = 0;	}/* * Try to invalidate the dentry if it turns out to be * possible. If there are other dentries that can be * reached through this one we can't delete it. */int d_invalidate(struct dentry * dentry){	/*	 * Check whether to do a partial shrink_dcache	 * to get rid of unused child entries.	 */	if (!list_empty(&dentry->d_subdirs)) {		shrink_dcache_parent(dentry);	}	/*	 * Somebody else still using it?	 *	 * If it's a directory, we can't drop it	 * for fear of somebody re-populating it	 * with children (even though dropping it	 * would make it unreachable from the root,	 * we might still populate it if it was a	 * working directory or similar).	 */	if (dentry->d_count > 1) {		if (dentry->d_inode && S_ISDIR(dentry->d_inode->i_mode))			return -EBUSY;	}	d_drop(dentry);	return 0;}/* * Select less valuable dentries to be pruned when we need * inodes or memory. The selected dentries are moved to the * old end of the list where prune_dcache() can find them. *  * Negative dentries are included in the selection so that * they don't accumulate at the end of the list. The count * returned is the total number of dentries selected, which * may be much larger than the requested number of inodes. */int select_dcache(int inode_count, int page_count){	struct list_head *next, *tail = &dentry_unused;	int found = 0;	int depth = dentry_stat.nr_unused >> 1;	unsigned long max_value = 4;	if (page_count)		max_value = -1;	next = tail->prev;	while (next != &dentry_unused && depth--) {		struct list_head *tmp = next;		struct dentry *dentry = list_entry(tmp, struct dentry, d_lru);		struct inode *inode = dentry->d_inode;		unsigned long value = 0;			next = tmp->prev;		if (dentry->d_count) {			dentry_stat.nr_unused--;			list_del(tmp);			INIT_LIST_HEAD(tmp);			continue;		}		/*		 * Select dentries based on the page cache count ...		 * should factor in number of uses as well. We take		 * all negative dentries so that they don't accumulate.		 * (We skip inodes that aren't immediately available.)		 */		if (inode) {			value = inode->i_nrpages;				if (value >= max_value)				continue;			if (inode->i_state || inode->i_count > 1)				continue;		}		/*		 * Move the selected dentries behind the tail.		 */		if (tmp != tail->prev) {			list_del(tmp);			list_add(tmp, tail->prev);		}		tail = tmp;		found++;		if (inode && --inode_count <= 0)			break;		if (page_count && (page_count -= value) <= 0)			break;	}	return found;}/* * Throw away a dentry - free the inode, dput the parent. * This requires that the LRU list has already been * removed. */static inline void prune_one_dentry(struct dentry * dentry){	struct dentry * parent;	list_del(&dentry->d_hash);	list_del(&dentry->d_child);	dentry_iput(dentry);	parent = dentry->d_parent;	d_free(dentry);	dput(parent);}/* * Shrink the dcache. This is done when we need * more memory, or simply when we need to unmount * something (at which point we need to unuse * all dentries). */void prune_dcache(int count){	for (;;) {		struct dentry *dentry;		struct list_head *tmp = dentry_unused.prev;		if (tmp == &dentry_unused)			break;		dentry_stat.nr_unused--;		list_del(tmp);		INIT_LIST_HEAD(tmp);		dentry = list_entry(tmp, struct dentry, d_lru);		if (!dentry->d_count) {			prune_one_dentry(dentry);			if (!--count)				break;		}	}}/* * Shrink the dcache for the specified super block. * This allows us to unmount a device without disturbing * the dcache for the other devices. * * This implementation makes just two traversals of the * unused list.  On the first pass we move the selected * dentries to the most recent end, and on the second * pass we free them.  The second pass must restart after * each dput(), but since the target dentries are all at * the end, it's really just a single traversal. */void shrink_dcache_sb(struct super_block * sb){	struct list_head *tmp, *next;	struct dentry *dentry;	/*	 * Pass one ... move the dentries for the specified	 * superblock to the most recent end of the unused list.	 */	next = dentry_unused.next;	while (next != &dentry_unused) {		tmp = next;		next = tmp->next;		dentry = list_entry(tmp, struct dentry, d_lru);		if (dentry->d_sb != sb)			continue;		list_del(tmp);		list_add(tmp, &dentry_unused);	}	/*	 * Pass two ... free the dentries for this superblock.	 */repeat:	next = dentry_unused.next;	while (next != &dentry_unused) {		tmp = next;		next = tmp->next;		dentry = list_entry(tmp, struct dentry, d_lru);		if (dentry->d_sb != sb)			continue;		if (dentry->d_count)			continue;		dentry_stat.nr_unused--;		list_del(tmp);		INIT_LIST_HEAD(tmp);		prune_one_dentry(dentry);		goto repeat;	}}/* * Check whether a root dentry would be in use if all of its * child dentries were freed. This allows a non-destructive * test for unmounting a device. */int is_root_busy(struct dentry *root){	struct dentry *this_parent = root;	struct list_head *next;	int count = root->d_count;repeat:	next = this_parent->d_subdirs.next;resume:	while (next != &this_parent->d_subdirs) {		struct list_head *tmp = next;		struct dentry *dentry = list_entry(tmp, struct dentry, d_child);		next = tmp->next;		/* Decrement count for unused children */		count += (dentry->d_count - 1);		if (!list_empty(&dentry->d_subdirs)) {			this_parent = dentry;			goto repeat;		}		/* root is busy if any leaf is busy */		if (dentry->d_count)			return 1;	}	/*	 * All done at this level ... ascend and resume the search.	 */	if (this_parent != root) {		next = this_parent->d_child.next; 		this_parent = this_parent->d_parent;		goto resume;	}	return (count > 1); /* remaining users? */}/* * Search the dentry child list for the specified parent, * and move any unused dentries to the end of the unused * list for prune_dcache(). We descend to the next level * whenever the d_subdirs list is non-empty and continue * searching. */static int select_parent(struct dentry * parent){	struct dentry *this_parent = parent;	struct list_head *next;	int found = 0;repeat:	next = this_parent->d_subdirs.next;resume:	while (next != &this_parent->d_subdirs) {		struct list_head *tmp = next;		struct dentry *dentry = list_entry(tmp, struct dentry, d_child);		next = tmp->next;		if (!dentry->d_count) {			list_del(&dentry->d_lru);			list_add(&dentry->d_lru, dentry_unused.prev);			found++;		}		/*		 * Descend a level if the d_subdirs list is non-empty.		 */		if (!list_empty(&dentry->d_subdirs)) {			this_parent = dentry;#ifdef DCACHE_DEBUGprintk(KERN_DEBUG "select_parent: descending to %s/%s, found=%d\n",dentry->d_parent->d_name.name, dentry->d_name.name, found);#endif			goto repeat;		}	}	/*	 * All done at this level ... ascend and resume the search.	 */	if (this_parent != parent) {		next = this_parent->d_child.next; 		this_parent = this_parent->d_parent;#ifdef DCACHE_DEBUGprintk(KERN_DEBUG "select_parent: ascending to %s/%s, found=%d\n",this_parent->d_parent->d_name.name, this_parent->d_name.name, found);#endif		goto resume;	}	return found;}/* * Prune the dcache to remove unused children of the parent dentry. */void shrink_dcache_parent(struct dentry * parent){	int found;	while ((found = select_parent(parent)) != 0)		prune_dcache(found);}/* * This is called from kswapd when we think we need some * more memory, but aren't really sure how much. So we * carefully try to free a _bit_ of our dcache, but not * too much. *

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?