aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--fs/dcache.c73
-rw-r--r--include/linux/dcache.h35
2 files changed, 64 insertions, 44 deletions
diff --git a/fs/dcache.c b/fs/dcache.c
index 61beb40dd6bf..1e124d4673c6 100644
--- a/fs/dcache.c
+++ b/fs/dcache.c
@@ -35,10 +35,24 @@
35#include <linux/hardirq.h> 35#include <linux/hardirq.h>
36#include "internal.h" 36#include "internal.h"
37 37
38/*
39 * Usage:
40 * dcache_hash_lock protects dcache hash table, s_anon lists
41 *
42 * Ordering:
43 * dcache_lock
44 * dentry->d_lock
45 * dcache_hash_lock
46 *
47 * if (dentry1 < dentry2)
48 * dentry1->d_lock
49 * dentry2->d_lock
50 */
38int sysctl_vfs_cache_pressure __read_mostly = 100; 51int sysctl_vfs_cache_pressure __read_mostly = 100;
39EXPORT_SYMBOL_GPL(sysctl_vfs_cache_pressure); 52EXPORT_SYMBOL_GPL(sysctl_vfs_cache_pressure);
40 53
41 __cacheline_aligned_in_smp DEFINE_SPINLOCK(dcache_lock); 54static __cacheline_aligned_in_smp DEFINE_SPINLOCK(dcache_hash_lock);
55__cacheline_aligned_in_smp DEFINE_SPINLOCK(dcache_lock);
42__cacheline_aligned_in_smp DEFINE_SEQLOCK(rename_lock); 56__cacheline_aligned_in_smp DEFINE_SEQLOCK(rename_lock);
43 57
44EXPORT_SYMBOL(dcache_lock); 58EXPORT_SYMBOL(dcache_lock);
@@ -196,6 +210,42 @@ static struct dentry *d_kill(struct dentry *dentry)
196 return parent; 210 return parent;
197} 211}
198 212
213/**
214 * d_drop - drop a dentry
215 * @dentry: dentry to drop
216 *
217 * d_drop() unhashes the entry from the parent dentry hashes, so that it won't
218 * be found through a VFS lookup any more. Note that this is different from
219 * deleting the dentry - d_delete will try to mark the dentry negative if
220 * possible, giving a successful _negative_ lookup, while d_drop will
221 * just make the cache lookup fail.
222 *
223 * d_drop() is used mainly for stuff that wants to invalidate a dentry for some
224 * reason (NFS timeouts or autofs deletes).
225 *
226 * __d_drop requires dentry->d_lock.
227 */
228void __d_drop(struct dentry *dentry)
229{
230 if (!(dentry->d_flags & DCACHE_UNHASHED)) {
231 dentry->d_flags |= DCACHE_UNHASHED;
232 spin_lock(&dcache_hash_lock);
233 hlist_del_rcu(&dentry->d_hash);
234 spin_unlock(&dcache_hash_lock);
235 }
236}
237EXPORT_SYMBOL(__d_drop);
238
239void d_drop(struct dentry *dentry)
240{
241 spin_lock(&dcache_lock);
242 spin_lock(&dentry->d_lock);
243 __d_drop(dentry);
244 spin_unlock(&dentry->d_lock);
245 spin_unlock(&dcache_lock);
246}
247EXPORT_SYMBOL(d_drop);
248
199/* 249/*
200 * This is dput 250 * This is dput
201 * 251 *
@@ -1199,7 +1249,9 @@ struct dentry *d_obtain_alias(struct inode *inode)
1199 tmp->d_flags |= DCACHE_DISCONNECTED; 1249 tmp->d_flags |= DCACHE_DISCONNECTED;
1200 tmp->d_flags &= ~DCACHE_UNHASHED; 1250 tmp->d_flags &= ~DCACHE_UNHASHED;
1201 list_add(&tmp->d_alias, &inode->i_dentry); 1251 list_add(&tmp->d_alias, &inode->i_dentry);
1252 spin_lock(&dcache_hash_lock);
1202 hlist_add_head(&tmp->d_hash, &inode->i_sb->s_anon); 1253 hlist_add_head(&tmp->d_hash, &inode->i_sb->s_anon);
1254 spin_unlock(&dcache_hash_lock);
1203 spin_unlock(&tmp->d_lock); 1255 spin_unlock(&tmp->d_lock);
1204 1256
1205 spin_unlock(&dcache_lock); 1257 spin_unlock(&dcache_lock);
@@ -1585,7 +1637,9 @@ void d_rehash(struct dentry * entry)
1585{ 1637{
1586 spin_lock(&dcache_lock); 1638 spin_lock(&dcache_lock);
1587 spin_lock(&entry->d_lock); 1639 spin_lock(&entry->d_lock);
1640 spin_lock(&dcache_hash_lock);
1588 _d_rehash(entry); 1641 _d_rehash(entry);
1642 spin_unlock(&dcache_hash_lock);
1589 spin_unlock(&entry->d_lock); 1643 spin_unlock(&entry->d_lock);
1590 spin_unlock(&dcache_lock); 1644 spin_unlock(&dcache_lock);
1591} 1645}
@@ -1692,8 +1746,6 @@ static void switch_names(struct dentry *dentry, struct dentry *target)
1692 */ 1746 */
1693static void d_move_locked(struct dentry * dentry, struct dentry * target) 1747static void d_move_locked(struct dentry * dentry, struct dentry * target)
1694{ 1748{
1695 struct hlist_head *list;
1696
1697 if (!dentry->d_inode) 1749 if (!dentry->d_inode)
1698 printk(KERN_WARNING "VFS: moving negative dcache entry\n"); 1750 printk(KERN_WARNING "VFS: moving negative dcache entry\n");
1699 1751
@@ -1710,14 +1762,11 @@ static void d_move_locked(struct dentry * dentry, struct dentry * target)
1710 } 1762 }
1711 1763
1712 /* Move the dentry to the target hash queue, if on different bucket */ 1764 /* Move the dentry to the target hash queue, if on different bucket */
1713 if (d_unhashed(dentry)) 1765 spin_lock(&dcache_hash_lock);
1714 goto already_unhashed; 1766 if (!d_unhashed(dentry))
1715 1767 hlist_del_rcu(&dentry->d_hash);
1716 hlist_del_rcu(&dentry->d_hash); 1768 __d_rehash(dentry, d_hash(target->d_parent, target->d_name.hash));
1717 1769 spin_unlock(&dcache_hash_lock);
1718already_unhashed:
1719 list = d_hash(target->d_parent, target->d_name.hash);
1720 __d_rehash(dentry, list);
1721 1770
1722 /* Unhash the target: dput() will then get rid of it */ 1771 /* Unhash the target: dput() will then get rid of it */
1723 __d_drop(target); 1772 __d_drop(target);
@@ -1914,7 +1963,9 @@ struct dentry *d_materialise_unique(struct dentry *dentry, struct inode *inode)
1914found_lock: 1963found_lock:
1915 spin_lock(&actual->d_lock); 1964 spin_lock(&actual->d_lock);
1916found: 1965found:
1966 spin_lock(&dcache_hash_lock);
1917 _d_rehash(actual); 1967 _d_rehash(actual);
1968 spin_unlock(&dcache_hash_lock);
1918 spin_unlock(&actual->d_lock); 1969 spin_unlock(&actual->d_lock);
1919 spin_unlock(&dcache_lock); 1970 spin_unlock(&dcache_lock);
1920out_nolock: 1971out_nolock:
diff --git a/include/linux/dcache.h b/include/linux/dcache.h
index cea27dfca532..2feb624b67f1 100644
--- a/include/linux/dcache.h
+++ b/include/linux/dcache.h
@@ -184,39 +184,6 @@ struct dentry_operations {
184extern spinlock_t dcache_lock; 184extern spinlock_t dcache_lock;
185extern seqlock_t rename_lock; 185extern seqlock_t rename_lock;
186 186
187/**
188 * d_drop - drop a dentry
189 * @dentry: dentry to drop
190 *
191 * d_drop() unhashes the entry from the parent dentry hashes, so that it won't
192 * be found through a VFS lookup any more. Note that this is different from
193 * deleting the dentry - d_delete will try to mark the dentry negative if
194 * possible, giving a successful _negative_ lookup, while d_drop will
195 * just make the cache lookup fail.
196 *
197 * d_drop() is used mainly for stuff that wants to invalidate a dentry for some
198 * reason (NFS timeouts or autofs deletes).
199 *
200 * __d_drop requires dentry->d_lock.
201 */
202
203static inline void __d_drop(struct dentry *dentry)
204{
205 if (!(dentry->d_flags & DCACHE_UNHASHED)) {
206 dentry->d_flags |= DCACHE_UNHASHED;
207 hlist_del_rcu(&dentry->d_hash);
208 }
209}
210
211static inline void d_drop(struct dentry *dentry)
212{
213 spin_lock(&dcache_lock);
214 spin_lock(&dentry->d_lock);
215 __d_drop(dentry);
216 spin_unlock(&dentry->d_lock);
217 spin_unlock(&dcache_lock);
218}
219
220static inline int dname_external(struct dentry *dentry) 187static inline int dname_external(struct dentry *dentry)
221{ 188{
222 return dentry->d_name.name != dentry->d_iname; 189 return dentry->d_name.name != dentry->d_iname;
@@ -228,6 +195,8 @@ static inline int dname_external(struct dentry *dentry)
228extern void d_instantiate(struct dentry *, struct inode *); 195extern void d_instantiate(struct dentry *, struct inode *);
229extern struct dentry * d_instantiate_unique(struct dentry *, struct inode *); 196extern struct dentry * d_instantiate_unique(struct dentry *, struct inode *);
230extern struct dentry * d_materialise_unique(struct dentry *, struct inode *); 197extern struct dentry * d_materialise_unique(struct dentry *, struct inode *);
198extern void __d_drop(struct dentry *dentry);
199extern void d_drop(struct dentry *dentry);
231extern void d_delete(struct dentry *); 200extern void d_delete(struct dentry *);
232 201
233/* allocate/de-allocate */ 202/* allocate/de-allocate */