(bmap->me_flags & ~XFS_BMAP_EXTENT_FLAGS))
                return -EFSCORRUPTED;
 
-       error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate,
-                       XFS_EXTENTADD_SPACE_RES(mp, XFS_DATA_FORK), 0, 0, &tp);
-       if (error)
-               return error;
-
-       budp = xfs_trans_get_bud(tp, buip);
-
        /* Grab the inode. */
-       error = xfs_iget(mp, tp, bmap->me_owner, 0, XFS_ILOCK_EXCL, &ip);
+       error = xfs_iget(mp, NULL, bmap->me_owner, 0, 0, &ip);
        if (error)
-               goto err_inode;
+               return error;
 
-       error = xfs_qm_dqattach_locked(ip, false);
+       error = xfs_qm_dqattach(ip);
        if (error)
-               goto err_inode;
+               goto err_rele;
 
        if (VFS_I(ip)->i_nlink == 0)
                xfs_iflags_set(ip, XFS_IRECOVERY);
 
+       /* Allocate transaction and do the work. */
+       error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate,
+                       XFS_EXTENTADD_SPACE_RES(mp, XFS_DATA_FORK), 0, 0, &tp);
+       if (error)
+               goto err_rele;
+
+       budp = xfs_trans_get_bud(tp, buip);
+       xfs_ilock(ip, XFS_ILOCK_EXCL);
        xfs_trans_ijoin(tp, ip, 0);
 
        count = bmap->me_len;
                        whichfork, bmap->me_startoff, bmap->me_startblock,
                        &count, state);
        if (error)
-               goto err_inode;
+               goto err_cancel;
 
        if (count > 0) {
                ASSERT(bui_type == XFS_BMAP_UNMAP);
                xfs_bmap_unmap_extent(tp, ip, &irec);
        }
 
+       /* Commit transaction, which frees the transaction. */
        error = xfs_defer_ops_capture_and_commit(tp, capture_list);
+       if (error)
+               goto err_unlock;
+
        xfs_iunlock(ip, XFS_ILOCK_EXCL);
        xfs_irele(ip);
-       return error;
+       return 0;
 
-err_inode:
+err_cancel:
        xfs_trans_cancel(tp);
-       if (ip) {
-               xfs_iunlock(ip, XFS_ILOCK_EXCL);
-               xfs_irele(ip);
-       }
+err_unlock:
+       xfs_iunlock(ip, XFS_ILOCK_EXCL);
+err_rele:
+       xfs_irele(ip);
        return error;
 }