]> www.infradead.org Git - users/hch/misc.git/commitdiff
nfsd: nfserr_jukebox in nlm_fopen should lead to a retry
authorOlga Kornievskaia <okorniev@redhat.com>
Thu, 21 Aug 2025 20:31:46 +0000 (16:31 -0400)
committerChuck Lever <chuck.lever@oracle.com>
Sun, 21 Sep 2025 23:24:50 +0000 (19:24 -0400)
When v3 NLM request finds a conflicting delegation, it triggers
a delegation recall and nfsd_open fails with EAGAIN. nfsd_open
then translates EAGAIN into nfserr_jukebox. In nlm_fopen, instead
of returning nlm_failed for when there is a conflicting delegation,
drop this NLM request so that the client retries. Once delegation
is recalled and if a local lock is claimed, a retry would lead to
nfsd returning a nlm_lck_blocked error or a successful nlm lock.

Fixes: d343fce148a4 ("[PATCH] knfsd: Allow lockd to drop replies as appropriate")
Cc: stable@vger.kernel.org # v6.6
Signed-off-by: Olga Kornievskaia <okorniev@redhat.com>
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
fs/nfsd/lockd.c

index edc9f75dc75c6d1ac8518c3e4c26a4c21b001d5d..6b042218668b822d63a07b6581476ea1f5db003e 100644 (file)
@@ -57,6 +57,21 @@ nlm_fopen(struct svc_rqst *rqstp, struct nfs_fh *f, struct file **filp,
        switch (nfserr) {
        case nfs_ok:
                return 0;
+       case nfserr_jukebox:
+               /* this error can indicate a presence of a conflicting
+                * delegation to an NLM lock request. Options are:
+                * (1) For now, drop this request and make the client
+                * retry. When delegation is returned, client's lock retry
+                * will complete.
+                * (2) NLM4_DENIED as per "spec" signals to the client
+                * that the lock is unavailable now but client can retry.
+                * Linux client implementation does not. It treats
+                * NLM4_DENIED same as NLM4_FAILED and errors the request.
+                * (3) For the future, treat this as blocked lock and try
+                * to callback when the delegation is returned but might
+                * not have a proper lock request to block on.
+                */
+               fallthrough;
        case nfserr_dropit:
                return nlm_drop_reply;
        case nfserr_stale: