Commit 71e3e356 authored by Dave Chinner's avatar Dave Chinner Committed by Darrick J. Wong

xfs: rework stale inodes in xfs_ifree_cluster

Once we have inodes pinning the cluster buffer and attached whenever
they are dirty, we no longer have a guarantee that the items are
flush locked when we lock the cluster buffer. Hence we cannot just
walk the buffer log item list and modify the attached inodes.

If the inode is not flush locked, we have to ILOCK it first and then
flush lock it to do all the prerequisite checks needed to avoid
races with other code. This is already handled by
xfs_ifree_get_one_inode(), so rework the inode iteration loop and
function to update all inodes in cache whether they are attached to
the buffer or not.

Note: we also remove the copying of the log item lsn to the
ili_flush_lsn as xfs_iflush_done() now uses the XFS_ISTALE flag to
trigger aborts and so flush lsn matching is not needed in IO
completion for processing freed inodes.
Signed-off-by: default avatarDave Chinner <dchinner@redhat.com>
Reviewed-by: default avatarDarrick J. Wong <darrick.wong@oracle.com>
Reviewed-by: default avatarBrian Foster <bfoster@redhat.com>
Signed-off-by: default avatarDarrick J. Wong <darrick.wong@oracle.com>
parent 02511a5a
...@@ -2517,17 +2517,19 @@ xfs_iunlink_remove( ...@@ -2517,17 +2517,19 @@ xfs_iunlink_remove(
} }
/* /*
* Look up the inode number specified and mark it stale if it is found. If it is * Look up the inode number specified and if it is not already marked XFS_ISTALE
* dirty, return the inode so it can be attached to the cluster buffer so it can * mark it stale. We should only find clean inodes in this lookup that aren't
* be processed appropriately when the cluster free transaction completes. * already stale.
*/ */
static struct xfs_inode * static void
xfs_ifree_get_one_inode( xfs_ifree_mark_inode_stale(
struct xfs_perag *pag, struct xfs_buf *bp,
struct xfs_inode *free_ip, struct xfs_inode *free_ip,
xfs_ino_t inum) xfs_ino_t inum)
{ {
struct xfs_mount *mp = pag->pag_mount; struct xfs_mount *mp = bp->b_mount;
struct xfs_perag *pag = bp->b_pag;
struct xfs_inode_log_item *iip;
struct xfs_inode *ip; struct xfs_inode *ip;
retry: retry:
...@@ -2535,8 +2537,10 @@ xfs_ifree_get_one_inode( ...@@ -2535,8 +2537,10 @@ xfs_ifree_get_one_inode(
ip = radix_tree_lookup(&pag->pag_ici_root, XFS_INO_TO_AGINO(mp, inum)); ip = radix_tree_lookup(&pag->pag_ici_root, XFS_INO_TO_AGINO(mp, inum));
/* Inode not in memory, nothing to do */ /* Inode not in memory, nothing to do */
if (!ip) if (!ip) {
goto out_rcu_unlock; rcu_read_unlock();
return;
}
/* /*
* because this is an RCU protected lookup, we could find a recently * because this is an RCU protected lookup, we could find a recently
...@@ -2547,9 +2551,9 @@ xfs_ifree_get_one_inode( ...@@ -2547,9 +2551,9 @@ xfs_ifree_get_one_inode(
spin_lock(&ip->i_flags_lock); spin_lock(&ip->i_flags_lock);
if (ip->i_ino != inum || __xfs_iflags_test(ip, XFS_ISTALE)) { if (ip->i_ino != inum || __xfs_iflags_test(ip, XFS_ISTALE)) {
spin_unlock(&ip->i_flags_lock); spin_unlock(&ip->i_flags_lock);
goto out_rcu_unlock; rcu_read_unlock();
return;
} }
spin_unlock(&ip->i_flags_lock);
/* /*
* Don't try to lock/unlock the current inode, but we _cannot_ skip the * Don't try to lock/unlock the current inode, but we _cannot_ skip the
...@@ -2559,43 +2563,53 @@ xfs_ifree_get_one_inode( ...@@ -2559,43 +2563,53 @@ xfs_ifree_get_one_inode(
*/ */
if (ip != free_ip) { if (ip != free_ip) {
if (!xfs_ilock_nowait(ip, XFS_ILOCK_EXCL)) { if (!xfs_ilock_nowait(ip, XFS_ILOCK_EXCL)) {
spin_unlock(&ip->i_flags_lock);
rcu_read_unlock(); rcu_read_unlock();
delay(1); delay(1);
goto retry; goto retry;
} }
}
ip->i_flags |= XFS_ISTALE;
spin_unlock(&ip->i_flags_lock);
rcu_read_unlock();
/* /*
* Check the inode number again in case we're racing with * If we can't get the flush lock, the inode is already attached. All
* freeing in xfs_reclaim_inode(). See the comments in that * we needed to do here is mark the inode stale so buffer IO completion
* function for more information as to why the initial check is * will remove it from the AIL.
* not sufficient.
*/ */
if (ip->i_ino != inum) { iip = ip->i_itemp;
xfs_iunlock(ip, XFS_ILOCK_EXCL); if (!xfs_iflock_nowait(ip)) {
goto out_rcu_unlock; ASSERT(!list_empty(&iip->ili_item.li_bio_list));
} ASSERT(iip->ili_last_fields);
goto out_iunlock;
} }
rcu_read_unlock(); ASSERT(!iip || list_empty(&iip->ili_item.li_bio_list));
xfs_iflock(ip);
xfs_iflags_set(ip, XFS_ISTALE);
/* /*
* We don't need to attach clean inodes or those only with unlogged * Clean inodes can be released immediately. Everything else has to go
* changes (which we throw away, anyway). * through xfs_iflush_abort() on journal commit as the flock
* synchronises removal of the inode from the cluster buffer against
* inode reclaim.
*/ */
if (!ip->i_itemp || xfs_inode_clean(ip)) { if (xfs_inode_clean(ip)) {
ASSERT(ip != free_ip);
xfs_ifunlock(ip); xfs_ifunlock(ip);
xfs_iunlock(ip, XFS_ILOCK_EXCL); goto out_iunlock;
goto out_no_inode;
} }
return ip;
out_rcu_unlock: /* we have a dirty inode in memory that has not yet been flushed. */
rcu_read_unlock(); ASSERT(iip->ili_fields);
out_no_inode: spin_lock(&iip->ili_lock);
return NULL; iip->ili_last_fields = iip->ili_fields;
iip->ili_fields = 0;
iip->ili_fsync_fields = 0;
spin_unlock(&iip->ili_lock);
list_add_tail(&iip->ili_item.li_bio_list, &bp->b_li_list);
ASSERT(iip->ili_last_fields);
out_iunlock:
if (ip != free_ip)
xfs_iunlock(ip, XFS_ILOCK_EXCL);
} }
/* /*
...@@ -2605,26 +2619,20 @@ xfs_ifree_get_one_inode( ...@@ -2605,26 +2619,20 @@ xfs_ifree_get_one_inode(
*/ */
STATIC int STATIC int
xfs_ifree_cluster( xfs_ifree_cluster(
xfs_inode_t *free_ip, struct xfs_inode *free_ip,
xfs_trans_t *tp, struct xfs_trans *tp,
struct xfs_icluster *xic) struct xfs_icluster *xic)
{ {
xfs_mount_t *mp = free_ip->i_mount; struct xfs_mount *mp = free_ip->i_mount;
struct xfs_ino_geometry *igeo = M_IGEO(mp);
struct xfs_buf *bp;
xfs_daddr_t blkno;
xfs_ino_t inum = xic->first_ino;
int nbufs; int nbufs;
int i, j; int i, j;
int ioffset; int ioffset;
xfs_daddr_t blkno;
xfs_buf_t *bp;
xfs_inode_t *ip;
struct xfs_inode_log_item *iip;
struct xfs_log_item *lip;
struct xfs_perag *pag;
struct xfs_ino_geometry *igeo = M_IGEO(mp);
xfs_ino_t inum;
int error; int error;
inum = xic->first_ino;
pag = xfs_perag_get(mp, XFS_INO_TO_AGNO(mp, inum));
nbufs = igeo->ialloc_blks / igeo->blocks_per_cluster; nbufs = igeo->ialloc_blks / igeo->blocks_per_cluster;
for (j = 0; j < nbufs; j++, inum += igeo->inodes_per_cluster) { for (j = 0; j < nbufs; j++, inum += igeo->inodes_per_cluster) {
...@@ -2653,10 +2661,8 @@ xfs_ifree_cluster( ...@@ -2653,10 +2661,8 @@ xfs_ifree_cluster(
error = xfs_trans_get_buf(tp, mp->m_ddev_targp, blkno, error = xfs_trans_get_buf(tp, mp->m_ddev_targp, blkno,
mp->m_bsize * igeo->blocks_per_cluster, mp->m_bsize * igeo->blocks_per_cluster,
XBF_UNMAPPED, &bp); XBF_UNMAPPED, &bp);
if (error) { if (error)
xfs_perag_put(pag);
return error; return error;
}
/* /*
* This buffer may not have been correctly initialised as we * This buffer may not have been correctly initialised as we
...@@ -2670,59 +2676,16 @@ xfs_ifree_cluster( ...@@ -2670,59 +2676,16 @@ xfs_ifree_cluster(
bp->b_ops = &xfs_inode_buf_ops; bp->b_ops = &xfs_inode_buf_ops;
/* /*
* Walk the inodes already attached to the buffer and mark them * Now we need to set all the cached clean inodes as XFS_ISTALE,
* stale. These will all have the flush locks held, so an * too. This requires lookups, and will skip inodes that we've
* in-memory inode walk can't lock them. By marking them all * already marked XFS_ISTALE.
* stale first, we will not attempt to lock them in the loop
* below as the XFS_ISTALE flag will be set.
*/
list_for_each_entry(lip, &bp->b_li_list, li_bio_list) {
if (lip->li_type == XFS_LI_INODE) {
iip = (struct xfs_inode_log_item *)lip;
xfs_trans_ail_copy_lsn(mp->m_ail,
&iip->ili_flush_lsn,
&iip->ili_item.li_lsn);
xfs_iflags_set(iip->ili_inode, XFS_ISTALE);
}
}
/*
* For each inode in memory attempt to add it to the inode
* buffer and set it up for being staled on buffer IO
* completion. This is safe as we've locked out tail pushing
* and flushing by locking the buffer.
*
* We have already marked every inode that was part of a
* transaction stale above, which means there is no point in
* even trying to lock them.
*/ */
for (i = 0; i < igeo->inodes_per_cluster; i++) { for (i = 0; i < igeo->inodes_per_cluster; i++)
ip = xfs_ifree_get_one_inode(pag, free_ip, inum + i); xfs_ifree_mark_inode_stale(bp, free_ip, inum + i);
if (!ip)
continue;
iip = ip->i_itemp;
spin_lock(&iip->ili_lock);
iip->ili_last_fields = iip->ili_fields;
iip->ili_fields = 0;
iip->ili_fsync_fields = 0;
spin_unlock(&iip->ili_lock);
xfs_trans_ail_copy_lsn(mp->m_ail, &iip->ili_flush_lsn,
&iip->ili_item.li_lsn);
list_add_tail(&iip->ili_item.li_bio_list,
&bp->b_li_list);
if (ip != free_ip)
xfs_iunlock(ip, XFS_ILOCK_EXCL);
}
xfs_trans_stale_inode_buf(tp, bp); xfs_trans_stale_inode_buf(tp, bp);
xfs_trans_binval(tp, bp); xfs_trans_binval(tp, bp);
} }
xfs_perag_put(pag);
return 0; return 0;
} }
......
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