Skip site navigation (1)Skip section navigation (2)
Date:      Mon, 20 Apr 2026 21:07:08 +0000
From:      Alan Somers <asomers@FreeBSD.org>
To:        src-committers@FreeBSD.org, dev-commits-src-all@FreeBSD.org, dev-commits-src-main@FreeBSD.org
Subject:   git: 374548e930bd - main - fusefs: better handling for low-memory conditions
Message-ID:  <69e6957c.26919.6645a2d6@gitrepo.freebsd.org>

index | next in thread | raw e-mail

The branch main has been updated by asomers:

URL: https://cgit.FreeBSD.org/src/commit/?id=374548e930bd9452351059c14e697036d903b1c4

commit 374548e930bd9452351059c14e697036d903b1c4
Author:     Alan Somers <asomers@FreeBSD.org>
AuthorDate: 2026-03-30 14:22:07 +0000
Commit:     Alan Somers <asomers@FreeBSD.org>
CommitDate: 2026-04-20 21:06:57 +0000

    fusefs: better handling for low-memory conditions
    
    Under conditions of low memory, getblk can fail.  fusefs was not
    handling those failures very systematically.  It was always using
    PCATCH, which appears to have been originally copy/pasted from the NFS
    client code, but isn't always appropriate:
    
    * During fuse_vnode_setsize_immediate, which can be called from many
      different VOPs and from the vn_delayed_setsize mechanism, remove
      PCATCH.  Some of these callers cannot tolerate allocate failure.
    
    * In fuse_inval_buf_range, don't assume that getblk will always succeed.
    
    * When calling fuse_inval_buf_range from VOP_ALLOCATE,
      VOP_COPY_FILE_RANGE, or VOP_WRITE (with IO_DIRECT), return EINTR if
      the allocation fails.
    
    * When calling fuse_inval_buf_range from VOP_DEALLOCATE, remove PCATCH.
      This VOP must not fail with EINTR.
    
    No new tests, because I can't force any particular getblk call to fail.
    
    PR:             293957
    Sponsored by:   ConnectWise
    Reported by:    zjk7@wp.pl
    MFC after:      1 week
---
 sys/fs/fuse/fuse_node.c  |  6 +-----
 sys/fs/fuse/fuse_vnops.c | 24 +++++++++++++++++-------
 2 files changed, 18 insertions(+), 12 deletions(-)

diff --git a/sys/fs/fuse/fuse_node.c b/sys/fs/fuse/fuse_node.c
index f4fb993a7ca1..6768e87fdef1 100644
--- a/sys/fs/fuse/fuse_node.c
+++ b/sys/fs/fuse/fuse_node.c
@@ -506,11 +506,7 @@ fuse_vnode_setsize_immediate(struct vnode *vp, bool shrink)
 		 */
 
 		lbn = newsize / iosize;
-		bp = getblk(vp, lbn, iosize, PCATCH, 0, 0);
-		if (!bp) {
-			err = EINTR;
-			goto out;
-		}
+		bp = getblk(vp, lbn, iosize, 0, 0, 0);
 		if (!(bp->b_flags & B_CACHE))
 			goto out;	/* Nothing to do */
 		MPASS(bp->b_flags & B_VMIO);
diff --git a/sys/fs/fuse/fuse_vnops.c b/sys/fs/fuse/fuse_vnops.c
index ad7a2dcf84ef..dd3cc5f16092 100644
--- a/sys/fs/fuse/fuse_vnops.c
+++ b/sys/fs/fuse/fuse_vnops.c
@@ -327,7 +327,8 @@ fuse_fifo_close(struct vop_close_args *ap)
 
 /* Invalidate a range of cached data, whether dirty of not */
 static int
-fuse_inval_buf_range(struct vnode *vp, off_t filesize, off_t start, off_t end)
+fuse_inval_buf_range(struct vnode *vp, off_t filesize, off_t start, off_t end,
+	int slpflag)
 {
 	struct buf *bp;
 	daddr_t left_lbn, end_lbn, right_lbn;
@@ -339,7 +340,9 @@ fuse_inval_buf_range(struct vnode *vp, off_t filesize, off_t start, off_t end)
 	end_lbn = howmany(end, iosize);
 	left_on = start & (iosize - 1);
 	if (left_on != 0) {
-		bp = getblk(vp, left_lbn, iosize, PCATCH, 0, 0);
+		bp = getblk(vp, left_lbn, iosize, slpflag, 0, 0);
+		if (!bp)
+			return (EINTR);
 		if ((bp->b_flags & B_CACHE) != 0 && bp->b_dirtyend >= left_on) {
 			/*
 			 * Flush the dirty buffer, because we don't have a
@@ -358,7 +361,9 @@ fuse_inval_buf_range(struct vnode *vp, off_t filesize, off_t start, off_t end)
 		right_lbn = end / iosize;
 		new_filesize = MAX(filesize, end);
 		right_blksize = MIN(iosize, new_filesize - iosize * right_lbn);
-		bp = getblk(vp, right_lbn, right_blksize, PCATCH, 0, 0);
+		bp = getblk(vp, right_lbn, right_blksize, slpflag, 0, 0);
+		if (!bp)
+			return (EINTR);
 		if ((bp->b_flags & B_CACHE) != 0 && bp->b_dirtyoff < right_on) {
 			/*
 			 * Flush the dirty buffer, because we don't have a
@@ -726,7 +731,10 @@ fuse_vnop_allocate(struct vop_allocate_args *ap)
 	err = fuse_vnode_size(vp, &filesize, cred, curthread);
 	if (err)
 		return (err);
-	fuse_inval_buf_range(vp, filesize, *offset, *offset + *len);
+	err = fuse_inval_buf_range(vp, filesize, *offset, *offset + *len,
+	    PCATCH);
+	if (err)
+		return (err);
 
 	fdisp_init(&fdi, sizeof(*ffi));
 	fdisp_make_vp(&fdi, FUSE_FALLOCATE, vp, curthread, cred);
@@ -1020,7 +1028,7 @@ fuse_vnop_copy_file_range(struct vop_copy_file_range_args *ap)
 
 	vnode_pager_clean_sync(invp);
 	err = fuse_inval_buf_range(outvp, outfilesize, *ap->a_outoffp,
-		*ap->a_outoffp + io.uio_resid);
+		*ap->a_outoffp + io.uio_resid, PCATCH);
 	if (err)
 		goto unlock;
 
@@ -2672,7 +2680,7 @@ fuse_vnop_write(struct vop_write_args *ap)
 		end = start + uio->uio_resid;
 		if (!pages) {
 			err = fuse_inval_buf_range(vp, filesize, start,
-			    end);
+			    end, PCATCH);
 			if (err)
 				goto out;
 		}
@@ -3206,7 +3214,9 @@ fuse_vnop_deallocate(struct vop_deallocate_args *ap)
 	err = fuse_vnode_size(vp, &filesize, cred, curthread);
 	if (err)
 		goto out;
-	fuse_inval_buf_range(vp, filesize, *offset, *offset + *len);
+	err = fuse_inval_buf_range(vp, filesize, *offset, *offset + *len, 0);
+	if (err)
+		goto out;
 
 	fdisp_init(&fdi, sizeof(*ffi));
 	fdisp_make_vp(&fdi, FUSE_FALLOCATE, vp, curthread, cred);


home | help

Want to link to this message? Use this
URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?69e6957c.26919.6645a2d6>