On Tue, May 30, 2006 at 12:06:33PM +0200, Jan Blunck wrote: > On Tue, May 30, David Chinner wrote: > > > You've just described the embodiment of the two order's of magnitude > > issue I mentioned. That's not a wrong assumption - think of the > > above case with global_unused count now being 1.28*10^7 instead of > > 1.28x10^4. How many dentries do you have to free before freeing any > > on the small superblock if we don't free one per call? (quick > > answer: 99.9%). > > > > If we shrink one per call, we've freed all 128 dentries while there > > is still 1*10^5 dentries on the large list. That seems like a much > > better balance to make within the constraints of the shrinker > > resolution we have to work with. > > With the effect that the dcache is completely useless for small filesystems > as long as there is one big one. Not necessarily. I think that as long as the small filesystem is not being used, then we _should_ be reclaiming slowly from it, regardless of how big the other filesystems are. That's the way the global list ends up working now as the dentries for the small filesystem get purged according to LRU. > Filesystems where regularily a small amount > of files is used don't have any cached dentries but the filesystem where > someone touched every file still has a lot of dentries in cache although they > are never used again. Or alternatively small filesystems with no activity and dentries that will never get used again never get trimmed while the large fielsytem with lots of activity gets trimmed. This can lead to thousands of pages being pinned in slabs that we don't try to free up until we've already free >90% of the overall caches. That's not very appealing, IMO. So I suggest the question is this - how do you define "not being used"? I guess dentry age or "last dentry was added to list at time X" (recorded in dput()) would be one way of determining this. Say something like: dput(): list_add(&dentry->d_lru, &dentry->d_sb->s_unused); dentry->s_sb->s_unused_age = jiffies + (&dentry->d_sb->s_dentry_stat.age_limit * HZ); prune_dcache(): tmp = sb->s_dentry_stat.nr_unused/((unused/count)+1); if ((tmp == 0) && time_after(jiffies, sb->s_unused_age)) tmp = 1; That would turn over small unused dentry lists that have not been modified for age_limit seconds. That means that small caches that are being used are not reclaimed prematurely, and those small caches would also be reclaimed after some time if they are not being used. That seems to address both our concerns..... Thoughts? Cheers, Dave. -- Dave Chinner R&D Software Enginner SGI Australian Software Group - To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html