The branch main has been updated by rmacklem:

URL: 
https://cgit.FreeBSD.org/src/commit/?id=392a82b25405760b61c22d8342d107ae4a724e65

commit 392a82b25405760b61c22d8342d107ae4a724e65
Author:     Rick Macklem <rmack...@freebsd.org>
AuthorDate: 2025-07-19 23:40:29 +0000
Commit:     Rick Macklem <rmack...@freebsd.org>
CommitDate: 2025-07-19 23:40:29 +0000

    nfsd: Fix handling of VOP_PATHCONF() for attribute replies
    
    des@ reported a panic in the NFSv4 server, where
    nfsv4_fillattr() did a VOP_PATHCONF() without having
    "vp" locked.
    
    Relocking the vnode is inefficient and, for Readdir,
    may cause deadlocks. As such, this patch handles
    VOP_PATHCONF() in the same way that the code checks
    for ACL support, by doing the VOP_PATHCONF() before
    the calls to nfsv4_filllattr() where the vnode is still locked.
    
    Reported by:    des
    Reviewed by:    kib (earlier version)
    Differential Revision:  https://reviews.freebsd.org/D51410
    Fixes:  c5d72d29fe0e ("nfsv4: Add support for the NFSv4 hidden and system 
attributes")
---
 sys/fs/nfs/nfs_commonsubs.c     | 17 +++++------------
 sys/fs/nfs/nfs_var.h            |  6 ++++--
 sys/fs/nfsclient/nfs_clrpcops.c |  3 ++-
 sys/fs/nfsclient/nfs_clstate.c  |  2 +-
 sys/fs/nfsserver/nfs_nfsdport.c | 29 +++++++++++++++++++++++------
 sys/fs/nfsserver/nfs_nfsdserv.c | 17 +++++++++++++++--
 6 files changed, 50 insertions(+), 24 deletions(-)

diff --git a/sys/fs/nfs/nfs_commonsubs.c b/sys/fs/nfs/nfs_commonsubs.c
index a3580d7f71f5..a957315aaa12 100644
--- a/sys/fs/nfs/nfs_commonsubs.c
+++ b/sys/fs/nfs/nfs_commonsubs.c
@@ -648,7 +648,7 @@ nfscl_fillsattr(struct nfsrv_descript *nd, struct vattr 
*vap,
                        NFSSETBIT_ATTRBIT(&attrbits, NFSATTRBIT_TIMECREATE);
                (void) nfsv4_fillattr(nd, vp->v_mount, vp, NULL, vap, NULL, 0,
                    &attrbits, NULL, NULL, 0, 0, 0, 0, (uint64_t)0, NULL,
-                   false);
+                   false, false, false);
                break;
        }
 }
