Commit b822f035 authored by Brian Foster's avatar Brian Foster Committed by Sasha Levin

xfs: update ag iterator to support wait on new inodes

[ Upstream commit ae2c4ac2 ]

The AG inode iterator currently skips new inodes as such inodes are
inserted into the inode radix tree before they are fully
constructed. Certain contexts require the ability to wait on the
construction of new inodes, however. The fs-wide dquot release from
the quotaoff sequence is an example of this.

Update the AG inode iterator to support the ability to wait on
inodes flagged with XFS_INEW upon request. Create a new
xfs_inode_ag_iterator_flags() interface and support a set of
iteration flags to modify the iteration behavior. When the
XFS_AGITER_INEW_WAIT flag is set, include XFS_INEW flags in the
radix tree inode lookup and wait on them before the callback is
executed.
Signed-off-by: default avatarBrian Foster <bfoster@redhat.com>
Reviewed-by: default avatarDarrick J. Wong <darrick.wong@oracle.com>
Signed-off-by: default avatarDarrick J. Wong <darrick.wong@oracle.com>
Signed-off-by: default avatarSasha Levin <alexander.levin@verizon.com>
parent 821afaae
...@@ -366,6 +366,22 @@ xfs_iget_cache_miss( ...@@ -366,6 +366,22 @@ xfs_iget_cache_miss(
return error; return error;
} }
static void
xfs_inew_wait(
struct xfs_inode *ip)
{
wait_queue_head_t *wq = bit_waitqueue(&ip->i_flags, __XFS_INEW_BIT);
DEFINE_WAIT_BIT(wait, &ip->i_flags, __XFS_INEW_BIT);
do {
prepare_to_wait(wq, &wait.wait, TASK_UNINTERRUPTIBLE);
if (!xfs_iflags_test(ip, XFS_INEW))
break;
schedule();
} while (true);
finish_wait(wq, &wait.wait);
}
/* /*
* Look up an inode by number in the given file system. * Look up an inode by number in the given file system.
* The inode is looked up in the cache held in each AG. * The inode is looked up in the cache held in each AG.
...@@ -468,9 +484,11 @@ xfs_iget( ...@@ -468,9 +484,11 @@ xfs_iget(
STATIC int STATIC int
xfs_inode_ag_walk_grab( xfs_inode_ag_walk_grab(
struct xfs_inode *ip) struct xfs_inode *ip,
int flags)
{ {
struct inode *inode = VFS_I(ip); struct inode *inode = VFS_I(ip);
bool newinos = !!(flags & XFS_AGITER_INEW_WAIT);
ASSERT(rcu_read_lock_held()); ASSERT(rcu_read_lock_held());
...@@ -488,7 +506,8 @@ xfs_inode_ag_walk_grab( ...@@ -488,7 +506,8 @@ xfs_inode_ag_walk_grab(
goto out_unlock_noent; goto out_unlock_noent;
/* avoid new or reclaimable inodes. Leave for reclaim code to flush */ /* avoid new or reclaimable inodes. Leave for reclaim code to flush */
if (__xfs_iflags_test(ip, XFS_INEW | XFS_IRECLAIMABLE | XFS_IRECLAIM)) if ((!newinos && __xfs_iflags_test(ip, XFS_INEW)) ||
__xfs_iflags_test(ip, XFS_IRECLAIMABLE | XFS_IRECLAIM))
goto out_unlock_noent; goto out_unlock_noent;
spin_unlock(&ip->i_flags_lock); spin_unlock(&ip->i_flags_lock);
...@@ -516,7 +535,8 @@ xfs_inode_ag_walk( ...@@ -516,7 +535,8 @@ xfs_inode_ag_walk(
void *args), void *args),
int flags, int flags,
void *args, void *args,
int tag) int tag,
int iter_flags)
{ {
uint32_t first_index; uint32_t first_index;
int last_error = 0; int last_error = 0;
...@@ -558,7 +578,7 @@ xfs_inode_ag_walk( ...@@ -558,7 +578,7 @@ xfs_inode_ag_walk(
for (i = 0; i < nr_found; i++) { for (i = 0; i < nr_found; i++) {
struct xfs_inode *ip = batch[i]; struct xfs_inode *ip = batch[i];
if (done || xfs_inode_ag_walk_grab(ip)) if (done || xfs_inode_ag_walk_grab(ip, iter_flags))
batch[i] = NULL; batch[i] = NULL;
/* /*
...@@ -586,6 +606,9 @@ xfs_inode_ag_walk( ...@@ -586,6 +606,9 @@ xfs_inode_ag_walk(
for (i = 0; i < nr_found; i++) { for (i = 0; i < nr_found; i++) {
if (!batch[i]) if (!batch[i])
continue; continue;
if ((iter_flags & XFS_AGITER_INEW_WAIT) &&
xfs_iflags_test(batch[i], XFS_INEW))
xfs_inew_wait(batch[i]);
error = execute(batch[i], flags, args); error = execute(batch[i], flags, args);
IRELE(batch[i]); IRELE(batch[i]);
if (error == -EAGAIN) { if (error == -EAGAIN) {
...@@ -638,12 +661,13 @@ xfs_eofblocks_worker( ...@@ -638,12 +661,13 @@ xfs_eofblocks_worker(
} }
int int
xfs_inode_ag_iterator( xfs_inode_ag_iterator_flags(
struct xfs_mount *mp, struct xfs_mount *mp,
int (*execute)(struct xfs_inode *ip, int flags, int (*execute)(struct xfs_inode *ip, int flags,
void *args), void *args),
int flags, int flags,
void *args) void *args,
int iter_flags)
{ {
struct xfs_perag *pag; struct xfs_perag *pag;
int error = 0; int error = 0;
...@@ -653,7 +677,8 @@ xfs_inode_ag_iterator( ...@@ -653,7 +677,8 @@ xfs_inode_ag_iterator(
ag = 0; ag = 0;
while ((pag = xfs_perag_get(mp, ag))) { while ((pag = xfs_perag_get(mp, ag))) {
ag = pag->pag_agno + 1; ag = pag->pag_agno + 1;
error = xfs_inode_ag_walk(mp, pag, execute, flags, args, -1); error = xfs_inode_ag_walk(mp, pag, execute, flags, args, -1,
iter_flags);
xfs_perag_put(pag); xfs_perag_put(pag);
if (error) { if (error) {
last_error = error; last_error = error;
...@@ -664,6 +689,17 @@ xfs_inode_ag_iterator( ...@@ -664,6 +689,17 @@ xfs_inode_ag_iterator(
return last_error; return last_error;
} }
int
xfs_inode_ag_iterator(
struct xfs_mount *mp,
int (*execute)(struct xfs_inode *ip, int flags,
void *args),
int flags,
void *args)
{
return xfs_inode_ag_iterator_flags(mp, execute, flags, args, 0);
}
int int
xfs_inode_ag_iterator_tag( xfs_inode_ag_iterator_tag(
struct xfs_mount *mp, struct xfs_mount *mp,
...@@ -681,7 +717,8 @@ xfs_inode_ag_iterator_tag( ...@@ -681,7 +717,8 @@ xfs_inode_ag_iterator_tag(
ag = 0; ag = 0;
while ((pag = xfs_perag_get_tag(mp, ag, tag))) { while ((pag = xfs_perag_get_tag(mp, ag, tag))) {
ag = pag->pag_agno + 1; ag = pag->pag_agno + 1;
error = xfs_inode_ag_walk(mp, pag, execute, flags, args, tag); error = xfs_inode_ag_walk(mp, pag, execute, flags, args, tag,
0);
xfs_perag_put(pag); xfs_perag_put(pag);
if (error) { if (error) {
last_error = error; last_error = error;
......
...@@ -48,6 +48,11 @@ struct xfs_eofblocks { ...@@ -48,6 +48,11 @@ struct xfs_eofblocks {
#define XFS_IGET_UNTRUSTED 0x2 #define XFS_IGET_UNTRUSTED 0x2
#define XFS_IGET_DONTCACHE 0x4 #define XFS_IGET_DONTCACHE 0x4
/*
* flags for AG inode iterator
*/
#define XFS_AGITER_INEW_WAIT 0x1 /* wait on new inodes */
int xfs_iget(struct xfs_mount *mp, struct xfs_trans *tp, xfs_ino_t ino, int xfs_iget(struct xfs_mount *mp, struct xfs_trans *tp, xfs_ino_t ino,
uint flags, uint lock_flags, xfs_inode_t **ipp); uint flags, uint lock_flags, xfs_inode_t **ipp);
...@@ -72,6 +77,9 @@ void xfs_eofblocks_worker(struct work_struct *); ...@@ -72,6 +77,9 @@ void xfs_eofblocks_worker(struct work_struct *);
int xfs_inode_ag_iterator(struct xfs_mount *mp, int xfs_inode_ag_iterator(struct xfs_mount *mp,
int (*execute)(struct xfs_inode *ip, int flags, void *args), int (*execute)(struct xfs_inode *ip, int flags, void *args),
int flags, void *args); int flags, void *args);
int xfs_inode_ag_iterator_flags(struct xfs_mount *mp,
int (*execute)(struct xfs_inode *ip, int flags, void *args),
int flags, void *args, int iter_flags);
int xfs_inode_ag_iterator_tag(struct xfs_mount *mp, int xfs_inode_ag_iterator_tag(struct xfs_mount *mp,
int (*execute)(struct xfs_inode *ip, int flags, void *args), int (*execute)(struct xfs_inode *ip, int flags, void *args),
int flags, void *args, int tag); int flags, void *args, int tag);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment