xfs: do not propagate ENODATA disk errors into xattr code

ENODATA (aka ENOATTR) has a very specific meaning in the xfs xattr code;
namely, that the requested attribute name could not be found.

However, a medium error from disk may also return ENODATA. At best,
this medium error may escape to userspace as "attribute not found"
when in fact it's an IO (disk) error.

At worst, we may oops in xfs_attr_leaf_get() when we do:

error = xfs_attr_leaf_hasname(args, &bp);
if (error == -ENOATTR) {
xfs_trans_brelse(args->trans, bp);
return error;
}

because an ENODATA/ENOATTR error from disk leaves us with a null bp,
and the xfs_trans_brelse will then null-deref it.

As discussed on the list, we really need to modify the lower level
IO functions to trap all disk errors and ensure that we don't let
unique errors like this leak up into higher xfs functions - many
like this should be remapped to EIO.

However, this patch directly addresses a reported bug in the xattr
code, and should be safe to backport to stable kernels. A larger-scope
patch to handle more unique errors at lower levels can follow later.

(Note, prior to 07120f1abdff we did not oops, but we did return the
wrong error code to userspace.)

Signed-off-by: Eric Sandeen <sandeen@redhat.com>
Fixes: 07120f1abdff ("xfs: Add xfs_has_attr and subroutines")
Cc: stable@vger.kernel.org # v5.9+
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Carlos Maiolino <cem@kernel.org>

authored by Eric Sandeen and committed by Carlos Maiolino ae668cd5 8e5a2441

+13
+7
fs/xfs/libxfs/xfs_attr_remote.c
··· 435 435 0, &bp, &xfs_attr3_rmt_buf_ops); 436 436 if (xfs_metadata_is_sick(error)) 437 437 xfs_dirattr_mark_sick(args->dp, XFS_ATTR_FORK); 438 + /* 439 + * ENODATA from disk implies a disk medium failure; 440 + * ENODATA for xattrs means attribute not found, so 441 + * disambiguate that here. 442 + */ 443 + if (error == -ENODATA) 444 + error = -EIO; 438 445 if (error) 439 446 return error; 440 447
+6
fs/xfs/libxfs/xfs_da_btree.c
··· 2833 2833 &bp, ops); 2834 2834 if (xfs_metadata_is_sick(error)) 2835 2835 xfs_dirattr_mark_sick(dp, whichfork); 2836 + /* 2837 + * ENODATA from disk implies a disk medium failure; ENODATA for 2838 + * xattrs means attribute not found, so disambiguate that here. 2839 + */ 2840 + if (error == -ENODATA && whichfork == XFS_ATTR_FORK) 2841 + error = -EIO; 2836 2842 if (error) 2837 2843 goto out_free; 2838 2844