@@ -2647,7 +2647,8 @@ nfsv4_fillattr(struct nfsrv_descript *nd, struct mount 
*mp, vnode_t vp,
     NFSACL_T *saclp, struct vattr *vap, fhandle_t *fhp, int rderror,
     nfsattrbit_t *attrbitp, struct ucred *cred, NFSPROC_T *p, int isdgram,
     int reterr, int supports_nfsv4acls, int at_root, uint64_t 
mounted_on_fileno,
-    struct statfs *pnfssf, bool xattrsupp)
+    struct statfs *pnfssf, bool xattrsupp, bool has_hiddensystem,
+    bool has_namedattr)
 {
        int bitpos, retnum = 0;
        u_int32_t *tl;
@@ -2662,7 +2663,6 @@ nfsv4_fillattr(struct nfsrv_descript *nd, struct mount 
*mp, vnode_t vp,
        struct timespec temptime;
        NFSACL_T *aclp, *naclp = NULL;
        short irflag;
-       long has_pathconf;
 #ifdef QUOTA
        struct dqblk dqb;
        uid_t savuid;
@@ -2767,11 +2767,7 @@ nfsv4_fillattr(struct nfsrv_descript *nd, struct mount 
*mp, vnode_t vp,
                            NFSCLRBIT_ATTRBIT(&attrbits,NFSATTRBIT_ACLSUPPORT);
                            NFSCLRBIT_ATTRBIT(&attrbits,NFSATTRBIT_ACL);
                        }
-                       if (cred == NULL || p == NULL || vp == NULL ||
-                           VOP_PATHCONF(vp, _PC_HAS_HIDDENSYSTEM,
-                           &has_pathconf) != 0)
-                           has_pathconf = 0;
-                       if (has_pathconf == 0) {
+                       if (!has_hiddensystem) {
                            NFSCLRBIT_ATTRBIT(&attrbits, NFSATTRBIT_HIDDEN);
                            NFSCLRBIT_ATTRBIT(&attrbits, NFSATTRBIT_SYSTEM);
                        }
@@ -2815,10 +2811,7 @@ nfsv4_fillattr(struct nfsrv_descript *nd, struct mount 
*mp, vnode_t vp,
                        break;
                case NFSATTRBIT_NAMEDATTR:
                        NFSM_BUILD(tl, u_int32_t *, NFSX_UNSIGNED);
-                       if (VOP_PATHCONF(vp, _PC_HAS_NAMEDATTR,
-                           &has_pathconf) != 0)
-                               has_pathconf = 0;
-                       if (has_pathconf != 0)
+                       if (has_namedattr)
                                *tl = newnfs_true;
                        else
                                *tl = newnfs_false;
diff --git a/sys/fs/nfs/nfs_var.h b/sys/fs/nfs/nfs_var.h
index a1f40e2b8c13..54f60a753c50 100644
--- a/sys/fs/nfs/nfs_var.h
+++ b/sys/fs/nfs/nfs_var.h
@@ -396,7 +396,8 @@ void nfsrv_wcc(struct nfsrv_descript *, int, struct 
nfsvattr *, int,
     struct nfsvattr *);
 int nfsv4_fillattr(struct nfsrv_descript *, struct mount *, vnode_t, NFSACL_T 
*,
     struct vattr *, fhandle_t *, int, nfsattrbit_t *, struct ucred *,
-    NFSPROC_T *, int, int, int, int, uint64_t, struct statfs *, bool);
+    NFSPROC_T *, int, int, int, int, uint64_t, struct statfs *, bool, bool,
+    bool);
 void nfsrv_fillattr(struct nfsrv_descript *, struct nfsvattr *);
 struct mbuf *nfsrv_adj(struct mbuf *, int, int);
 void nfsrv_postopattr(struct nfsrv_descript *, int, struct nfsvattr *);
@@ -735,7 +736,8 @@ int nfsvno_updfilerev(vnode_t, struct nfsvattr *, struct 
nfsrv_descript *,
     NFSPROC_T *);
 int nfsvno_fillattr(struct nfsrv_descript *, struct mount *, vnode_t,
     struct nfsvattr *, fhandle_t *, int, nfsattrbit_t *,
-    struct ucred *, NFSPROC_T *, int, int, int, int, uint64_t, bool);
+    struct ucred *, NFSPROC_T *, int, int, int, int, uint64_t, bool, bool,
+    bool);
 int nfsrv_sattr(struct nfsrv_descript *, vnode_t, struct nfsvattr *, 
nfsattrbit_t *,
     NFSACL_T *, NFSPROC_T *);
 int nfsv4_sattr(struct nfsrv_descript *, vnode_t, struct nfsvattr *, 
nfsattrbit_t *,
diff --git a/sys/fs/nfsclient/nfs_clrpcops.c b/sys/fs/nfsclient/nfs_clrpcops.c
index 9a10a51b0d33..2f3c59b68518 100644
--- a/sys/fs/nfsclient/nfs_clrpcops.c
+++ b/sys/fs/nfsclient/nfs_clrpcops.c
@@ -5436,7 +5436,8 @@ nfsrpc_setaclrpc(vnode_t vp, struct ucred *cred, 
NFSPROC_T *p,
        NFSZERO_ATTRBIT(&attrbits);
        NFSSETBIT_ATTRBIT(&attrbits, NFSATTRBIT_ACL);
        (void) nfsv4_fillattr(nd, vp->v_mount, vp, aclp, NULL, NULL, 0,
-           &attrbits, NULL, NULL, 0, 0, 0, 0, (uint64_t)0, NULL, false);
+           &attrbits, NULL, NULL, 0, 0, 0, 0, (uint64_t)0, NULL, false, false,
+           false);
        error = nfscl_request(nd, vp, p, cred);
        if (error)
                return (error);
diff --git a/sys/fs/nfsclient/nfs_clstate.c b/sys/fs/nfsclient/nfs_clstate.c
index c7e22c610a4e..99a781640c53 100644
--- a/sys/fs/nfsclient/nfs_clstate.c
+++ b/sys/fs/nfsclient/nfs_clstate.c
@@ -3701,7 +3701,7 @@ nfscl_docb(struct nfsrv_descript *nd, NFSPROC_T *p)
                        if (!error)
                                (void) nfsv4_fillattr(nd, NULL, NULL, NULL, &va,
                                    NULL, 0, &rattrbits, NULL, p, 0, 0, 0, 0,
-                                   (uint64_t)0, NULL, false);
+                                   (uint64_t)0, NULL, false, false, false);
                        break;
                case NFSV4OP_CBRECALL:
                        NFSCL_DEBUG(4, "cbrecall\n");
diff --git a/sys/fs/nfsserver/nfs_nfsdport.c b/sys/fs/nfsserver/nfs_nfsdport.c
index ef57453d11fa..4f0d5946d6b9 100644
--- a/sys/fs/nfsserver/nfs_nfsdport.c
+++ b/sys/fs/nfsserver/nfs_nfsdport.c
@@ -2113,7 +2113,7 @@ nfsvno_fillattr(struct nfsrv_descript *nd, struct mount 
*mp, struct vnode *vp,
     struct nfsvattr *nvap, fhandle_t *fhp, int rderror, nfsattrbit_t *attrbitp,
     struct ucred *cred, struct thread *p, int isdgram, int reterr,
     int supports_nfsv4acls, int at_root, uint64_t mounted_on_fileno,
-    bool xattrsupp)
+    bool xattrsupp, bool has_hiddensystem, bool has_namedattr)
 {
        struct statfs *sf;
        int error;
@@ -2132,7 +2132,7 @@ nfsvno_fillattr(struct nfsrv_descript *nd, struct mount 
*mp, struct vnode *vp,
        }
        error = nfsv4_fillattr(nd, mp, vp, NULL, &nvap->na_vattr, fhp, rderror,
            attrbitp, cred, p, isdgram, reterr, supports_nfsv4acls, at_root,
-           mounted_on_fileno, sf, xattrsupp);
+           mounted_on_fileno, sf, xattrsupp, has_hiddensystem, has_namedattr);
        free(sf, M_TEMP);
        NFSEXITCODE2(0, nd);
        return (error);
@@ -2464,7 +2464,8 @@ nfsrvd_readdirplus(struct nfsrv_descript *nd, int isdgram,
        struct thread *p = curthread;
        int bextpg0, bextpg1, bextpgsiz0, bextpgsiz1;
        size_t atsiz;
-       bool xattrsupp;
+       long pathval;
+       bool has_hiddensystem, has_namedattr, xattrsupp;
 
        if (nd->nd_repstat) {
                nfsrv_postopattr(nd, getret, &at);
@@ -2940,6 +2941,8 @@ again:
                                txdr_hyper(*cookiep, tl);
                                dirlen += nfsm_strtom(nd, dp->d_name, nlen);
                                xattrsupp = false;
+                               has_hiddensystem = false;
+                               has_namedattr = false;
                                if (nvp != NULL) {
                                        supports_nfsv4acls =
                                            nfs_supportsnfsv4acls(nvp);
@@ -2951,6 +2954,18 @@ again:
                                                    nd->nd_cred, p);
                                                xattrsupp = ret != EOPNOTSUPP;
                                        }
+                                       if (VOP_PATHCONF(nvp,
+                                           _PC_HAS_HIDDENSYSTEM, &pathval) !=
+                                           0)
+                                               pathval = 0;
+                                       has_hiddensystem = pathval > 0;
+                                       pathval = 0;
+                                       if (NFSISSET_ATTRBIT(&attrbits,
+                                           NFSATTRBIT_NAMEDATTR) &&
+                                           VOP_PATHCONF(nvp, _PC_HAS_NAMEDATTR,
+                                           &pathval) != 0)
+                                               pathval = 0;
+                                       has_namedattr = pathval > 0;
                                        NFSVOPUNLOCK(nvp);
                                } else
                                        supports_nfsv4acls = 0;
@@ -2970,13 +2985,15 @@ again:
                                            nvp, nvap, &nfh, r, &rderrbits,
                                            nd->nd_cred, p, isdgram, 0,
                                            supports_nfsv4acls, at_root,
-                                           mounted_on_fileno, xattrsupp);
+                                           mounted_on_fileno, xattrsupp,
+                                           has_hiddensystem, has_namedattr);
                                } else {
                                        dirlen += nfsvno_fillattr(nd, new_mp,
                                            nvp, nvap, &nfh, r, &attrbits,
                                            nd->nd_cred, p, isdgram, 0,
                                            supports_nfsv4acls, at_root,
-                                           mounted_on_fileno, xattrsupp);
+                                           mounted_on_fileno, xattrsupp,
+                                           has_hiddensystem, has_namedattr);
                                }
                                if (nvp != NULL)
                                        vrele(nvp);
