diff options
-rw-r--r-- | fs/xfs/linux-2.6/xfs_sync.c | 53 | ||||
-rw-r--r-- | fs/xfs/linux-2.6/xfs_vnode.c | 6 | ||||
-rw-r--r-- | fs/xfs/linux-2.6/xfs_vnode.h | 5 | ||||
-rw-r--r-- | fs/xfs/quota/xfs_qm_syscalls.c | 16 |
4 files changed, 37 insertions, 43 deletions
diff --git a/fs/xfs/linux-2.6/xfs_sync.c b/fs/xfs/linux-2.6/xfs_sync.c index 59da3327a6b5..461c1dc35d37 100644 --- a/fs/xfs/linux-2.6/xfs_sync.c +++ b/fs/xfs/linux-2.6/xfs_sync.c | |||
@@ -131,10 +131,7 @@ xfs_sync_inodes_ag( | |||
131 | int flags, | 131 | int flags, |
132 | int *bypassed) | 132 | int *bypassed) |
133 | { | 133 | { |
134 | xfs_inode_t *ip = NULL; | ||
135 | struct inode *vp = NULL; | ||
136 | xfs_perag_t *pag = &mp->m_perag[ag]; | 134 | xfs_perag_t *pag = &mp->m_perag[ag]; |
137 | boolean_t vnode_refed = B_FALSE; | ||
138 | int nr_found; | 135 | int nr_found; |
139 | int first_index = 0; | 136 | int first_index = 0; |
140 | int error = 0; | 137 | int error = 0; |
@@ -156,6 +153,10 @@ xfs_sync_inodes_ag( | |||
156 | } | 153 | } |
157 | 154 | ||
158 | do { | 155 | do { |
156 | struct inode *inode; | ||
157 | boolean_t inode_refed; | ||
158 | xfs_inode_t *ip = NULL; | ||
159 | |||
159 | /* | 160 | /* |
160 | * use a gang lookup to find the next inode in the tree | 161 | * use a gang lookup to find the next inode in the tree |
161 | * as the tree is sparse and a gang lookup walks to find | 162 | * as the tree is sparse and a gang lookup walks to find |
@@ -177,14 +178,14 @@ xfs_sync_inodes_ag( | |||
177 | * skip inodes in reclaim. Let xfs_syncsub do that for | 178 | * skip inodes in reclaim. Let xfs_syncsub do that for |
178 | * us so we don't need to worry. | 179 | * us so we don't need to worry. |
179 | */ | 180 | */ |
180 | vp = VFS_I(ip); | 181 | if (xfs_iflags_test(ip, (XFS_IRECLAIM|XFS_IRECLAIMABLE))) { |
181 | if (!vp) { | ||
182 | read_unlock(&pag->pag_ici_lock); | 182 | read_unlock(&pag->pag_ici_lock); |
183 | continue; | 183 | continue; |
184 | } | 184 | } |
185 | 185 | ||
186 | /* bad inodes are dealt with elsewhere */ | 186 | /* bad inodes are dealt with elsewhere */ |
187 | if (VN_BAD(vp)) { | 187 | inode = VFS_I(ip); |
188 | if (is_bad_inode(inode)) { | ||
188 | read_unlock(&pag->pag_ici_lock); | 189 | read_unlock(&pag->pag_ici_lock); |
189 | continue; | 190 | continue; |
190 | } | 191 | } |
@@ -196,30 +197,29 @@ xfs_sync_inodes_ag( | |||
196 | } | 197 | } |
197 | 198 | ||
198 | /* | 199 | /* |
199 | * The inode lock here actually coordinates with the almost | 200 | * If we can't get a reference on the VFS_I, the inode must be |
200 | * spurious inode lock in xfs_ireclaim() to prevent the vnode | 201 | * in reclaim. If we can get the inode lock without blocking, |
201 | * we handle here without a reference from being freed while we | 202 | * it is safe to flush the inode because we hold the tree lock |
202 | * reference it. If we lock the inode while it's on the mount | 203 | * and xfs_iextract will block right now. Hence if we lock the |
203 | * list here, then the spurious inode lock in xfs_ireclaim() | 204 | * inode while holding the tree lock, xfs_ireclaim() is |
204 | * after the inode is pulled from the mount list will sleep | 205 | * guaranteed to block on the inode lock we now hold and hence |
205 | * until we release it here. This keeps the vnode from being | 206 | * it is safe to reference the inode until we drop the inode |
206 | * freed while we reference it. | 207 | * locks completely. |
207 | */ | 208 | */ |
208 | if (xfs_ilock_nowait(ip, lock_flags) == 0) { | 209 | inode_refed = B_FALSE; |
209 | vp = vn_grab(vp); | 210 | if (igrab(inode)) { |
210 | read_unlock(&pag->pag_ici_lock); | 211 | read_unlock(&pag->pag_ici_lock); |
211 | if (!vp) | ||
212 | continue; | ||
213 | xfs_ilock(ip, lock_flags); | 212 | xfs_ilock(ip, lock_flags); |
214 | 213 | inode_refed = B_TRUE; | |
215 | ASSERT(vp == VFS_I(ip)); | ||
216 | ASSERT(ip->i_mount == mp); | ||
217 | |||
218 | vnode_refed = B_TRUE; | ||
219 | } else { | 214 | } else { |
220 | /* safe to unlock here as we have a reference */ | 215 | if (!xfs_ilock_nowait(ip, lock_flags)) { |
216 | /* leave it to reclaim */ | ||
217 | read_unlock(&pag->pag_ici_lock); | ||
218 | continue; | ||
219 | } | ||
221 | read_unlock(&pag->pag_ici_lock); | 220 | read_unlock(&pag->pag_ici_lock); |
222 | } | 221 | } |
222 | |||
223 | /* | 223 | /* |
224 | * If we have to flush data or wait for I/O completion | 224 | * If we have to flush data or wait for I/O completion |
225 | * we need to drop the ilock that we currently hold. | 225 | * we need to drop the ilock that we currently hold. |
@@ -240,7 +240,7 @@ xfs_sync_inodes_ag( | |||
240 | xfs_ilock(ip, XFS_ILOCK_SHARED); | 240 | xfs_ilock(ip, XFS_ILOCK_SHARED); |
241 | } | 241 | } |
242 | 242 | ||
243 | if ((flags & SYNC_DELWRI) && VN_DIRTY(vp)) { | 243 | if ((flags & SYNC_DELWRI) && VN_DIRTY(inode)) { |
244 | xfs_iunlock(ip, XFS_ILOCK_SHARED); | 244 | xfs_iunlock(ip, XFS_ILOCK_SHARED); |
245 | error = xfs_flush_pages(ip, 0, -1, fflag, FI_NONE); | 245 | error = xfs_flush_pages(ip, 0, -1, fflag, FI_NONE); |
246 | if (flags & SYNC_IOWAIT) | 246 | if (flags & SYNC_IOWAIT) |
@@ -268,9 +268,8 @@ xfs_sync_inodes_ag( | |||
268 | if (lock_flags) | 268 | if (lock_flags) |
269 | xfs_iunlock(ip, lock_flags); | 269 | xfs_iunlock(ip, lock_flags); |
270 | 270 | ||
271 | if (vnode_refed) { | 271 | if (inode_refed) { |
272 | IRELE(ip); | 272 | IRELE(ip); |
273 | vnode_refed = B_FALSE; | ||
274 | } | 273 | } |
275 | 274 | ||
276 | if (error) | 275 | if (error) |
diff --git a/fs/xfs/linux-2.6/xfs_vnode.c b/fs/xfs/linux-2.6/xfs_vnode.c index b52528bbbfff..dceb6dbaa2da 100644 --- a/fs/xfs/linux-2.6/xfs_vnode.c +++ b/fs/xfs/linux-2.6/xfs_vnode.c | |||
@@ -90,10 +90,10 @@ vn_ioerror( | |||
90 | */ | 90 | */ |
91 | static inline int xfs_icount(struct xfs_inode *ip) | 91 | static inline int xfs_icount(struct xfs_inode *ip) |
92 | { | 92 | { |
93 | struct inode *vp = VFS_I(ip); | 93 | struct inode *inode = VFS_I(ip); |
94 | 94 | ||
95 | if (vp) | 95 | if (!inode) |
96 | return vn_count(vp); | 96 | return atomic_read(&inode->i_count); |
97 | return -1; | 97 | return -1; |
98 | } | 98 | } |
99 | 99 | ||
diff --git a/fs/xfs/linux-2.6/xfs_vnode.h b/fs/xfs/linux-2.6/xfs_vnode.h index 683ce16210ff..bf89e41c3b8d 100644 --- a/fs/xfs/linux-2.6/xfs_vnode.h +++ b/fs/xfs/linux-2.6/xfs_vnode.h | |||
@@ -80,11 +80,6 @@ do { \ | |||
80 | iput(VFS_I(ip)); \ | 80 | iput(VFS_I(ip)); \ |
81 | } while (0) | 81 | } while (0) |
82 | 82 | ||
83 | static inline struct inode *vn_grab(struct inode *vp) | ||
84 | { | ||
85 | return igrab(vp); | ||
86 | } | ||
87 | |||
88 | /* | 83 | /* |
89 | * Dealing with bad inodes | 84 | * Dealing with bad inodes |
90 | */ | 85 | */ |
diff --git a/fs/xfs/quota/xfs_qm_syscalls.c b/fs/xfs/quota/xfs_qm_syscalls.c index 26152b9ccc6f..4254b074223b 100644 --- a/fs/xfs/quota/xfs_qm_syscalls.c +++ b/fs/xfs/quota/xfs_qm_syscalls.c | |||
@@ -1031,13 +1031,13 @@ xfs_qm_dqrele_inodes_ag( | |||
1031 | uint flags) | 1031 | uint flags) |
1032 | { | 1032 | { |
1033 | xfs_inode_t *ip = NULL; | 1033 | xfs_inode_t *ip = NULL; |
1034 | struct inode *vp = NULL; | ||
1035 | xfs_perag_t *pag = &mp->m_perag[ag]; | 1034 | xfs_perag_t *pag = &mp->m_perag[ag]; |
1036 | int first_index = 0; | 1035 | int first_index = 0; |
1037 | int nr_found; | 1036 | int nr_found; |
1038 | 1037 | ||
1039 | do { | 1038 | do { |
1040 | boolean_t vnode_refd = B_FALSE; | 1039 | boolean_t inode_refed; |
1040 | struct inode *inode; | ||
1041 | 1041 | ||
1042 | /* | 1042 | /* |
1043 | * use a gang lookup to find the next inode in the tree | 1043 | * use a gang lookup to find the next inode in the tree |
@@ -1057,19 +1057,19 @@ xfs_qm_dqrele_inodes_ag( | |||
1057 | first_index = XFS_INO_TO_AGINO(mp, ip->i_ino + 1); | 1057 | first_index = XFS_INO_TO_AGINO(mp, ip->i_ino + 1); |
1058 | 1058 | ||
1059 | /* skip quota inodes and those in reclaim */ | 1059 | /* skip quota inodes and those in reclaim */ |
1060 | vp = VFS_I(ip); | 1060 | inode = VFS_I(ip); |
1061 | if (!vp || ip == XFS_QI_UQIP(mp) || ip == XFS_QI_GQIP(mp)) { | 1061 | if (!inode || ip == XFS_QI_UQIP(mp) || ip == XFS_QI_GQIP(mp)) { |
1062 | ASSERT(ip->i_udquot == NULL); | 1062 | ASSERT(ip->i_udquot == NULL); |
1063 | ASSERT(ip->i_gdquot == NULL); | 1063 | ASSERT(ip->i_gdquot == NULL); |
1064 | read_unlock(&pag->pag_ici_lock); | 1064 | read_unlock(&pag->pag_ici_lock); |
1065 | continue; | 1065 | continue; |
1066 | } | 1066 | } |
1067 | if (xfs_ilock_nowait(ip, XFS_ILOCK_EXCL) == 0) { | 1067 | if (xfs_ilock_nowait(ip, XFS_ILOCK_EXCL) == 0) { |
1068 | vp = vn_grab(vp); | 1068 | inode = igrab(inode); |
1069 | read_unlock(&pag->pag_ici_lock); | 1069 | read_unlock(&pag->pag_ici_lock); |
1070 | if (!vp) | 1070 | if (!inode) |
1071 | continue; | 1071 | continue; |
1072 | vnode_refd = B_TRUE; | 1072 | inode_refed = B_TRUE; |
1073 | xfs_ilock(ip, XFS_ILOCK_EXCL); | 1073 | xfs_ilock(ip, XFS_ILOCK_EXCL); |
1074 | } else { | 1074 | } else { |
1075 | read_unlock(&pag->pag_ici_lock); | 1075 | read_unlock(&pag->pag_ici_lock); |
@@ -1084,7 +1084,7 @@ xfs_qm_dqrele_inodes_ag( | |||
1084 | ip->i_gdquot = NULL; | 1084 | ip->i_gdquot = NULL; |
1085 | } | 1085 | } |
1086 | xfs_iunlock(ip, XFS_ILOCK_EXCL); | 1086 | xfs_iunlock(ip, XFS_ILOCK_EXCL); |
1087 | if (vnode_refd) | 1087 | if (inode_refed) |
1088 | IRELE(ip); | 1088 | IRELE(ip); |
1089 | } while (nr_found); | 1089 | } while (nr_found); |
1090 | } | 1090 | } |