]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
nfsd: fix nfsd_file reference leak in nfsd4_add_rdaccess_to_wrdeleg()
authorChuck Lever <chuck.lever@oracle.com>
Mon, 1 Dec 2025 22:09:55 +0000 (17:09 -0500)
committerChuck Lever <chuck.lever@oracle.com>
Thu, 25 Dec 2025 02:29:24 +0000 (21:29 -0500)
nfsd4_add_rdaccess_to_wrdeleg() unconditionally overwrites
fp->fi_fds[O_RDONLY] with a newly acquired nfsd_file. However, if
the client already has a SHARE_ACCESS_READ open from a previous OPEN
operation, this action overwrites the existing pointer without
releasing its reference, orphaning the previous reference.

Additionally, the function originally stored the same nfsd_file
pointer in both fp->fi_fds[O_RDONLY] and fp->fi_rdeleg_file with
only a single reference. When put_deleg_file() runs, it clears
fi_rdeleg_file and calls nfs4_file_put_access() to release the file.

However, nfs4_file_put_access() only releases fi_fds[O_RDONLY] when
the fi_access[O_RDONLY] counter drops to zero. If another READ open
exists on the file, the counter remains elevated and the nfsd_file
reference from the delegation is never released. This potentially
causes open conflicts on that file.

Then, on server shutdown, these leaks cause __nfsd_file_cache_purge()
to encounter files with an elevated reference count that cannot be
cleaned up, ultimately triggering a BUG() in kmem_cache_destroy()
because there are still nfsd_file objects allocated in that cache.

Fixes: e7a8ebc305f2 ("NFSD: Offer write delegation for OPEN with OPEN4_SHARE_ACCESS_WRITE")
Cc: stable@vger.kernel.org
Reviewed-by: Jeff Layton <jlayton@kernel.org>
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
fs/nfsd/nfs4state.c

index 35004568d43eb27254802f6f5784a3c04c20fe08..11877b96dc4c6f643c5a7f79a509ee5096f371d8 100644 (file)
@@ -1218,8 +1218,10 @@ static void put_deleg_file(struct nfs4_file *fp)
 
        if (nf)
                nfsd_file_put(nf);
-       if (rnf)
+       if (rnf) {
+               nfsd_file_put(rnf);
                nfs4_file_put_access(fp, NFS4_SHARE_ACCESS_READ);
+       }
 }
 
 static void nfsd4_finalize_deleg_timestamps(struct nfs4_delegation *dp, struct file *f)
@@ -6231,10 +6233,14 @@ nfsd4_add_rdaccess_to_wrdeleg(struct svc_rqst *rqstp, struct nfsd4_open *open,
                fp = stp->st_stid.sc_file;
                spin_lock(&fp->fi_lock);
                __nfs4_file_get_access(fp, NFS4_SHARE_ACCESS_READ);
-               fp = stp->st_stid.sc_file;
-               fp->fi_fds[O_RDONLY] = nf;
-               fp->fi_rdeleg_file = nf;
+               if (!fp->fi_fds[O_RDONLY]) {
+                       fp->fi_fds[O_RDONLY] = nf;
+                       nf = NULL;
+               }
+               fp->fi_rdeleg_file = nfsd_file_get(fp->fi_fds[O_RDONLY]);
                spin_unlock(&fp->fi_lock);
+               if (nf)
+                       nfsd_file_put(nf);
        }
        return true;
 }