[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <aTN6d0Qkh3WKt796@kernel.org>
Date: Fri, 5 Dec 2025 19:36:07 -0500
From: Mike Snitzer <snitzer@...nel.org>
To: Trond Myklebust <trondmy@...nel.org>
Cc: Stephen Rothwell <sfr@...b.auug.org.au>,
Linus Torvalds <torvalds@...ux-foundation.org>,
Christian Brauner <brauner@...nel.org>,
Linux Kernel Mailing List <linux-kernel@...r.kernel.org>,
Linux Next Mailing List <linux-next@...r.kernel.org>,
linux-nfs@...r.kernel.org, linux-stable@...r.kernel.org
Subject: Re: [6.19 PATCH] nfs/localio: fix regression due to out-of-order
__put_cred [was: Re: linux-next: manual merge of the nfs tree with Linus'
tree]
On Fri, Dec 05, 2025 at 07:01:13PM -0500, Trond Myklebust wrote:
> On Thu, 2025-12-04 at 20:08 -0500, Mike Snitzer wrote:
> > Hi Stephen,
> >
> > On Fri, Dec 05, 2025 at 11:19:42AM +1100, Stephen Rothwell wrote:
> > > Hi all,
> > >
> > > Today's linux-next merge of the nfs tree got a conflict in:
> > >
> > > fs/nfs/localio.c
> > >
> > > between commits:
> > >
> > > 94afb627dfc2 ("nfs: use credential guards in
> > > nfs_local_call_read()")
> > > bff3c841f7bd ("nfs: use credential guards in
> > > nfs_local_call_write()")
> > > 1d18101a644e ("Merge tag 'kernel-6.19-rc1.cred' of
> > > git://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs")
> > >
> > > from Linus' tree and commit:
> > >
> > > 30a4385509b4 ("nfs/localio: fix regression due to out-of-order
> > > __put_cred")
> > >
> > > from the nfs tree.
> >
> > The NFS tree's commit 30a4385509b4 needed to be rebased (taken care
> > of
> > below), which complicates the 6.18-stable backport (equivalent of the
> > nfs tree's commit 30a4385509b4 must be sent to linux-stable@ rather
> > than it being cherry-picked once the below updated fix applied to
> > Linus' tree).
> >
> > > I fixed it up (I just dropped the nfs tree commit) and can carry
> > > the
> > > fix as necessary. This is now fixed as far as linux-next is
> > > concerned,
> > > but any non trivial conflicts should be mentioned to your upstream
> > > maintainer when your tree is submitted for merging. You may also
> > > want
> > > to consider cooperating with the maintainer of the conflicting tree
> > > to
> > > minimise any particularly complex conflicts.
> >
> > Trond and Linus,
> >
> > Here is the fix for 6.19 rebased ontop of Linus' tree:
> >
> > From: Mike Snitzer <snitzer@...nel.org>
> > Date: Wed, 26 Nov 2025 01:01:25 -0500
> > Subject: [PATCH] nfs/localio: fix regression due to out-of-order
> > __put_cred
> >
> > Commit f2060bdc21d7 ("nfs/localio: add refcounting for each iocb IO
> > associated with NFS pgio header") inadvertantly reintroduced the same
> > potential for __put_cred() triggering BUG_ON(cred == current->cred)
> > that commit 992203a1fba5 ("nfs/localio: restore creds before
> > releasing
> > pageio data") fixed.
> >
> > Fix this by saving and restoring the cred around each
> > {read,write}_iter
> > call within the respective for loop of nfs_local_call_{read,write}
> > using scoped_with_creds().
> >
> > NOTE: this fix started by first reverting the following commits:
> >
> > 94afb627dfc2 ("nfs: use credential guards in nfs_local_call_read()")
> > bff3c841f7bd ("nfs: use credential guards in
> > nfs_local_call_write()")
> > 1d18101a644e ("Merge tag 'kernel-6.19-rc1.cred' of
> > git://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs")
> >
> > followed by narrowly fixing the cred lifetime issue by using
> > scoped_with_creds(). In doing so, this commit's changes appear more
> > extensive than they really are (as evidenced by comparing to v6.18's
> > fs/nfs/localio.c).
> >
> > Reported-by: Zorro Lang <zlang@...hat.com>
> > Fixes: f2060bdc21d7 ("nfs/localio: add refcounting for each iocb IO
> > associated with NFS pgio header")
> > Cc: linux-stable@...r.kernel.org # a custom 6.18-stable backport is
> > required
> > Signed-off-by: Mike Snitzer <snitzer@...nel.org>
> > Signed-off-by: Trond Myklebust <trond.myklebust@...merspace.com>
> >
> > diff --git a/fs/nfs/localio.c b/fs/nfs/localio.c
> > index 49ed90c6b9f2..f33bfa7b58e6 100644
> > --- a/fs/nfs/localio.c
> > +++ b/fs/nfs/localio.c
> > @@ -615,8 +615,11 @@ static void nfs_local_read_aio_complete(struct
> > kiocb *kiocb, long ret)
> > nfs_local_pgio_aio_complete(iocb); /* Calls
> > nfs_local_read_aio_complete_work */
> > }
> >
> > -static void do_nfs_local_call_read(struct nfs_local_kiocb *iocb,
> > struct file *filp)
> > +static void nfs_local_call_read(struct work_struct *work)
> > {
> > + struct nfs_local_kiocb *iocb =
> > + container_of(work, struct nfs_local_kiocb, work);
> > + struct file *filp = iocb->kiocb.ki_filp;
> > bool force_done = false;
> > ssize_t status;
> > int n_iters;
> > @@ -633,7 +636,9 @@ static void do_nfs_local_call_read(struct
> > nfs_local_kiocb *iocb, struct file *fi
> > } else
> > iocb->kiocb.ki_flags &= ~IOCB_DIRECT;
> >
> > - status = filp->f_op->read_iter(&iocb->kiocb, &iocb-
> > >iters[i]);
> > + scoped_with_creds(filp->f_cred)
> > + status = filp->f_op->read_iter(&iocb->kiocb,
> > &iocb->iters[i]);
> > +
> > if (status != -EIOCBQUEUED) {
> > if (unlikely(status >= 0 && status < iocb-
> > >iters[i].count))
> > force_done = true; /* Partial read
> > */
> > @@ -645,16 +650,6 @@ static void do_nfs_local_call_read(struct
> > nfs_local_kiocb *iocb, struct file *fi
> > }
> > }
> >
> > -static void nfs_local_call_read(struct work_struct *work)
> > -{
> > - struct nfs_local_kiocb *iocb =
> > - container_of(work, struct nfs_local_kiocb, work);
> > - struct file *filp = iocb->kiocb.ki_filp;
> > -
> > - scoped_with_creds(filp->f_cred)
> > - do_nfs_local_call_read(iocb, filp);
> > -}
> > -
> > static int
> > nfs_local_do_read(struct nfs_local_kiocb *iocb,
> > const struct rpc_call_ops *call_ops)
> > @@ -822,13 +817,18 @@ static void nfs_local_write_aio_complete(struct
> > kiocb *kiocb, long ret)
> > nfs_local_pgio_aio_complete(iocb); /* Calls
> > nfs_local_write_aio_complete_work */
> > }
> >
> > -static ssize_t do_nfs_local_call_write(struct nfs_local_kiocb *iocb,
> > - struct file *filp)
> > +static void nfs_local_call_write(struct work_struct *work)
> > {
> > + struct nfs_local_kiocb *iocb =
> > + container_of(work, struct nfs_local_kiocb, work);
> > + struct file *filp = iocb->kiocb.ki_filp;
> > + unsigned long old_flags = current->flags;
> > bool force_done = false;
> > ssize_t status;
> > int n_iters;
> >
> > + current->flags |= PF_LOCAL_THROTTLE | PF_MEMALLOC_NOIO;
> > +
> > file_start_write(filp);
> > n_iters = atomic_read(&iocb->n_iters);
> > for (int i = 0; i < n_iters ; i++) {
> > @@ -842,7 +842,9 @@ static ssize_t do_nfs_local_call_write(struct
> > nfs_local_kiocb *iocb,
> > } else
> > iocb->kiocb.ki_flags &= ~IOCB_DIRECT;
> >
> > - status = filp->f_op->write_iter(&iocb->kiocb, &iocb-
> > >iters[i]);
> > + scoped_with_creds(filp->f_cred)
> > + status = filp->f_op->write_iter(&iocb-
> > >kiocb, &iocb->iters[i]);
> > +
> > if (status != -EIOCBQUEUED) {
> > if (unlikely(status >= 0 && status < iocb-
> > >iters[i].count))
> > force_done = true; /* Partial write
> > */
> > @@ -854,22 +856,6 @@ static ssize_t do_nfs_local_call_write(struct
> > nfs_local_kiocb *iocb,
> > }
> > file_end_write(filp);
> >
> > - return status;
> > -}
> > -
> > -static void nfs_local_call_write(struct work_struct *work)
> > -{
> > - struct nfs_local_kiocb *iocb =
> > - container_of(work, struct nfs_local_kiocb, work);
> > - struct file *filp = iocb->kiocb.ki_filp;
> > - unsigned long old_flags = current->flags;
> > - ssize_t status;
> > -
> > - current->flags |= PF_LOCAL_THROTTLE | PF_MEMALLOC_NOIO;
> > -
> > - scoped_with_creds(filp->f_cred)
> > - status = do_nfs_local_call_write(iocb, filp);
> > -
> > current->flags = old_flags;
> > }
> >
>
> OK, so what is the easiest way to merge this?
>
> Should I just remove the "old" patch from my tree, and submit that
> patch directly to stable@...r.kernel.org as a fix for 6.18?
Yes, that will need to happen no matter what.
> That would allow Christian to pick up this (after perhaps removing the stable and
> Fixes tags above), and submit it as part of his merge, thus fixing the
> 6.19 kernel.
>
> Thoughts? Preferences?
Christian already sent his merge, at the top you'll see Stephen
referenced commits that are in Linus' tree. Those are the commits I
referenced in the above updated patch's header as having reverted.
Maybe you prepare a 2nd post-merge NFS client pull that is based on
tip of Linus' tree with only my patch applied? (after adjusting tags
like you suggested).
Or Linus picks up my patch directly but first adjusts its tags?
Christian provided his Reviewed-by earlier today, if Linus were to
pick it up you'd do well to first provide your Reviewed-by: (which
would replace the Signed-off-by I mistakenly left attributed to you in
the updated patch).
However you all decide to skin this cat: sorry for the trouble!
Thanks,
Mike
Powered by blists - more mailing lists