aboutsummaryrefslogtreecommitdiffstats
path: root/fs/xfs/xfs_iget.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/xfs/xfs_iget.c')
-rw-r--r--fs/xfs/xfs_iget.c32
1 files changed, 22 insertions, 10 deletions
diff --git a/fs/xfs/xfs_iget.c b/fs/xfs/xfs_iget.c
index 837cae78153..bf4dc5eb4cf 100644
--- a/fs/xfs/xfs_iget.c
+++ b/fs/xfs/xfs_iget.c
@@ -52,7 +52,7 @@ xfs_iget_cache_hit(
52 int lock_flags) __releases(pag->pag_ici_lock) 52 int lock_flags) __releases(pag->pag_ici_lock)
53{ 53{
54 struct xfs_mount *mp = ip->i_mount; 54 struct xfs_mount *mp = ip->i_mount;
55 int error = 0; 55 int error = EAGAIN;
56 56
57 /* 57 /*
58 * If INEW is set this inode is being set up 58 * If INEW is set this inode is being set up
@@ -60,7 +60,6 @@ xfs_iget_cache_hit(
60 * Pause and try again. 60 * Pause and try again.
61 */ 61 */
62 if (xfs_iflags_test(ip, (XFS_INEW|XFS_IRECLAIM))) { 62 if (xfs_iflags_test(ip, (XFS_INEW|XFS_IRECLAIM))) {
63 error = EAGAIN;
64 XFS_STATS_INC(xs_ig_frecycle); 63 XFS_STATS_INC(xs_ig_frecycle);
65 goto out_error; 64 goto out_error;
66 } 65 }
@@ -73,7 +72,6 @@ xfs_iget_cache_hit(
73 * error immediately so we don't remove it from the reclaim 72 * error immediately so we don't remove it from the reclaim
74 * list and potentially leak the inode. 73 * list and potentially leak the inode.
75 */ 74 */
76
77 if ((ip->i_d.di_mode == 0) && !(flags & XFS_IGET_CREATE)) { 75 if ((ip->i_d.di_mode == 0) && !(flags & XFS_IGET_CREATE)) {
78 error = ENOENT; 76 error = ENOENT;
79 goto out_error; 77 goto out_error;
@@ -91,27 +89,42 @@ xfs_iget_cache_hit(
91 error = ENOMEM; 89 error = ENOMEM;
92 goto out_error; 90 goto out_error;
93 } 91 }
92
93 /*
94 * We must set the XFS_INEW flag before clearing the
95 * XFS_IRECLAIMABLE flag so that if a racing lookup does
96 * not find the XFS_IRECLAIMABLE above but has the igrab()
97 * below succeed we can safely check XFS_INEW to detect
98 * that this inode is still being initialised.
99 */
94 xfs_iflags_set(ip, XFS_INEW); 100 xfs_iflags_set(ip, XFS_INEW);
95 xfs_iflags_clear(ip, XFS_IRECLAIMABLE); 101 xfs_iflags_clear(ip, XFS_IRECLAIMABLE);
96 102
97 /* clear the radix tree reclaim flag as well. */ 103 /* clear the radix tree reclaim flag as well. */
98 __xfs_inode_clear_reclaim_tag(mp, pag, ip); 104 __xfs_inode_clear_reclaim_tag(mp, pag, ip);
99 read_unlock(&pag->pag_ici_lock);
100 } else if (!igrab(VFS_I(ip))) { 105 } else if (!igrab(VFS_I(ip))) {
101 /* If the VFS inode is being torn down, pause and try again. */ 106 /* If the VFS inode is being torn down, pause and try again. */
102 error = EAGAIN;
103 XFS_STATS_INC(xs_ig_frecycle); 107 XFS_STATS_INC(xs_ig_frecycle);
104 goto out_error; 108 goto out_error;
105 } else { 109 } else if (xfs_iflags_test(ip, XFS_INEW)) {
106 /* we've got a live one */ 110 /*
107 read_unlock(&pag->pag_ici_lock); 111 * We are racing with another cache hit that is
112 * currently recycling this inode out of the XFS_IRECLAIMABLE
113 * state. Wait for the initialisation to complete before
114 * continuing.
115 */
116 wait_on_inode(VFS_I(ip));
108 } 117 }
109 118
110 if (ip->i_d.di_mode == 0 && !(flags & XFS_IGET_CREATE)) { 119 if (ip->i_d.di_mode == 0 && !(flags & XFS_IGET_CREATE)) {
111 error = ENOENT; 120 error = ENOENT;
112 goto out; 121 iput(VFS_I(ip));
122 goto out_error;
113 } 123 }
114 124
125 /* We've got a live one. */
126 read_unlock(&pag->pag_ici_lock);
127
115 if (lock_flags != 0) 128 if (lock_flags != 0)
116 xfs_ilock(ip, lock_flags); 129 xfs_ilock(ip, lock_flags);
117 130
@@ -122,7 +135,6 @@ xfs_iget_cache_hit(
122 135
123out_error: 136out_error:
124 read_unlock(&pag->pag_ici_lock); 137 read_unlock(&pag->pag_ici_lock);
125out:
126 return error; 138 return error;
127} 139}
128 140