Message ID | 20171012112732.39137-1-bfoster@redhat.com (mailing list archive) |
---|---|
State | Accepted, archived |
Headers | show |
On Thu, Oct 12, 2017 at 07:27:32AM -0400, Brian Foster wrote: > The child buffer read in xfs_attr3_node_inactive() should never > reach a hole in the attr fork. If this occurs, it is likely due to a > bug. Prior to commit cd87d867 ("xfs: don't crash on unexpected holes > in dir/attr btrees"), this would result in a crash. Now that the > crash has been fixed, this is a silent failure. > > Pass -1 to xfs_da3_node_read() from xfs_da3_node_inactive() to > indicate that reading from a hole is an error. This logs an error to > syslog and fails the inode inactivation, leaving the inode on the AG > unlinked list until removed by xfs_repair (or log recovery). Also > update the subsequent code to reflect that the read now returns a > non-NULL buffer or an error. > > Signed-off-by: Brian Foster <bfoster@redhat.com> Looks ok, Reviewed-by: Darrick J. Wong <darrick.wong@oracle.com> Since we've already fixed the crash problem and the dangling pointer problem (as soon as I send this week's fixes to Linus, anyway), I think this can wait for 4.15, unless anyone wants me to push sooner. --D > --- > fs/xfs/xfs_attr_inactive.c | 69 ++++++++++++++++++++++------------------------ > 1 file changed, 33 insertions(+), 36 deletions(-) > > diff --git a/fs/xfs/xfs_attr_inactive.c b/fs/xfs/xfs_attr_inactive.c > index e3a950e..52818ea 100644 > --- a/fs/xfs/xfs_attr_inactive.c > +++ b/fs/xfs/xfs_attr_inactive.c > @@ -251,47 +251,44 @@ xfs_attr3_node_inactive( > * traversal of the tree so we may deal with many blocks > * before we come back to this one. > */ > - error = xfs_da3_node_read(*trans, dp, child_fsb, -2, &child_bp, > - XFS_ATTR_FORK); > + error = xfs_da3_node_read(*trans, dp, child_fsb, -1, &child_bp, > + XFS_ATTR_FORK); > if (error) > return error; > - if (child_bp) { > - /* save for re-read later */ > - child_blkno = XFS_BUF_ADDR(child_bp); > > - /* > - * Invalidate the subtree, however we have to. > - */ > - info = child_bp->b_addr; > - switch (info->magic) { > - case cpu_to_be16(XFS_DA_NODE_MAGIC): > - case cpu_to_be16(XFS_DA3_NODE_MAGIC): > - error = xfs_attr3_node_inactive(trans, dp, > - child_bp, level + 1); > - break; > - case cpu_to_be16(XFS_ATTR_LEAF_MAGIC): > - case cpu_to_be16(XFS_ATTR3_LEAF_MAGIC): > - error = xfs_attr3_leaf_inactive(trans, dp, > - child_bp); > - break; > - default: > - error = -EIO; > - xfs_trans_brelse(*trans, child_bp); > - break; > - } > - if (error) > - return error; > + /* save for re-read later */ > + child_blkno = XFS_BUF_ADDR(child_bp); > > - /* > - * Remove the subsidiary block from the cache > - * and from the log. > - */ > - error = xfs_da_get_buf(*trans, dp, 0, child_blkno, > - &child_bp, XFS_ATTR_FORK); > - if (error) > - return error; > - xfs_trans_binval(*trans, child_bp); > + /* > + * Invalidate the subtree, however we have to. > + */ > + info = child_bp->b_addr; > + switch (info->magic) { > + case cpu_to_be16(XFS_DA_NODE_MAGIC): > + case cpu_to_be16(XFS_DA3_NODE_MAGIC): > + error = xfs_attr3_node_inactive(trans, dp, child_bp, > + level + 1); > + break; > + case cpu_to_be16(XFS_ATTR_LEAF_MAGIC): > + case cpu_to_be16(XFS_ATTR3_LEAF_MAGIC): > + error = xfs_attr3_leaf_inactive(trans, dp, child_bp); > + break; > + default: > + error = -EIO; > + xfs_trans_brelse(*trans, child_bp); > + break; > } > + if (error) > + return error; > + > + /* > + * Remove the subsidiary block from the cache and from the log. > + */ > + error = xfs_da_get_buf(*trans, dp, 0, child_blkno, &child_bp, > + XFS_ATTR_FORK); > + if (error) > + return error; > + xfs_trans_binval(*trans, child_bp); > > /* > * If we're not done, re-read the parent to get the next > -- > 2.9.5 > > -- > To unsubscribe from this list: send the line "unsubscribe linux-xfs" in > the body of a message to majordomo@vger.kernel.org > More majordomo info at http://vger.kernel.org/majordomo-info.html -- To unsubscribe from this list: send the line "unsubscribe linux-xfs" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
diff --git a/fs/xfs/xfs_attr_inactive.c b/fs/xfs/xfs_attr_inactive.c index e3a950e..52818ea 100644 --- a/fs/xfs/xfs_attr_inactive.c +++ b/fs/xfs/xfs_attr_inactive.c @@ -251,47 +251,44 @@ xfs_attr3_node_inactive( * traversal of the tree so we may deal with many blocks * before we come back to this one. */ - error = xfs_da3_node_read(*trans, dp, child_fsb, -2, &child_bp, - XFS_ATTR_FORK); + error = xfs_da3_node_read(*trans, dp, child_fsb, -1, &child_bp, + XFS_ATTR_FORK); if (error) return error; - if (child_bp) { - /* save for re-read later */ - child_blkno = XFS_BUF_ADDR(child_bp); - /* - * Invalidate the subtree, however we have to. - */ - info = child_bp->b_addr; - switch (info->magic) { - case cpu_to_be16(XFS_DA_NODE_MAGIC): - case cpu_to_be16(XFS_DA3_NODE_MAGIC): - error = xfs_attr3_node_inactive(trans, dp, - child_bp, level + 1); - break; - case cpu_to_be16(XFS_ATTR_LEAF_MAGIC): - case cpu_to_be16(XFS_ATTR3_LEAF_MAGIC): - error = xfs_attr3_leaf_inactive(trans, dp, - child_bp); - break; - default: - error = -EIO; - xfs_trans_brelse(*trans, child_bp); - break; - } - if (error) - return error; + /* save for re-read later */ + child_blkno = XFS_BUF_ADDR(child_bp); - /* - * Remove the subsidiary block from the cache - * and from the log. - */ - error = xfs_da_get_buf(*trans, dp, 0, child_blkno, - &child_bp, XFS_ATTR_FORK); - if (error) - return error; - xfs_trans_binval(*trans, child_bp); + /* + * Invalidate the subtree, however we have to. + */ + info = child_bp->b_addr; + switch (info->magic) { + case cpu_to_be16(XFS_DA_NODE_MAGIC): + case cpu_to_be16(XFS_DA3_NODE_MAGIC): + error = xfs_attr3_node_inactive(trans, dp, child_bp, + level + 1); + break; + case cpu_to_be16(XFS_ATTR_LEAF_MAGIC): + case cpu_to_be16(XFS_ATTR3_LEAF_MAGIC): + error = xfs_attr3_leaf_inactive(trans, dp, child_bp); + break; + default: + error = -EIO; + xfs_trans_brelse(*trans, child_bp); + break; } + if (error) + return error; + + /* + * Remove the subsidiary block from the cache and from the log. + */ + error = xfs_da_get_buf(*trans, dp, 0, child_blkno, &child_bp, + XFS_ATTR_FORK); + if (error) + return error; + xfs_trans_binval(*trans, child_bp); /* * If we're not done, re-read the parent to get the next
The child buffer read in xfs_attr3_node_inactive() should never reach a hole in the attr fork. If this occurs, it is likely due to a bug. Prior to commit cd87d867 ("xfs: don't crash on unexpected holes in dir/attr btrees"), this would result in a crash. Now that the crash has been fixed, this is a silent failure. Pass -1 to xfs_da3_node_read() from xfs_da3_node_inactive() to indicate that reading from a hole is an error. This logs an error to syslog and fails the inode inactivation, leaving the inode on the AG unlinked list until removed by xfs_repair (or log recovery). Also update the subsequent code to reflect that the read now returns a non-NULL buffer or an error. Signed-off-by: Brian Foster <bfoster@redhat.com> --- fs/xfs/xfs_attr_inactive.c | 69 ++++++++++++++++++++++------------------------ 1 file changed, 33 insertions(+), 36 deletions(-)