diff options
Diffstat (limited to 'fs/xfs/xfs_iget.c')
-rw-r--r-- | fs/xfs/xfs_iget.c | 32 |
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 | ||
123 | out_error: | 136 | out_error: |
124 | read_unlock(&pag->pag_ici_lock); | 137 | read_unlock(&pag->pag_ici_lock); |
125 | out: | ||
126 | return error; | 138 | return error; |
127 | } | 139 | } |
128 | 140 | ||