Linux filesystem development
 help / color / mirror / Atom feed
From: David Howells <dhowells@redhat.com>
To: Christian Brauner <christian@brauner.io>
Cc: David Howells <dhowells@redhat.com>,
	Paulo Alcantara <pc@manguebit.org>,
	netfs@lists.linux.dev, linux-afs@lists.infradead.org,
	linux-cifs@vger.kernel.org, ceph-devel@vger.kernel.org,
	linux-fsdevel@vger.kernel.org, linux-kernel@vger.kernel.org
Subject: [PATCH v6 14/24] netfs: Fix potential deadlock in write-through mode
Date: Tue, 12 May 2026 13:33:51 +0100	[thread overview]
Message-ID: <20260512123404.719402-15-dhowells@redhat.com> (raw)
In-Reply-To: <20260512123404.719402-1-dhowells@redhat.com>

Fix netfs_advance_writethrough() to always unlock the supplied folio and to
mark it dirty if it isn't yet written to the end.  Unfortunately, it can't
be marked for writeback until the folio is done with as that may cause a
deadlock against mmapped reads and writes.

Even though it has been marked dirty, premature writeback can't occur as
the caller is holding both inode->i_rwsem (which will prevent concurrent
truncation, fallocation, DIO and other writes) and ictx->wb_lock (which
will cause flushing to wait and writeback to skip or wait).

Note that this may be easier to deal with once the queuing of folios is
split from the generation of subrequests.

Fixes: 288ace2f57c9 ("netfs: New writeback implementation")
Closes: https://sashiko.dev/#/patchset/20260427154639.180684-1-dhowells%40redhat.com
Signed-off-by: David Howells <dhowells@redhat.com>
cc: Paulo Alcantara <pc@manguebit.org>
cc: netfs@lists.linux.dev
cc: linux-fsdevel@vger.kernel.org
---
 fs/netfs/write_issue.c | 39 +++++++++++++++++++++++++--------------
 1 file changed, 25 insertions(+), 14 deletions(-)

