]> git.ipfire.org Git - thirdparty/xfsprogs-dev.git/commitdiff
libxfs: shut down filesystem if we xfs_trans_cancel with deferred work items
authorDarrick J. Wong <djwong@kernel.org>
Fri, 25 Feb 2022 22:32:48 +0000 (17:32 -0500)
committerEric Sandeen <sandeen@sandeen.net>
Fri, 25 Feb 2022 22:32:48 +0000 (17:32 -0500)
While debugging some very strange rmap corruption reports in connection
with the online directory repair code.  I root-caused the error to the
following incorrect sequence:

<start repair transaction>
<expand directory, causing a deferred rmap to be queued>
<roll transaction>
<cancel transaction>

Obviously, we should have committed the transaction instead of
cancelling it.  Thinking more broadly, however, xfs_trans_cancel should
have warned us that we were throwing away work item that we already
committed to performing.  This is not correct, and we need to shut down
the filesystem.

Change xfs_trans_cancel to complain in the loudest manner if we're
cancelling any transaction with deferred work items attached.

Signed-off-by: Darrick J. Wong <djwong@kernel.org>
Reviewed-by: Eric Sandeen <sandeen@redhat.com>
Signed-off-by: Eric Sandeen <sandeen@sandeen.net>
libxfs/trans.c

index fd2e6f9dfbd99beec2dc24366e8ac1abd7905195..8c16cb8d98a661fa1542cd6da4e1cac61926bed6 100644 (file)
@@ -318,13 +318,30 @@ void
 libxfs_trans_cancel(
        struct xfs_trans        *tp)
 {
+       bool                    dirty;
+
        trace_xfs_trans_cancel(tp, _RET_IP_);
 
        if (tp == NULL)
                return;
+       dirty = (tp->t_flags & XFS_TRANS_DIRTY);
 
-       if (tp->t_flags & XFS_TRANS_PERM_LOG_RES)
+       /*
+        * It's never valid to cancel a transaction with deferred ops attached,
+        * because the transaction is effectively dirty.  Complain about this
+        * loudly before freeing the in-memory defer items.
+        */
+       if (!list_empty(&tp->t_dfops)) {
+               ASSERT(list_empty(&tp->t_dfops));
+               ASSERT(tp->t_flags & XFS_TRANS_PERM_LOG_RES);
+               dirty = true;
                xfs_defer_cancel(tp);
+       }
+
+       if (dirty) {
+               fprintf(stderr, _("Cancelling dirty transaction!\n"));
+               abort();
+       }
 
        xfs_trans_free_items(tp);
        xfs_trans_free(tp);