]> git.ipfire.org Git - thirdparty/xfsprogs-dev.git/commitdiff
xfs: zap broken inode forks
authorDarrick J. Wong <djwong@kernel.org>
Mon, 15 Apr 2024 23:07:39 +0000 (16:07 -0700)
committerDarrick J. Wong <djwong@kernel.org>
Wed, 17 Apr 2024 21:06:25 +0000 (14:06 -0700)
Source kernel commit: e744cef206055954517648070d2b3aaa3d2515ba

Determine if inode fork damage is responsible for the inode being unable
to pass the ifork verifiers in xfs_iget and zap the fork contents if
this is true.  Once this is done the fork will be empty but we'll be
able to construct an in-core inode, and a subsequent call to the inode
fork repair ioctl will search the rmapbt to rebuild the records that
were in the fork.

Signed-off-by: Darrick J. Wong <djwong@kernel.org>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Reviewed-by: Bill O'Donnell <bodonnel@redhat.com>
libxfs/xfs_attr_leaf.c
libxfs/xfs_attr_leaf.h
libxfs/xfs_bmap.c
libxfs/xfs_bmap.h
libxfs/xfs_dir2_priv.h
libxfs/xfs_dir2_sf.c
libxfs/xfs_inode_fork.c
libxfs/xfs_shared.h
libxfs/xfs_symlink_remote.c

index baa168318f91676ab2dd8a7ee2d51190ebab6540..8329348eb78b8602fa14d65fb1498bd0cc50cc76 100644 (file)
@@ -1037,23 +1037,16 @@ xfs_attr_shortform_allfit(
        return xfs_attr_shortform_bytesfit(dp, bytes);
 }
 
-/* Verify the consistency of an inline attribute fork. */
+/* Verify the consistency of a raw inline attribute fork. */
 xfs_failaddr_t
 xfs_attr_shortform_verify(
-       struct xfs_inode                *ip)
+       struct xfs_attr_shortform       *sfp,
+       size_t                          size)
 {
-       struct xfs_attr_shortform       *sfp;
        struct xfs_attr_sf_entry        *sfep;
        struct xfs_attr_sf_entry        *next_sfep;
        char                            *endp;
-       struct xfs_ifork                *ifp;
        int                             i;
-       int64_t                         size;
-
-       ASSERT(ip->i_af.if_format == XFS_DINODE_FMT_LOCAL);
-       ifp = xfs_ifork_ptr(ip, XFS_ATTR_FORK);
-       sfp = (struct xfs_attr_shortform *)ifp->if_u1.if_data;
-       size = ifp->if_bytes;
 
        /*
         * Give up if the attribute is way too short.
index 368f4d9fa1d596a43649c307676597b05a74e656..ce6743463c868174f3a7f11563dd9f8a88fda001 100644 (file)
@@ -56,7 +56,8 @@ int   xfs_attr_sf_findname(struct xfs_da_args *args,
                             unsigned int *basep);
 int    xfs_attr_shortform_allfit(struct xfs_buf *bp, struct xfs_inode *dp);
 int    xfs_attr_shortform_bytesfit(struct xfs_inode *dp, int bytes);
-xfs_failaddr_t xfs_attr_shortform_verify(struct xfs_inode *ip);
+xfs_failaddr_t xfs_attr_shortform_verify(struct xfs_attr_shortform *sfp,
+               size_t size);
 void   xfs_attr_fork_remove(struct xfs_inode *ip, struct xfs_trans *tp);
 
 /*
index 6d23c5e3e652b7a0e640ee04f2f7c447ac6dd772..534a516b59bad72b50cd8696ac4f23c57727ce86 100644 (file)
@@ -6162,19 +6162,18 @@ xfs_bmap_finish_one(
        return error;
 }
 
-/* Check that an inode's extent does not have invalid flags or bad ranges. */
+/* Check that an extent does not have invalid flags or bad ranges. */
 xfs_failaddr_t
-xfs_bmap_validate_extent(
-       struct xfs_inode        *ip,
+xfs_bmap_validate_extent_raw(
+       struct xfs_mount        *mp,
+       bool                    rtfile,
        int                     whichfork,
        struct xfs_bmbt_irec    *irec)
 {
-       struct xfs_mount        *mp = ip->i_mount;
-
        if (!xfs_verify_fileext(mp, irec->br_startoff, irec->br_blockcount))
                return __this_address;
 
-       if (XFS_IS_REALTIME_INODE(ip) && whichfork == XFS_DATA_FORK) {
+       if (rtfile && whichfork == XFS_DATA_FORK) {
                if (!xfs_verify_rtbext(mp, irec->br_startblock,
                                           irec->br_blockcount))
                        return __this_address;
@@ -6204,3 +6203,14 @@ xfs_bmap_intent_destroy_cache(void)
        kmem_cache_destroy(xfs_bmap_intent_cache);
        xfs_bmap_intent_cache = NULL;
 }
+
+/* Check that an inode's extent does not have invalid flags or bad ranges. */
+xfs_failaddr_t
+xfs_bmap_validate_extent(
+       struct xfs_inode        *ip,
+       int                     whichfork,
+       struct xfs_bmbt_irec    *irec)
+{
+       return xfs_bmap_validate_extent_raw(ip->i_mount,
+                       XFS_IS_REALTIME_INODE(ip), whichfork, irec);
+}
index e33470e39728d58fec33907f34860db700634d12..8518324db2855631b0e4d87a08b0c487b47300b9 100644 (file)
@@ -263,6 +263,8 @@ static inline uint32_t xfs_bmap_fork_to_state(int whichfork)
        }
 }
 
+xfs_failaddr_t xfs_bmap_validate_extent_raw(struct xfs_mount *mp, bool rtfile,
+               int whichfork, struct xfs_bmbt_irec *irec);
 xfs_failaddr_t xfs_bmap_validate_extent(struct xfs_inode *ip, int whichfork,
                struct xfs_bmbt_irec *irec);
 int xfs_bmap_complain_bad_rec(struct xfs_inode *ip, int whichfork,
index 7404a9ff1a9298257d1d19969f73067eb73be82f..1db2e60ba827fe649cb1fd56172ed88518827928 100644 (file)
@@ -175,7 +175,8 @@ extern int xfs_dir2_sf_create(struct xfs_da_args *args, xfs_ino_t pino);
 extern int xfs_dir2_sf_lookup(struct xfs_da_args *args);
 extern int xfs_dir2_sf_removename(struct xfs_da_args *args);
 extern int xfs_dir2_sf_replace(struct xfs_da_args *args);
-extern xfs_failaddr_t xfs_dir2_sf_verify(struct xfs_inode *ip);
+xfs_failaddr_t xfs_dir2_sf_verify(struct xfs_mount *mp,
+               struct xfs_dir2_sf_hdr *sfp, int64_t size);
 int xfs_dir2_sf_entsize(struct xfs_mount *mp,
                struct xfs_dir2_sf_hdr *hdr, int len);
 void xfs_dir2_sf_put_ino(struct xfs_mount *mp, struct xfs_dir2_sf_hdr *hdr,
index 08b36c95ced29d9adc3d923bc4268d6fc79a9702..260eccacf176ddfeebcafb51beb6688d8a011805 100644 (file)
@@ -707,11 +707,10 @@ xfs_dir2_sf_check(
 /* Verify the consistency of an inline directory. */
 xfs_failaddr_t
 xfs_dir2_sf_verify(
-       struct xfs_inode                *ip)
+       struct xfs_mount                *mp,
+       struct xfs_dir2_sf_hdr          *sfp,
+       int64_t                         size)
 {
-       struct xfs_mount                *mp = ip->i_mount;
-       struct xfs_ifork                *ifp = xfs_ifork_ptr(ip, XFS_DATA_FORK);
-       struct xfs_dir2_sf_hdr          *sfp;
        struct xfs_dir2_sf_entry        *sfep;
        struct xfs_dir2_sf_entry        *next_sfep;
        char                            *endp;
@@ -719,15 +718,9 @@ xfs_dir2_sf_verify(
        int                             i;
        int                             i8count;
        int                             offset;
-       int64_t                         size;
        int                             error;
        uint8_t                         filetype;
 
-       ASSERT(ifp->if_format == XFS_DINODE_FMT_LOCAL);
-
-       sfp = (struct xfs_dir2_sf_hdr *)ifp->if_u1.if_data;
-       size = ifp->if_bytes;
-
        /*
         * Give up if the directory is way too short.
         */
index 5cc056ff7a1b161080041c65aa5e6bce4479d8f8..3e2d7882a07754eac6d29fab704981f0652cb6a2 100644 (file)
@@ -700,12 +700,22 @@ xfs_ifork_verify_local_data(
        xfs_failaddr_t          fa = NULL;
 
        switch (VFS_I(ip)->i_mode & S_IFMT) {
-       case S_IFDIR:
-               fa = xfs_dir2_sf_verify(ip);
+       case S_IFDIR: {
+               struct xfs_mount        *mp = ip->i_mount;
+               struct xfs_ifork        *ifp = xfs_ifork_ptr(ip, XFS_DATA_FORK);
+               struct xfs_dir2_sf_hdr  *sfp;
+
+               sfp = (struct xfs_dir2_sf_hdr *)ifp->if_u1.if_data;
+               fa = xfs_dir2_sf_verify(mp, sfp, ifp->if_bytes);
                break;
-       case S_IFLNK:
-               fa = xfs_symlink_shortform_verify(ip);
+       }
+       case S_IFLNK: {
+               struct xfs_ifork        *ifp = xfs_ifork_ptr(ip, XFS_DATA_FORK);
+
+               fa = xfs_symlink_shortform_verify(ifp->if_u1.if_data,
+                               ifp->if_bytes);
                break;
+       }
        default:
                break;
        }
@@ -727,11 +737,20 @@ xfs_ifork_verify_local_attr(
        struct xfs_ifork        *ifp = &ip->i_af;
        xfs_failaddr_t          fa;
 
-       if (!xfs_inode_has_attr_fork(ip))
+       if (!xfs_inode_has_attr_fork(ip)) {
                fa = __this_address;
-       else
-               fa = xfs_attr_shortform_verify(ip);
+       } else {
+               struct xfs_attr_shortform       *sfp;
+               struct xfs_ifork                *ifp;
+               int64_t                         size;
 
+               ASSERT(ip->i_af.if_format == XFS_DINODE_FMT_LOCAL);
+               ifp = xfs_ifork_ptr(ip, XFS_ATTR_FORK);
+               sfp = (struct xfs_attr_shortform *)ifp->if_u1.if_data;
+               size = ifp->if_bytes;
+
+               fa = xfs_attr_shortform_verify(sfp, size);
+       }
        if (fa) {
                xfs_inode_verifier_error(ip, -EFSCORRUPTED, "attr fork",
                                ifp->if_u1.if_data, ifp->if_bytes, fa);
index c4381388c0c1a10f25c6f8bc722dba91ac459cd3..4220d3584c1b0b03b0fb6138ad86b2c9a440de4c 100644 (file)
@@ -139,7 +139,7 @@ bool xfs_symlink_hdr_ok(xfs_ino_t ino, uint32_t offset,
                        uint32_t size, struct xfs_buf *bp);
 void xfs_symlink_local_to_remote(struct xfs_trans *tp, struct xfs_buf *bp,
                                 struct xfs_inode *ip, struct xfs_ifork *ifp);
-xfs_failaddr_t xfs_symlink_shortform_verify(struct xfs_inode *ip);
+xfs_failaddr_t xfs_symlink_shortform_verify(void *sfp, int64_t size);
 
 /* Computed inode geometry for the filesystem. */
 struct xfs_ino_geometry {
index 29c9f1cc1212d31da2353b17cae42b891338abf6..cf894b5276ac29489564470ba5490bbaa3af6450 100644 (file)
@@ -199,15 +199,11 @@ xfs_symlink_local_to_remote(
  */
 xfs_failaddr_t
 xfs_symlink_shortform_verify(
-       struct xfs_inode        *ip)
+       void                    *sfp,
+       int64_t                 size)
 {
-       struct xfs_ifork        *ifp = xfs_ifork_ptr(ip, XFS_DATA_FORK);
-       char                    *sfp = (char *)ifp->if_u1.if_data;
-       int                     size = ifp->if_bytes;
        char                    *endp = sfp + size;
 
-       ASSERT(ifp->if_format == XFS_DINODE_FMT_LOCAL);
-
        /*
         * Zero length symlinks should never occur in memory as they are
         * never allowed to exist on disk.