diff --git a/fs/netfs/write_issue.c b/fs/netfs/write_issue.c
index b0e9690bb90c..03961622996b 100644
--- a/fs/netfs/write_issue.c
+++ b/fs/netfs/write_issue.c
@@ -414,12 +414,7 @@ static int netfs_write_folio(struct netfs_io_request *wreq,
 	if (streamw)
 		netfs_issue_write(wreq, cache);
 
-	/* Flip the page to the writeback state and unlock.  If we're called
-	 * from write-through, then the page has already been put into the wb
-	 * state.
-	 */
-	if (wreq->origin == NETFS_WRITEBACK)
-		folio_start_writeback(folio);
+	folio_start_writeback(folio);
 	folio_unlock(folio);
 
 	if (fgroup == NETFS_FOLIO_COPY_TO_CACHE) {
@@ -647,29 +642,41 @@ int netfs_advance_writethrough(struct netfs_io_request *wreq, struct writeback_c
 			       struct folio *folio, size_t copied, bool to_page_end,
 			       struct folio **writethrough_cache)
 {
+	int ret;
+
 	_enter("R=%x ic=%zu ws=%u cp=%zu tp=%u",
 	       wreq->debug_id, wreq->buffer.iter.count, wreq->wsize, copied, to_page_end);
 
-	if (!*writethrough_cache) {
-		if (folio_test_dirty(folio))
-			/* Sigh.  mmap. */
-			folio_clear_dirty_for_io(folio);
+	/* The folio is locked. */
 
+	if (*writethrough_cache != folio) {
+		if (*writethrough_cache) {
+			/* Did the folio get moved? */
+			folio_put(*writethrough_cache);
+			*writethrough_cache = NULL;
+		}
 		/* We can make multiple writes to the folio... */
-		folio_start_writeback(folio);
 		if (wreq->len == 0)
 			trace_netfs_folio(folio, netfs_folio_trace_wthru);
 		else
 			trace_netfs_folio(folio, netfs_folio_trace_wthru_plus);
 		*writethrough_cache = folio;
+		folio_get(folio);
 	}
 
 	wreq->len += copied;
-	if (!to_page_end)
+
+	if (!to_page_end) {
+		folio_mark_dirty(folio);
+		folio_unlock(folio);
 		return 0;
+	}
 
+	ret = netfs_write_folio(wreq, wbc, folio);
+	folio_put(*writethrough_cache);
 	*writethrough_cache = NULL;
-	return netfs_write_folio(wreq, wbc, folio);
+	wreq->submitted = wreq->len;
+	return ret;
 }
 
 /*
@@ -683,8 +690,12 @@ ssize_t netfs_end_writethrough(struct netfs_io_request *wreq, struct writeback_c
 
 	_enter("R=%x", wreq->debug_id);
 
-	if (writethrough_cache)
+	if (writethrough_cache) {
+		folio_lock(writethrough_cache);
 		netfs_write_folio(wreq, wbc, writethrough_cache);
+		folio_put(writethrough_cache);
+		wreq->submitted = wreq->len;
+	}
 
 	netfs_end_issue_write(wreq);
 


  parent reply	other threads:[~2026-05-12 12:35 UTC|newest]

Thread overview: 28+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2026-05-12 12:33 [PATCH v6 00/24] netfs: Miscellaneous fixes David Howells
2026-05-12 12:33 ` [PATCH v6 01/24] netfs: Fix cancellation of a DIO and single read subrequests David Howells
2026-05-12 12:33 ` [PATCH v6 02/24] netfs: Fix missing locking around retry adding new subreqs David Howells
2026-05-12 12:33 ` [PATCH v6 03/24] netfs: Fix missing barriers when accessing stream->subrequests locklessly David Howells
2026-05-12 12:33 ` [PATCH v6 04/24] netfs: Fix netfs_read_to_pagecache() to pause on subreq failure David Howells
2026-05-12 12:33 ` [PATCH v6 05/24] netfs: Fix potential for tearing in ->remote_i_size and ->zero_point David Howells
2026-05-12 12:33 ` [PATCH v6 06/24] netfs: Fix zeropoint update where i_size > remote_i_size David Howells
2026-05-12 12:33 ` [PATCH v6 07/24] netfs: fix VM_BUG_ON_FOLIO() issue in netfs_write_begin() call David Howells
2026-05-12 12:33 ` [PATCH v6 08/24] netfs: Fix potential uninitialised var in netfs_extract_user_iter() David Howells
2026-05-12 13:07   ` CharSyam
2026-05-12 13:22     ` CharSyam
2026-05-12 12:33 ` [PATCH v6 09/24] netfs: fix error handling " David Howells
2026-05-12 12:33 ` [PATCH v6 10/24] netfs: Fix overrun check " David Howells
2026-05-12 12:33 ` [PATCH v6 11/24] netfs: Fix netfs_invalidate_folio() to clear dirty bit if all changes gone David Howells
2026-05-12 12:33 ` [PATCH v6 12/24] netfs: Defer the emission of trace_netfs_folio() David Howells
2026-05-12 12:33 ` [PATCH v6 13/24] netfs: Fix streaming write being overwritten David Howells
2026-05-12 12:33 ` David Howells [this message]
2026-05-12 12:33 ` [PATCH v6 15/24] netfs: Fix read-gaps to remove netfs_folio from filled folio David Howells
2026-05-12 12:33 ` [PATCH v6 16/24] netfs: Fix write streaming disablement if fd open O_RDWR David Howells
2026-05-12 12:33 ` [PATCH v6 17/24] netfs: Fix early put of sink folio in netfs_read_gaps() David Howells
2026-05-12 12:33 ` [PATCH v6 18/24] netfs: Fix leak of request in netfs_write_begin() error handling David Howells
2026-05-12 12:33 ` [PATCH v6 19/24] netfs: Fix potential UAF in netfs_unlock_abandoned_read_pages() David Howells
2026-05-12 12:33 ` [PATCH v6 20/24] netfs: Fix partial invalidation of streaming-write folio David Howells
2026-05-12 12:33 ` [PATCH v6 21/24] netfs: Fix folio->private handling in netfs_perform_write() David Howells
2026-05-12 12:33 ` [PATCH v6 22/24] netfs: Fix netfs_read_folio() to wait on writeback David Howells
2026-05-12 12:34 ` [PATCH v6 23/24] netfs, afs: Fix write skipping in dir/link writepages David Howells
2026-05-12 12:34 ` [PATCH v6 24/24] afs: Fix the locking used by afs_get_link() David Howells
2026-05-12 12:43 ` [PATCH v6 00/24] netfs: Miscellaneous fixes Christian Brauner

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20260512123404.719402-15-dhowells@redhat.com \
    --to=dhowells@redhat.com \
    --cc=ceph-devel@vger.kernel.org \
    --cc=christian@brauner.io \
    --cc=linux-afs@lists.infradead.org \
    --cc=linux-cifs@vger.kernel.org \
    --cc=linux-fsdevel@vger.kernel.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=netfs@lists.linux.dev \
    --cc=pc@manguebit.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox