Track the union of requested and supported child attributes in the
delegation, and only encode the attributes in that union when sending
add/remove/rename updates.

Signed-off-by: Jeff Layton <jlay...@kernel.org>
---
 fs/nfsd/nfs4proc.c  |  2 ++
 fs/nfsd/nfs4state.c | 18 ++++++++++++++++++
 fs/nfsd/nfs4xdr.c   | 15 ++++++---------
 fs/nfsd/state.h     |  3 +++
 4 files changed, 29 insertions(+), 9 deletions(-)

diff --git a/fs/nfsd/nfs4proc.c b/fs/nfsd/nfs4proc.c
index 
220fc873db8f08a90ac74e51ac9d931fe7edb9e4..774d18dd2f1a31d299a8426c3462847de6c88115
 100644
--- a/fs/nfsd/nfs4proc.c
+++ b/fs/nfsd/nfs4proc.c
@@ -2385,6 +2385,8 @@ nfsd4_get_dir_delegation(struct svc_rqst *rqstp,
 
        gdd->gddrnf_status = GDD4_OK;
        memcpy(&gdd->gddr_stateid, &dd->dl_stid.sc_stateid, 
sizeof(gdd->gddr_stateid));
+       gdd->gddr_child_attributes[0] = dd->dl_child_attrs[0];
+       gdd->gddr_child_attributes[1] = dd->dl_child_attrs[1];
        nfs4_put_stid(&dd->dl_stid);
        return nfs_ok;
 }
diff --git a/fs/nfsd/nfs4state.c b/fs/nfsd/nfs4state.c
index 
9b5f559ff6125a551f73ac103f8af2e3763dc3e6..368face4d0b7001914b209b858dc1baa366535f6
 100644
--- a/fs/nfsd/nfs4state.c
+++ b/fs/nfsd/nfs4state.c
@@ -9643,6 +9643,21 @@ nfsd4_deleg_getattr_conflict(struct svc_rqst *rqstp, 
struct dentry *dentry,
        return status;
 }
 
+#define GDD_WORD0_CHILD_ATTRS  (FATTR4_WORD0_TYPE |            \
+                                FATTR4_WORD0_CHANGE |          \
+                                FATTR4_WORD0_SIZE |            \
+                                FATTR4_WORD0_FILEID |          \
+                                FATTR4_WORD0_FILEHANDLE)
+
+#define GDD_WORD1_CHILD_ATTRS  (FATTR4_WORD1_MODE |            \
+                                FATTR4_WORD1_NUMLINKS |        \
+                                FATTR4_WORD1_RAWDEV |          \
+                                FATTR4_WORD1_SPACE_USED |      \
+                                FATTR4_WORD1_TIME_ACCESS |     \
+                                FATTR4_WORD1_TIME_METADATA |   \
+                                FATTR4_WORD1_TIME_MODIFY |     \
+                                FATTR4_WORD1_TIME_CREATE)
+
 /**
  * nfsd_get_dir_deleg - attempt to get a directory delegation
  * @cstate: compound state
@@ -9689,6 +9704,9 @@ nfsd_get_dir_deleg(struct nfsd4_compound_state *cstate,
        if (!dp)
                goto out_delegees;
 
+       dp->dl_child_attrs[0] = gdd->gdda_child_attributes[0] & 
GDD_WORD0_CHILD_ATTRS;
+       dp->dl_child_attrs[1] = gdd->gdda_child_attributes[1] & 
GDD_WORD1_CHILD_ATTRS;
+
        fl = nfs4_alloc_init_lease(dp, gdd->gddr_notification[0]);
        if (!fl)
                goto out_put_stid;
diff --git a/fs/nfsd/nfs4xdr.c b/fs/nfsd/nfs4xdr.c
index 
11b622aca5111502b483f269b1fce6a684804645..0c411c758279177837c078d393048aaebf31d46f
 100644
--- a/fs/nfsd/nfs4xdr.c
+++ b/fs/nfsd/nfs4xdr.c
@@ -3837,18 +3837,15 @@ nfsd4_setup_notify_entry4(struct notify_entry4 *ne, 
struct xdr_stream *xdr,
 
        args.change_attr = nfsd4_change_attribute(&args.stat);
 
-       attrmask[0] = FATTR4_WORD0_TYPE | FATTR4_WORD0_CHANGE |
-                     FATTR4_WORD0_SIZE | FATTR4_WORD0_FILEID;
-       attrmask[1] = FATTR4_WORD1_MODE | FATTR4_WORD1_NUMLINKS | 
FATTR4_WORD1_RAWDEV |
-                     FATTR4_WORD1_SPACE_USED | FATTR4_WORD1_TIME_ACCESS |
-                     FATTR4_WORD1_TIME_METADATA | FATTR4_WORD1_TIME_MODIFY;
+       attrmask[0] = dp->dl_child_attrs[0];
+       attrmask[1] = dp->dl_child_attrs[1];
        attrmask[2] = 0;
 
-       if (setup_notify_fhandle(dentry, fi, &args))
-               attrmask[0] |= FATTR4_WORD0_FILEHANDLE;
+       if (!setup_notify_fhandle(dentry, fi, &args))
+               attrmask[0] &= ~FATTR4_WORD0_FILEHANDLE;
 
-       if (args.stat.result_mask & STATX_BTIME)
-               attrmask[1] |= FATTR4_WORD1_TIME_CREATE;
+       if (!(args.stat.result_mask & STATX_BTIME))
+               attrmask[1] &= ~FATTR4_WORD1_TIME_CREATE;
 
        ne->ne_attrs.attrmask.count = 2;
        ne->ne_attrs.attr_vals.data = (u8 *)xdr->p;
diff --git a/fs/nfsd/state.h b/fs/nfsd/state.h
index 
6e066f0721e6a48394e182b3c273a44d2fbb652d..73869ae25bcdf63cc29f9ba49bdac20e21a812bd
 100644
--- a/fs/nfsd/state.h
+++ b/fs/nfsd/state.h
@@ -281,6 +281,9 @@ struct nfs4_delegation {
        struct timespec64       dl_atime;
        struct timespec64       dl_mtime;
        struct timespec64       dl_ctime;
+
+       /* For dir delegations */
+       uint32_t                dl_child_attrs[2];
 };
 
 static inline bool deleg_is_read(u32 dl_type)

-- 
2.51.0


Reply via email to