@@ -6368,7 +6385,7 @@ nfsrv_setacldsdorpc(fhandle_t *fhp, struct ucred *cred, 
NFSPROC_T *p,
         * the same type (VREG).
         */
        nfsv4_fillattr(nd, NULL, vp, aclp, NULL, NULL, 0, &attrbits, NULL,
-           NULL, 0, 0, 0, 0, 0, NULL, false);
+           NULL, 0, 0, 0, 0, 0, NULL, false, false, false);
        error = newnfs_request(nd, nmp, NULL, &nmp->nm_sockreq, NULL, p, cred,
            NFS_PROG, NFS_VER4, NULL, 1, NULL, NULL);
        if (error != 0) {
diff --git a/sys/fs/nfsserver/nfs_nfsdserv.c b/sys/fs/nfsserver/nfs_nfsdserv.c
index 1ae765917178..9eebcda548c6 100644
--- a/sys/fs/nfsserver/nfs_nfsdserv.c
+++ b/sys/fs/nfsserver/nfs_nfsdserv.c
@@ -251,7 +251,8 @@ nfsrvd_getattr(struct nfsrv_descript *nd, int isdgram,
        accmode_t accmode;
        struct thread *p = curthread;
        size_t atsiz;
-       bool xattrsupp;
+       long pathval;
+       bool has_hiddensystem, has_namedattr, xattrsupp;
 
        if (nd->nd_repstat)
                goto out;
@@ -318,6 +319,17 @@ nfsrvd_getattr(struct nfsrv_descript *nd, int isdgram,
                                            p);
                                        xattrsupp = ret != EOPNOTSUPP;
                                }
+                               if (VOP_PATHCONF(vp, _PC_HAS_HIDDENSYSTEM,
+                                   &pathval) != 0)
+                                       pathval = 0;
+                               has_hiddensystem = pathval > 0;
+                               pathval = 0;
+                               if (NFSISSET_ATTRBIT(&attrbits,
+                                   NFSATTRBIT_NAMEDATTR) &&
+                                   VOP_PATHCONF(vp, _PC_HAS_NAMEDATTR,
+                                   &pathval) != 0)
+                                       pathval = 0;
+                               has_namedattr = pathval > 0;
                                mp = vp->v_mount;
                                if (nfsrv_enable_crossmntpt != 0 &&
                                    vp->v_type == VDIR &&
@@ -352,7 +364,8 @@ nfsrvd_getattr(struct nfsrv_descript *nd, int isdgram,
                                            &fh, 0, &attrbits, nd->nd_cred, p,
                                            isdgram, 1, supports_nfsv4acls,
                                            at_root, mounted_on_fileno,
-                                           xattrsupp);
+                                           xattrsupp, has_hiddensystem,
+                                           has_namedattr);
                                        vfs_unbusy(mp);
                                }
                                vrele(vp);

Reply via email to