diff mbox series

[v2,6/8] NFSv4: Support the offline bit

Message ID 20211227190504.309612-7-trondmy@kernel.org (mailing list archive)
State New, archived
Headers show
Series Support btime and other NFSv4 specific attributes | expand

Commit Message

Trond Myklebust Dec. 27, 2021, 7:05 p.m. UTC
From: Trond Myklebust <trond.myklebust@primarydata.com>

Add tracking of the NFSv4 'offline' attribute.

Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
Signed-off-by: Lance Shelton <lance.shelton@hammerspace.com>
Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com>
---
 fs/nfs/inode.c          | 11 +++++++++++
 fs/nfs/nfs4proc.c       |  6 ++++++
 fs/nfs/nfs4trace.h      |  3 ++-
 fs/nfs/nfs4xdr.c        | 31 ++++++++++++++++++++++++++++++-
 include/linux/nfs4.h    |  1 +
 include/linux/nfs_fs.h  |  1 +
 include/linux/nfs_xdr.h |  5 ++++-
 7 files changed, 55 insertions(+), 3 deletions(-)

Comments

Schumaker, Anna Jan. 7, 2022, 2:23 p.m. UTC | #1
Hi Trond,

On Tue, Dec 28, 2021 at 3:03 AM <trondmy@kernel.org> wrote:
>
> From: Trond Myklebust <trond.myklebust@primarydata.com>
>
> Add tracking of the NFSv4 'offline' attribute.
>
> Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
> Signed-off-by: Lance Shelton <lance.shelton@hammerspace.com>
> Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com>
> ---
>  fs/nfs/inode.c          | 11 +++++++++++
>  fs/nfs/nfs4proc.c       |  6 ++++++
>  fs/nfs/nfs4trace.h      |  3 ++-
>  fs/nfs/nfs4xdr.c        | 31 ++++++++++++++++++++++++++++++-
>  include/linux/nfs4.h    |  1 +
>  include/linux/nfs_fs.h  |  1 +
>  include/linux/nfs_xdr.h |  5 ++++-
>  7 files changed, 55 insertions(+), 3 deletions(-)
>
> diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c
> index 4673b091ea31..33f4410190b6 100644
> --- a/fs/nfs/inode.c
> +++ b/fs/nfs/inode.c
> @@ -528,6 +528,7 @@ nfs_fhget(struct super_block *sb, struct nfs_fh *fh, struct nfs_fattr *fattr)
>                 nfsi->archive = 0;
>                 nfsi->hidden = 0;
>                 nfsi->system = 0;
> +               nfsi->offline = 0;
>                 inode_set_iversion_raw(inode, 0);
>                 inode->i_size = 0;
>                 clear_nlink(inode);
> @@ -606,6 +607,10 @@ nfs_fhget(struct super_block *sb, struct nfs_fh *fh, struct nfs_fattr *fattr)
>                 } else if (fattr_supported & NFS_ATTR_FATTR_SPACE_USED &&
>                            fattr->size != 0)
>                         nfs_set_cache_invalid(inode, NFS_INO_INVALID_BLOCKS);
> +               if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> +                       nfsi->offline = (fattr->hsa_flags & NFS_HSA_OFFLINE) ? 1 : 0;
> +               else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
> +                       nfs_set_cache_invalid(inode, NFS_INO_INVALID_WINATTR);
>
>                 nfs_setsecurity(inode, fattr);
>
> @@ -2274,6 +2279,12 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr)
>                 nfsi->cache_validity |=
>                         save_cache_validity & NFS_INO_INVALID_BLOCKS;
>
> +       if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> +               nfsi->offline = (fattr->hsa_flags & NFS_HSA_OFFLINE) ? 1 : 0;
> +       else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
> +               nfsi->cache_validity |=
> +                       save_cache_validity & NFS_INO_INVALID_WINATTR;
> +
>         /* Update attrtimeo value if we're out of the unstable period */
>         if (attr_changed) {
>                 nfs_inc_stats(inode, NFSIOS_ATTRINVALIDATE);
> diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
> index 4e6cc54016ba..713a71fb3020 100644
> --- a/fs/nfs/nfs4proc.c
> +++ b/fs/nfs/nfs4proc.c
> @@ -219,6 +219,7 @@ const u32 nfs4_fattr_bitmap[3] = {
>  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
>         FATTR4_WORD2_SECURITY_LABEL
>  #endif
> +       | FATTR4_WORD2_OFFLINE
>  };

This won't compile if CONFIG_NFS_V4_SECURITY_LABEL=n:

fs/nfs/nfs4proc.c:218:9: error: expected expression before ‘|’ token
  218 |         | FATTR4_WORD2_OFFLINE
      |         ^
make[2]: *** [scripts/Makefile.build:287: fs/nfs/nfs4proc.o] Error 1
make[1]: *** [scripts/Makefile.build:549: fs/nfs] Error 2
make: *** [Makefile:1846: fs] Error 2
make: *** Waiting for unfinished jobs....

Anna

>
>  static const u32 nfs4_pnfs_open_bitmap[3] = {
> @@ -245,6 +246,7 @@ static const u32 nfs4_pnfs_open_bitmap[3] = {
>  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
>         | FATTR4_WORD2_SECURITY_LABEL
>  #endif
> +       | FATTR4_WORD2_OFFLINE
>  };
>
>  static const u32 nfs4_open_noattr_bitmap[3] = {
> @@ -325,6 +327,7 @@ static void nfs4_bitmap_copy_adjust(__u32 *dst, const __u32 *src,
>         if (!(cache_validity & NFS_INO_INVALID_WINATTR)) {
>                 dst[0] &= ~(FATTR4_WORD0_ARCHIVE | FATTR4_WORD0_HIDDEN);
>                 dst[1] &= ~(FATTR4_WORD1_SYSTEM | FATTR4_WORD1_TIME_BACKUP);
> +               dst[2] &= ~FATTR4_WORD2_OFFLINE;
>         }
>  }
>
> @@ -3927,6 +3930,8 @@ static int _nfs4_server_capabilities(struct nfs_server *server, struct nfs_fh *f
>                 memcpy(server->attr_bitmask_nl, res.attr_bitmask,
>                                 sizeof(server->attr_bitmask));
>                 server->attr_bitmask_nl[2] &= ~FATTR4_WORD2_SECURITY_LABEL;
> +               if (!(res.attr_bitmask[2] & FATTR4_WORD2_OFFLINE))
> +                       server->fattr_valid &= ~NFS_ATTR_FATTR_OFFLINE;
>
>                 memcpy(server->cache_consistency_bitmask, res.attr_bitmask, sizeof(server->cache_consistency_bitmask));
>                 server->cache_consistency_bitmask[0] &= FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE;
> @@ -5486,6 +5491,7 @@ static void nfs4_bitmask_set(__u32 bitmask[NFS4_BITMASK_SZ], const __u32 *src,
>         if (cache_validity & NFS_INO_INVALID_WINATTR) {
>                 bitmask[0] |= FATTR4_WORD0_ARCHIVE | FATTR4_WORD0_HIDDEN;
>                 bitmask[1] |= FATTR4_WORD1_SYSTEM | FATTR4_WORD1_TIME_BACKUP;
> +               bitmask[2] |= FATTR4_WORD2_OFFLINE;
>         }
>
>         if (cache_validity & NFS_INO_INVALID_SIZE)
> diff --git a/fs/nfs/nfs4trace.h b/fs/nfs/nfs4trace.h
> index 5e72639b469e..02c78d66c36d 100644
> --- a/fs/nfs/nfs4trace.h
> +++ b/fs/nfs/nfs4trace.h
> @@ -35,7 +35,8 @@
>                 { NFS_ATTR_FATTR_HIDDEN, "HIDDEN" }, \
>                 { NFS_ATTR_FATTR_SYSTEM, "SYSTEM" }, \
>                 { NFS_ATTR_FATTR_ARCHIVE, "ARCHIVE" }, \
> -               { NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" })
> +               { NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" }, \
> +               { NFS_ATTR_FATTR_OFFLINE, "OFFLINE" })
>
>  DECLARE_EVENT_CLASS(nfs4_clientid_event,
>                 TP_PROTO(
> diff --git a/fs/nfs/nfs4xdr.c b/fs/nfs/nfs4xdr.c
> index 280df43c5bf2..a45872f860ec 100644
> --- a/fs/nfs/nfs4xdr.c
> +++ b/fs/nfs/nfs4xdr.c
> @@ -1623,7 +1623,7 @@ static void encode_readdir(struct xdr_stream *xdr, const struct nfs4_readdir_arg
>                             FATTR4_WORD1_TIME_CREATE |
>                             FATTR4_WORD1_TIME_METADATA |
>                             FATTR4_WORD1_TIME_MODIFY;
> -               attrs[2] |= FATTR4_WORD2_SECURITY_LABEL;
> +               attrs[2] |= FATTR4_WORD2_SECURITY_LABEL | FATTR4_WORD2_OFFLINE;
>                 dircount >>= 1;
>         }
>         /* Use mounted_on_fileid only if the server supports it */
> @@ -4354,6 +4354,29 @@ static int decode_attr_xattrsupport(struct xdr_stream *xdr, uint32_t *bitmap,
>         return 0;
>  }
>
> +static int decode_attr_offline(struct xdr_stream *xdr, uint32_t *bitmap,
> +                              uint32_t *res, uint64_t *flags)
> +{
> +       int status = 0;
> +       __be32 *p;
> +
> +       if (unlikely(bitmap[2] & (FATTR4_WORD2_OFFLINE - 1U)))
> +               return -EIO;
> +       if (likely(bitmap[2] & FATTR4_WORD2_OFFLINE)) {
> +               p = xdr_inline_decode(xdr, 4);
> +               if (unlikely(!p))
> +                       return -EIO;
> +               if (be32_to_cpup(p))
> +                       *res |= NFS_HSA_OFFLINE;
> +               else
> +                       *res &= ~NFS_HSA_OFFLINE;
> +               bitmap[2] &= ~FATTR4_WORD2_OFFLINE;
> +               *flags |= NFS_ATTR_FATTR_OFFLINE;
> +       }
> +       dprintk("%s: system file: =%s\n", __func__, (*res & NFS_HSA_OFFLINE) == 0 ? "false" : "true");
> +       return status;
> +}
> +
>  static int verify_attr_len(struct xdr_stream *xdr, unsigned int savep, uint32_t attrlen)
>  {
>         unsigned int attrwords = XDR_QUADLEN(attrlen);
> @@ -4842,6 +4865,12 @@ static int decode_getfattr_attrs(struct xdr_stream *xdr, uint32_t *bitmap,
>                 fattr->valid |= status;
>         }
>
> +       status = decode_attr_offline(xdr, bitmap, &fattr->hsa_flags,
> +                                    &fattr->valid);
> +       if (status < 0)
> +               goto xdr_error;
> +
> +       status = 0;
>  xdr_error:
>         dprintk("%s: xdr returned %d\n", __func__, -status);
>         return status;
> diff --git a/include/linux/nfs4.h b/include/linux/nfs4.h
> index 5662d8be04eb..817b349c24ca 100644
> --- a/include/linux/nfs4.h
> +++ b/include/linux/nfs4.h
> @@ -460,6 +460,7 @@ enum lock_type4 {
>  #define FATTR4_WORD2_SECURITY_LABEL     (1UL << 16)
>  #define FATTR4_WORD2_MODE_UMASK                (1UL << 17)
>  #define FATTR4_WORD2_XATTR_SUPPORT     (1UL << 18)
> +#define FATTR4_WORD2_OFFLINE           (1UL << 19)
>
>  /* MDS threshold bitmap bits */
>  #define THRESHOLD_RD                    (1UL << 0)
> diff --git a/include/linux/nfs_fs.h b/include/linux/nfs_fs.h
> index 1839fed16b18..61f852ebb0a4 100644
> --- a/include/linux/nfs_fs.h
> +++ b/include/linux/nfs_fs.h
> @@ -148,6 +148,7 @@ struct nfs_inode {
>         unsigned char           archive : 1;
>         unsigned char           hidden : 1;
>         unsigned char           system : 1;
> +       unsigned char           offline : 1;
>
>         /*
>          * read_cache_jiffies is when we started read-caching this inode.
> diff --git a/include/linux/nfs_xdr.h b/include/linux/nfs_xdr.h
> index dedc7e0bf2b9..f5588bc22cb1 100644
> --- a/include/linux/nfs_xdr.h
> +++ b/include/linux/nfs_xdr.h
> @@ -21,6 +21,7 @@
>  #define NFS_HSA_HIDDEN         BIT(0)
>  #define NFS_HSA_SYSTEM         BIT(1)
>  #define NFS_HSA_ARCHIVE                BIT(2)
> +#define NFS_HSA_OFFLINE                BIT(3)
>
>  struct nfs4_string {
>         unsigned int len;
> @@ -119,6 +120,7 @@ struct nfs_fattr {
>  #define NFS_ATTR_FATTR_SYSTEM           BIT_ULL(28)
>  #define NFS_ATTR_FATTR_ARCHIVE          BIT_ULL(29)
>  #define NFS_ATTR_FATTR_TIME_BACKUP      BIT_ULL(30)
> +#define NFS_ATTR_FATTR_OFFLINE          BIT_ULL(31)
>
>  #define NFS_ATTR_FATTR (NFS_ATTR_FATTR_TYPE \
>                 | NFS_ATTR_FATTR_MODE \
> @@ -144,7 +146,8 @@ struct nfs_fattr {
>                 | NFS_ATTR_FATTR_SYSTEM \
>                 | NFS_ATTR_FATTR_ARCHIVE \
>                 | NFS_ATTR_FATTR_TIME_BACKUP \
> -               | NFS_ATTR_FATTR_V4_SECURITY_LABEL)
> +               | NFS_ATTR_FATTR_V4_SECURITY_LABEL \
> +               | NFS_ATTR_FATTR_OFFLINE)
>
>  /*
>   * Maximal number of supported layout drivers.
> --
> 2.33.1
>
Trond Myklebust Jan. 7, 2022, 2:26 p.m. UTC | #2
On Fri, 2022-01-07 at 09:23 -0500, Anna Schumaker wrote:
> Hi Trond,
> 
> On Tue, Dec 28, 2021 at 3:03 AM <trondmy@kernel.org> wrote:
> > 
> > From: Trond Myklebust <trond.myklebust@primarydata.com>
> > 
> > Add tracking of the NFSv4 'offline' attribute.
> > 
> > Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
> > Signed-off-by: Lance Shelton <lance.shelton@hammerspace.com>
> > Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com>
> > ---
> >  fs/nfs/inode.c          | 11 +++++++++++
> >  fs/nfs/nfs4proc.c       |  6 ++++++
> >  fs/nfs/nfs4trace.h      |  3 ++-
> >  fs/nfs/nfs4xdr.c        | 31 ++++++++++++++++++++++++++++++-
> >  include/linux/nfs4.h    |  1 +
> >  include/linux/nfs_fs.h  |  1 +
> >  include/linux/nfs_xdr.h |  5 ++++-
> >  7 files changed, 55 insertions(+), 3 deletions(-)
> > 
> > diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c
> > index 4673b091ea31..33f4410190b6 100644
> > --- a/fs/nfs/inode.c
> > +++ b/fs/nfs/inode.c
> > @@ -528,6 +528,7 @@ nfs_fhget(struct super_block *sb, struct nfs_fh
> > *fh, struct nfs_fattr *fattr)
> >                 nfsi->archive = 0;
> >                 nfsi->hidden = 0;
> >                 nfsi->system = 0;
> > +               nfsi->offline = 0;
> >                 inode_set_iversion_raw(inode, 0);
> >                 inode->i_size = 0;
> >                 clear_nlink(inode);
> > @@ -606,6 +607,10 @@ nfs_fhget(struct super_block *sb, struct
> > nfs_fh *fh, struct nfs_fattr *fattr)
> >                 } else if (fattr_supported &
> > NFS_ATTR_FATTR_SPACE_USED &&
> >                            fattr->size != 0)
> >                         nfs_set_cache_invalid(inode,
> > NFS_INO_INVALID_BLOCKS);
> > +               if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> > +                       nfsi->offline = (fattr->hsa_flags &
> > NFS_HSA_OFFLINE) ? 1 : 0;
> > +               else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
> > +                       nfs_set_cache_invalid(inode,
> > NFS_INO_INVALID_WINATTR);
> > 
> >                 nfs_setsecurity(inode, fattr);
> > 
> > @@ -2274,6 +2279,12 @@ static int nfs_update_inode(struct inode
> > *inode, struct nfs_fattr *fattr)
> >                 nfsi->cache_validity |=
> >                         save_cache_validity &
> > NFS_INO_INVALID_BLOCKS;
> > 
> > +       if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> > +               nfsi->offline = (fattr->hsa_flags &
> > NFS_HSA_OFFLINE) ? 1 : 0;
> > +       else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
> > +               nfsi->cache_validity |=
> > +                       save_cache_validity &
> > NFS_INO_INVALID_WINATTR;
> > +
> >         /* Update attrtimeo value if we're out of the unstable
> > period */
> >         if (attr_changed) {
> >                 nfs_inc_stats(inode, NFSIOS_ATTRINVALIDATE);
> > diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
> > index 4e6cc54016ba..713a71fb3020 100644
> > --- a/fs/nfs/nfs4proc.c
> > +++ b/fs/nfs/nfs4proc.c
> > @@ -219,6 +219,7 @@ const u32 nfs4_fattr_bitmap[3] = {
> >  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> >         FATTR4_WORD2_SECURITY_LABEL
> >  #endif
> > +       | FATTR4_WORD2_OFFLINE
> >  };
> 
> This won't compile if CONFIG_NFS_V4_SECURITY_LABEL=n:
> 
> fs/nfs/nfs4proc.c:218:9: error: expected expression before ‘|’ token
>   218 |         | FATTR4_WORD2_OFFLINE

Oh... Can you fix it up by just moving that '|' inside the #ifdef?
Schumaker, Anna Jan. 7, 2022, 2:31 p.m. UTC | #3
On Fri, Jan 7, 2022 at 9:23 AM Anna Schumaker <anna.schumaker@netapp.com> wrote:
>
> Hi Trond,
>
> On Tue, Dec 28, 2021 at 3:03 AM <trondmy@kernel.org> wrote:
> >
> > From: Trond Myklebust <trond.myklebust@primarydata.com>
> >
> > Add tracking of the NFSv4 'offline' attribute.
> >
> > Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
> > Signed-off-by: Lance Shelton <lance.shelton@hammerspace.com>
> > Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com>
> > ---
> >  fs/nfs/inode.c          | 11 +++++++++++
> >  fs/nfs/nfs4proc.c       |  6 ++++++
> >  fs/nfs/nfs4trace.h      |  3 ++-
> >  fs/nfs/nfs4xdr.c        | 31 ++++++++++++++++++++++++++++++-
> >  include/linux/nfs4.h    |  1 +
> >  include/linux/nfs_fs.h  |  1 +
> >  include/linux/nfs_xdr.h |  5 ++++-
> >  7 files changed, 55 insertions(+), 3 deletions(-)
> >
> > diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c
> > index 4673b091ea31..33f4410190b6 100644
> > --- a/fs/nfs/inode.c
> > +++ b/fs/nfs/inode.c
> > @@ -528,6 +528,7 @@ nfs_fhget(struct super_block *sb, struct nfs_fh *fh, struct nfs_fattr *fattr)
> >                 nfsi->archive = 0;
> >                 nfsi->hidden = 0;
> >                 nfsi->system = 0;
> > +               nfsi->offline = 0;
> >                 inode_set_iversion_raw(inode, 0);
> >                 inode->i_size = 0;
> >                 clear_nlink(inode);
> > @@ -606,6 +607,10 @@ nfs_fhget(struct super_block *sb, struct nfs_fh *fh, struct nfs_fattr *fattr)
> >                 } else if (fattr_supported & NFS_ATTR_FATTR_SPACE_USED &&
> >                            fattr->size != 0)
> >                         nfs_set_cache_invalid(inode, NFS_INO_INVALID_BLOCKS);
> > +               if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> > +                       nfsi->offline = (fattr->hsa_flags & NFS_HSA_OFFLINE) ? 1 : 0;
> > +               else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
> > +                       nfs_set_cache_invalid(inode, NFS_INO_INVALID_WINATTR);
> >
> >                 nfs_setsecurity(inode, fattr);
> >
> > @@ -2274,6 +2279,12 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr)
> >                 nfsi->cache_validity |=
> >                         save_cache_validity & NFS_INO_INVALID_BLOCKS;
> >
> > +       if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> > +               nfsi->offline = (fattr->hsa_flags & NFS_HSA_OFFLINE) ? 1 : 0;
> > +       else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
> > +               nfsi->cache_validity |=
> > +                       save_cache_validity & NFS_INO_INVALID_WINATTR;
> > +
> >         /* Update attrtimeo value if we're out of the unstable period */
> >         if (attr_changed) {
> >                 nfs_inc_stats(inode, NFSIOS_ATTRINVALIDATE);
> > diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
> > index 4e6cc54016ba..713a71fb3020 100644
> > --- a/fs/nfs/nfs4proc.c
> > +++ b/fs/nfs/nfs4proc.c
> > @@ -219,6 +219,7 @@ const u32 nfs4_fattr_bitmap[3] = {
> >  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> >         FATTR4_WORD2_SECURITY_LABEL
> >  #endif
> > +       | FATTR4_WORD2_OFFLINE
> >  };
>
> This won't compile if CONFIG_NFS_V4_SECURITY_LABEL=n:
>
> fs/nfs/nfs4proc.c:218:9: error: expected expression before ‘|’ token
>   218 |         | FATTR4_WORD2_OFFLINE
>       |         ^
> make[2]: *** [scripts/Makefile.build:287: fs/nfs/nfs4proc.o] Error 1
> make[1]: *** [scripts/Makefile.build:549: fs/nfs] Error 2
> make: *** [Makefile:1846: fs] Error 2
> make: *** Waiting for unfinished jobs....

I can fix it up by doing:

diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
index 5d3dd647dbf6..9bd7a785eb1b 100644
--- a/fs/nfs/nfs4proc.c
+++ b/fs/nfs/nfs4proc.c
@@ -212,10 +212,10 @@ const u32 nfs4_fattr_bitmap[3] = {
        | FATTR4_WORD1_TIME_METADATA
        | FATTR4_WORD1_TIME_MODIFY
        | FATTR4_WORD1_MOUNTED_ON_FILEID,
+       FATTR4_WORD2_OFFLINE
 #ifdef CONFIG_NFS_V4_SECURITY_LABEL
-       FATTR4_WORD2_SECURITY_LABEL
+       | FATTR4_WORD2_SECURITY_LABEL
 #endif
-       | FATTR4_WORD2_OFFLINE
 };

 static const u32 nfs4_pnfs_open_bitmap[3] = {

If that works for you I can squash it into your patch in my tree

Anna

>
> Anna
>
> >
> >  static const u32 nfs4_pnfs_open_bitmap[3] = {
> > @@ -245,6 +246,7 @@ static const u32 nfs4_pnfs_open_bitmap[3] = {
> >  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> >         | FATTR4_WORD2_SECURITY_LABEL
> >  #endif
> > +       | FATTR4_WORD2_OFFLINE
> >  };
> >
> >  static const u32 nfs4_open_noattr_bitmap[3] = {
> > @@ -325,6 +327,7 @@ static void nfs4_bitmap_copy_adjust(__u32 *dst, const __u32 *src,
> >         if (!(cache_validity & NFS_INO_INVALID_WINATTR)) {
> >                 dst[0] &= ~(FATTR4_WORD0_ARCHIVE | FATTR4_WORD0_HIDDEN);
> >                 dst[1] &= ~(FATTR4_WORD1_SYSTEM | FATTR4_WORD1_TIME_BACKUP);
> > +               dst[2] &= ~FATTR4_WORD2_OFFLINE;
> >         }
> >  }
> >
> > @@ -3927,6 +3930,8 @@ static int _nfs4_server_capabilities(struct nfs_server *server, struct nfs_fh *f
> >                 memcpy(server->attr_bitmask_nl, res.attr_bitmask,
> >                                 sizeof(server->attr_bitmask));
> >                 server->attr_bitmask_nl[2] &= ~FATTR4_WORD2_SECURITY_LABEL;
> > +               if (!(res.attr_bitmask[2] & FATTR4_WORD2_OFFLINE))
> > +                       server->fattr_valid &= ~NFS_ATTR_FATTR_OFFLINE;
> >
> >                 memcpy(server->cache_consistency_bitmask, res.attr_bitmask, sizeof(server->cache_consistency_bitmask));
> >                 server->cache_consistency_bitmask[0] &= FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE;
> > @@ -5486,6 +5491,7 @@ static void nfs4_bitmask_set(__u32 bitmask[NFS4_BITMASK_SZ], const __u32 *src,
> >         if (cache_validity & NFS_INO_INVALID_WINATTR) {
> >                 bitmask[0] |= FATTR4_WORD0_ARCHIVE | FATTR4_WORD0_HIDDEN;
> >                 bitmask[1] |= FATTR4_WORD1_SYSTEM | FATTR4_WORD1_TIME_BACKUP;
> > +               bitmask[2] |= FATTR4_WORD2_OFFLINE;
> >         }
> >
> >         if (cache_validity & NFS_INO_INVALID_SIZE)
> > diff --git a/fs/nfs/nfs4trace.h b/fs/nfs/nfs4trace.h
> > index 5e72639b469e..02c78d66c36d 100644
> > --- a/fs/nfs/nfs4trace.h
> > +++ b/fs/nfs/nfs4trace.h
> > @@ -35,7 +35,8 @@
> >                 { NFS_ATTR_FATTR_HIDDEN, "HIDDEN" }, \
> >                 { NFS_ATTR_FATTR_SYSTEM, "SYSTEM" }, \
> >                 { NFS_ATTR_FATTR_ARCHIVE, "ARCHIVE" }, \
> > -               { NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" })
> > +               { NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" }, \
> > +               { NFS_ATTR_FATTR_OFFLINE, "OFFLINE" })
> >
> >  DECLARE_EVENT_CLASS(nfs4_clientid_event,
> >                 TP_PROTO(
> > diff --git a/fs/nfs/nfs4xdr.c b/fs/nfs/nfs4xdr.c
> > index 280df43c5bf2..a45872f860ec 100644
> > --- a/fs/nfs/nfs4xdr.c
> > +++ b/fs/nfs/nfs4xdr.c
> > @@ -1623,7 +1623,7 @@ static void encode_readdir(struct xdr_stream *xdr, const struct nfs4_readdir_arg
> >                             FATTR4_WORD1_TIME_CREATE |
> >                             FATTR4_WORD1_TIME_METADATA |
> >                             FATTR4_WORD1_TIME_MODIFY;
> > -               attrs[2] |= FATTR4_WORD2_SECURITY_LABEL;
> > +               attrs[2] |= FATTR4_WORD2_SECURITY_LABEL | FATTR4_WORD2_OFFLINE;
> >                 dircount >>= 1;
> >         }
> >         /* Use mounted_on_fileid only if the server supports it */
> > @@ -4354,6 +4354,29 @@ static int decode_attr_xattrsupport(struct xdr_stream *xdr, uint32_t *bitmap,
> >         return 0;
> >  }
> >
> > +static int decode_attr_offline(struct xdr_stream *xdr, uint32_t *bitmap,
> > +                              uint32_t *res, uint64_t *flags)
> > +{
> > +       int status = 0;
> > +       __be32 *p;
> > +
> > +       if (unlikely(bitmap[2] & (FATTR4_WORD2_OFFLINE - 1U)))
> > +               return -EIO;
> > +       if (likely(bitmap[2] & FATTR4_WORD2_OFFLINE)) {
> > +               p = xdr_inline_decode(xdr, 4);
> > +               if (unlikely(!p))
> > +                       return -EIO;
> > +               if (be32_to_cpup(p))
> > +                       *res |= NFS_HSA_OFFLINE;
> > +               else
> > +                       *res &= ~NFS_HSA_OFFLINE;
> > +               bitmap[2] &= ~FATTR4_WORD2_OFFLINE;
> > +               *flags |= NFS_ATTR_FATTR_OFFLINE;
> > +       }
> > +       dprintk("%s: system file: =%s\n", __func__, (*res & NFS_HSA_OFFLINE) == 0 ? "false" : "true");
> > +       return status;
> > +}
> > +
> >  static int verify_attr_len(struct xdr_stream *xdr, unsigned int savep, uint32_t attrlen)
> >  {
> >         unsigned int attrwords = XDR_QUADLEN(attrlen);
> > @@ -4842,6 +4865,12 @@ static int decode_getfattr_attrs(struct xdr_stream *xdr, uint32_t *bitmap,
> >                 fattr->valid |= status;
> >         }
> >
> > +       status = decode_attr_offline(xdr, bitmap, &fattr->hsa_flags,
> > +                                    &fattr->valid);
> > +       if (status < 0)
> > +               goto xdr_error;
> > +
> > +       status = 0;
> >  xdr_error:
> >         dprintk("%s: xdr returned %d\n", __func__, -status);
> >         return status;
> > diff --git a/include/linux/nfs4.h b/include/linux/nfs4.h
> > index 5662d8be04eb..817b349c24ca 100644
> > --- a/include/linux/nfs4.h
> > +++ b/include/linux/nfs4.h
> > @@ -460,6 +460,7 @@ enum lock_type4 {
> >  #define FATTR4_WORD2_SECURITY_LABEL     (1UL << 16)
> >  #define FATTR4_WORD2_MODE_UMASK                (1UL << 17)
> >  #define FATTR4_WORD2_XATTR_SUPPORT     (1UL << 18)
> > +#define FATTR4_WORD2_OFFLINE           (1UL << 19)
> >
> >  /* MDS threshold bitmap bits */
> >  #define THRESHOLD_RD                    (1UL << 0)
> > diff --git a/include/linux/nfs_fs.h b/include/linux/nfs_fs.h
> > index 1839fed16b18..61f852ebb0a4 100644
> > --- a/include/linux/nfs_fs.h
> > +++ b/include/linux/nfs_fs.h
> > @@ -148,6 +148,7 @@ struct nfs_inode {
> >         unsigned char           archive : 1;
> >         unsigned char           hidden : 1;
> >         unsigned char           system : 1;
> > +       unsigned char           offline : 1;
> >
> >         /*
> >          * read_cache_jiffies is when we started read-caching this inode.
> > diff --git a/include/linux/nfs_xdr.h b/include/linux/nfs_xdr.h
> > index dedc7e0bf2b9..f5588bc22cb1 100644
> > --- a/include/linux/nfs_xdr.h
> > +++ b/include/linux/nfs_xdr.h
> > @@ -21,6 +21,7 @@
> >  #define NFS_HSA_HIDDEN         BIT(0)
> >  #define NFS_HSA_SYSTEM         BIT(1)
> >  #define NFS_HSA_ARCHIVE                BIT(2)
> > +#define NFS_HSA_OFFLINE                BIT(3)
> >
> >  struct nfs4_string {
> >         unsigned int len;
> > @@ -119,6 +120,7 @@ struct nfs_fattr {
> >  #define NFS_ATTR_FATTR_SYSTEM           BIT_ULL(28)
> >  #define NFS_ATTR_FATTR_ARCHIVE          BIT_ULL(29)
> >  #define NFS_ATTR_FATTR_TIME_BACKUP      BIT_ULL(30)
> > +#define NFS_ATTR_FATTR_OFFLINE          BIT_ULL(31)
> >
> >  #define NFS_ATTR_FATTR (NFS_ATTR_FATTR_TYPE \
> >                 | NFS_ATTR_FATTR_MODE \
> > @@ -144,7 +146,8 @@ struct nfs_fattr {
> >                 | NFS_ATTR_FATTR_SYSTEM \
> >                 | NFS_ATTR_FATTR_ARCHIVE \
> >                 | NFS_ATTR_FATTR_TIME_BACKUP \
> > -               | NFS_ATTR_FATTR_V4_SECURITY_LABEL)
> > +               | NFS_ATTR_FATTR_V4_SECURITY_LABEL \
> > +               | NFS_ATTR_FATTR_OFFLINE)
> >
> >  /*
> >   * Maximal number of supported layout drivers.
> > --
> > 2.33.1
> >
Trond Myklebust Jan. 7, 2022, 2:36 p.m. UTC | #4
On Fri, 2022-01-07 at 09:31 -0500, Anna Schumaker wrote:
> On Fri, Jan 7, 2022 at 9:23 AM Anna Schumaker
> <anna.schumaker@netapp.com> wrote:
> > 
> > Hi Trond,
> > 
> > On Tue, Dec 28, 2021 at 3:03 AM <trondmy@kernel.org> wrote:
> > > 
> > > From: Trond Myklebust <trond.myklebust@primarydata.com>
> > > 
> > > Add tracking of the NFSv4 'offline' attribute.
> > > 
> > > Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
> > > Signed-off-by: Lance Shelton <lance.shelton@hammerspace.com>
> > > Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com>
> > > ---
> > >  fs/nfs/inode.c          | 11 +++++++++++
> > >  fs/nfs/nfs4proc.c       |  6 ++++++
> > >  fs/nfs/nfs4trace.h      |  3 ++-
> > >  fs/nfs/nfs4xdr.c        | 31 ++++++++++++++++++++++++++++++-
> > >  include/linux/nfs4.h    |  1 +
> > >  include/linux/nfs_fs.h  |  1 +
> > >  include/linux/nfs_xdr.h |  5 ++++-
> > >  7 files changed, 55 insertions(+), 3 deletions(-)
> > > 
> > > diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c
> > > index 4673b091ea31..33f4410190b6 100644
> > > --- a/fs/nfs/inode.c
> > > +++ b/fs/nfs/inode.c
> > > @@ -528,6 +528,7 @@ nfs_fhget(struct super_block *sb, struct
> > > nfs_fh *fh, struct nfs_fattr *fattr)
> > >                 nfsi->archive = 0;
> > >                 nfsi->hidden = 0;
> > >                 nfsi->system = 0;
> > > +               nfsi->offline = 0;
> > >                 inode_set_iversion_raw(inode, 0);
> > >                 inode->i_size = 0;
> > >                 clear_nlink(inode);
> > > @@ -606,6 +607,10 @@ nfs_fhget(struct super_block *sb, struct
> > > nfs_fh *fh, struct nfs_fattr *fattr)
> > >                 } else if (fattr_supported &
> > > NFS_ATTR_FATTR_SPACE_USED &&
> > >                            fattr->size != 0)
> > >                         nfs_set_cache_invalid(inode,
> > > NFS_INO_INVALID_BLOCKS);
> > > +               if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> > > +                       nfsi->offline = (fattr->hsa_flags &
> > > NFS_HSA_OFFLINE) ? 1 : 0;
> > > +               else if (fattr_supported &
> > > NFS_ATTR_FATTR_OFFLINE)
> > > +                       nfs_set_cache_invalid(inode,
> > > NFS_INO_INVALID_WINATTR);
> > > 
> > >                 nfs_setsecurity(inode, fattr);
> > > 
> > > @@ -2274,6 +2279,12 @@ static int nfs_update_inode(struct inode
> > > *inode, struct nfs_fattr *fattr)
> > >                 nfsi->cache_validity |=
> > >                         save_cache_validity &
> > > NFS_INO_INVALID_BLOCKS;
> > > 
> > > +       if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> > > +               nfsi->offline = (fattr->hsa_flags &
> > > NFS_HSA_OFFLINE) ? 1 : 0;
> > > +       else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
> > > +               nfsi->cache_validity |=
> > > +                       save_cache_validity &
> > > NFS_INO_INVALID_WINATTR;
> > > +
> > >         /* Update attrtimeo value if we're out of the unstable
> > > period */
> > >         if (attr_changed) {
> > >                 nfs_inc_stats(inode, NFSIOS_ATTRINVALIDATE);
> > > diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
> > > index 4e6cc54016ba..713a71fb3020 100644
> > > --- a/fs/nfs/nfs4proc.c
> > > +++ b/fs/nfs/nfs4proc.c
> > > @@ -219,6 +219,7 @@ const u32 nfs4_fattr_bitmap[3] = {
> > >  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> > >         FATTR4_WORD2_SECURITY_LABEL
> > >  #endif
> > > +       | FATTR4_WORD2_OFFLINE
> > >  };
> > 
> > This won't compile if CONFIG_NFS_V4_SECURITY_LABEL=n:
> > 
> > fs/nfs/nfs4proc.c:218:9: error: expected expression before ‘|’
> > token
> >   218 |         | FATTR4_WORD2_OFFLINE
> >       |         ^
> > make[2]: *** [scripts/Makefile.build:287: fs/nfs/nfs4proc.o] Error
> > 1
> > make[1]: *** [scripts/Makefile.build:549: fs/nfs] Error 2
> > make: *** [Makefile:1846: fs] Error 2
> > make: *** Waiting for unfinished jobs....
> 
> I can fix it up by doing:
> 
> diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
> index 5d3dd647dbf6..9bd7a785eb1b 100644
> --- a/fs/nfs/nfs4proc.c
> +++ b/fs/nfs/nfs4proc.c
> @@ -212,10 +212,10 @@ const u32 nfs4_fattr_bitmap[3] = {
>         | FATTR4_WORD1_TIME_METADATA
>         | FATTR4_WORD1_TIME_MODIFY
>         | FATTR4_WORD1_MOUNTED_ON_FILEID,
> +       FATTR4_WORD2_OFFLINE
>  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> -       FATTR4_WORD2_SECURITY_LABEL
> +       | FATTR4_WORD2_SECURITY_LABEL
>  #endif
> -       | FATTR4_WORD2_OFFLINE
>  };
> 
>  static const u32 nfs4_pnfs_open_bitmap[3] = {
> 
> If that works for you I can squash it into your patch in my tree

Sure. I was thinking the opposite, but this works too.

i.e. my suggestion was just

#ifdef CONFIG_NFS_V4_SECURITY_LABEL
        FATTR4_WORD2_SECURITY_LABEL |
#endif
        FATTR4_WORD2_OFFLINE

However there is no functional difference between the two, so let's go
with yours.

> 
> Anna
> 
> > 
> > Anna
> > 
> > > 
> > >  static const u32 nfs4_pnfs_open_bitmap[3] = {
> > > @@ -245,6 +246,7 @@ static const u32 nfs4_pnfs_open_bitmap[3] = {
> > >  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> > >         | FATTR4_WORD2_SECURITY_LABEL
> > >  #endif
> > > +       | FATTR4_WORD2_OFFLINE
> > >  };
> > > 
> > >  static const u32 nfs4_open_noattr_bitmap[3] = {
> > > @@ -325,6 +327,7 @@ static void nfs4_bitmap_copy_adjust(__u32
> > > *dst, const __u32 *src,
> > >         if (!(cache_validity & NFS_INO_INVALID_WINATTR)) {
> > >                 dst[0] &= ~(FATTR4_WORD0_ARCHIVE |
> > > FATTR4_WORD0_HIDDEN);
> > >                 dst[1] &= ~(FATTR4_WORD1_SYSTEM |
> > > FATTR4_WORD1_TIME_BACKUP);
> > > +               dst[2] &= ~FATTR4_WORD2_OFFLINE;
> > >         }
> > >  }
> > > 
> > > @@ -3927,6 +3930,8 @@ static int _nfs4_server_capabilities(struct
> > > nfs_server *server, struct nfs_fh *f
> > >                 memcpy(server->attr_bitmask_nl, res.attr_bitmask,
> > >                                 sizeof(server->attr_bitmask));
> > >                 server->attr_bitmask_nl[2] &=
> > > ~FATTR4_WORD2_SECURITY_LABEL;
> > > +               if (!(res.attr_bitmask[2] &
> > > FATTR4_WORD2_OFFLINE))
> > > +                       server->fattr_valid &=
> > > ~NFS_ATTR_FATTR_OFFLINE;
> > > 
> > >                 memcpy(server->cache_consistency_bitmask,
> > > res.attr_bitmask, sizeof(server->cache_consistency_bitmask));
> > >                 server->cache_consistency_bitmask[0] &=
> > > FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE;
> > > @@ -5486,6 +5491,7 @@ static void nfs4_bitmask_set(__u32
> > > bitmask[NFS4_BITMASK_SZ], const __u32 *src,
> > >         if (cache_validity & NFS_INO_INVALID_WINATTR) {
> > >                 bitmask[0] |= FATTR4_WORD0_ARCHIVE |
> > > FATTR4_WORD0_HIDDEN;
> > >                 bitmask[1] |= FATTR4_WORD1_SYSTEM |
> > > FATTR4_WORD1_TIME_BACKUP;
> > > +               bitmask[2] |= FATTR4_WORD2_OFFLINE;
> > >         }
> > > 
> > >         if (cache_validity & NFS_INO_INVALID_SIZE)
> > > diff --git a/fs/nfs/nfs4trace.h b/fs/nfs/nfs4trace.h
> > > index 5e72639b469e..02c78d66c36d 100644
> > > --- a/fs/nfs/nfs4trace.h
> > > +++ b/fs/nfs/nfs4trace.h
> > > @@ -35,7 +35,8 @@
> > >                 { NFS_ATTR_FATTR_HIDDEN, "HIDDEN" }, \
> > >                 { NFS_ATTR_FATTR_SYSTEM, "SYSTEM" }, \
> > >                 { NFS_ATTR_FATTR_ARCHIVE, "ARCHIVE" }, \
> > > -               { NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" })
> > > +               { NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" }, \
> > > +               { NFS_ATTR_FATTR_OFFLINE, "OFFLINE" })
> > > 
> > >  DECLARE_EVENT_CLASS(nfs4_clientid_event,
> > >                 TP_PROTO(
> > > diff --git a/fs/nfs/nfs4xdr.c b/fs/nfs/nfs4xdr.c
> > > index 280df43c5bf2..a45872f860ec 100644
> > > --- a/fs/nfs/nfs4xdr.c
> > > +++ b/fs/nfs/nfs4xdr.c
> > > @@ -1623,7 +1623,7 @@ static void encode_readdir(struct
> > > xdr_stream *xdr, const struct nfs4_readdir_arg
> > >                             FATTR4_WORD1_TIME_CREATE |
> > >                             FATTR4_WORD1_TIME_METADATA |
> > >                             FATTR4_WORD1_TIME_MODIFY;
> > > -               attrs[2] |= FATTR4_WORD2_SECURITY_LABEL;
> > > +               attrs[2] |= FATTR4_WORD2_SECURITY_LABEL |
> > > FATTR4_WORD2_OFFLINE;
> > >                 dircount >>= 1;
> > >         }
> > >         /* Use mounted_on_fileid only if the server supports it
> > > */
> > > @@ -4354,6 +4354,29 @@ static int decode_attr_xattrsupport(struct
> > > xdr_stream *xdr, uint32_t *bitmap,
> > >         return 0;
> > >  }
> > > 
> > > +static int decode_attr_offline(struct xdr_stream *xdr, uint32_t
> > > *bitmap,
> > > +                              uint32_t *res, uint64_t *flags)
> > > +{
> > > +       int status = 0;
> > > +       __be32 *p;
> > > +
> > > +       if (unlikely(bitmap[2] & (FATTR4_WORD2_OFFLINE - 1U)))
> > > +               return -EIO;
> > > +       if (likely(bitmap[2] & FATTR4_WORD2_OFFLINE)) {
> > > +               p = xdr_inline_decode(xdr, 4);
> > > +               if (unlikely(!p))
> > > +                       return -EIO;
> > > +               if (be32_to_cpup(p))
> > > +                       *res |= NFS_HSA_OFFLINE;
> > > +               else
> > > +                       *res &= ~NFS_HSA_OFFLINE;
> > > +               bitmap[2] &= ~FATTR4_WORD2_OFFLINE;
> > > +               *flags |= NFS_ATTR_FATTR_OFFLINE;
> > > +       }
> > > +       dprintk("%s: system file: =%s\n", __func__, (*res &
> > > NFS_HSA_OFFLINE) == 0 ? "false" : "true");
> > > +       return status;
> > > +}
> > > +
> > >  static int verify_attr_len(struct xdr_stream *xdr, unsigned int
> > > savep, uint32_t attrlen)
> > >  {
> > >         unsigned int attrwords = XDR_QUADLEN(attrlen);
> > > @@ -4842,6 +4865,12 @@ static int decode_getfattr_attrs(struct
> > > xdr_stream *xdr, uint32_t *bitmap,
> > >                 fattr->valid |= status;
> > >         }
> > > 
> > > +       status = decode_attr_offline(xdr, bitmap, &fattr-
> > > >hsa_flags,
> > > +                                    &fattr->valid);
> > > +       if (status < 0)
> > > +               goto xdr_error;
> > > +
> > > +       status = 0;
> > >  xdr_error:
> > >         dprintk("%s: xdr returned %d\n", __func__, -status);
> > >         return status;
> > > diff --git a/include/linux/nfs4.h b/include/linux/nfs4.h
> > > index 5662d8be04eb..817b349c24ca 100644
> > > --- a/include/linux/nfs4.h
> > > +++ b/include/linux/nfs4.h
> > > @@ -460,6 +460,7 @@ enum lock_type4 {
> > >  #define FATTR4_WORD2_SECURITY_LABEL     (1UL << 16)
> > >  #define FATTR4_WORD2_MODE_UMASK                (1UL << 17)
> > >  #define FATTR4_WORD2_XATTR_SUPPORT     (1UL << 18)
> > > +#define FATTR4_WORD2_OFFLINE           (1UL << 19)
> > > 
> > >  /* MDS threshold bitmap bits */
> > >  #define THRESHOLD_RD                    (1UL << 0)
> > > diff --git a/include/linux/nfs_fs.h b/include/linux/nfs_fs.h
> > > index 1839fed16b18..61f852ebb0a4 100644
> > > --- a/include/linux/nfs_fs.h
> > > +++ b/include/linux/nfs_fs.h
> > > @@ -148,6 +148,7 @@ struct nfs_inode {
> > >         unsigned char           archive : 1;
> > >         unsigned char           hidden : 1;
> > >         unsigned char           system : 1;
> > > +       unsigned char           offline : 1;
> > > 
> > >         /*
> > >          * read_cache_jiffies is when we started read-caching
> > > this inode.
> > > diff --git a/include/linux/nfs_xdr.h b/include/linux/nfs_xdr.h
> > > index dedc7e0bf2b9..f5588bc22cb1 100644
> > > --- a/include/linux/nfs_xdr.h
> > > +++ b/include/linux/nfs_xdr.h
> > > @@ -21,6 +21,7 @@
> > >  #define NFS_HSA_HIDDEN         BIT(0)
> > >  #define NFS_HSA_SYSTEM         BIT(1)
> > >  #define NFS_HSA_ARCHIVE                BIT(2)
> > > +#define NFS_HSA_OFFLINE                BIT(3)
> > > 
> > >  struct nfs4_string {
> > >         unsigned int len;
> > > @@ -119,6 +120,7 @@ struct nfs_fattr {
> > >  #define NFS_ATTR_FATTR_SYSTEM           BIT_ULL(28)
> > >  #define NFS_ATTR_FATTR_ARCHIVE          BIT_ULL(29)
> > >  #define NFS_ATTR_FATTR_TIME_BACKUP      BIT_ULL(30)
> > > +#define NFS_ATTR_FATTR_OFFLINE          BIT_ULL(31)
> > > 
> > >  #define NFS_ATTR_FATTR (NFS_ATTR_FATTR_TYPE \
> > >                 | NFS_ATTR_FATTR_MODE \
> > > @@ -144,7 +146,8 @@ struct nfs_fattr {
> > >                 | NFS_ATTR_FATTR_SYSTEM \
> > >                 | NFS_ATTR_FATTR_ARCHIVE \
> > >                 | NFS_ATTR_FATTR_TIME_BACKUP \
> > > -               | NFS_ATTR_FATTR_V4_SECURITY_LABEL)
> > > +               | NFS_ATTR_FATTR_V4_SECURITY_LABEL \
> > > +               | NFS_ATTR_FATTR_OFFLINE)
> > > 
> > >  /*
> > >   * Maximal number of supported layout drivers.
> > > --
> > > 2.33.1
> > >
Schumaker, Anna Jan. 7, 2022, 8:36 p.m. UTC | #5
On Fri, Jan 7, 2022 at 3:17 PM Trond Myklebust <trondmy@hammerspace.com> wrote:
>
> On Fri, 2022-01-07 at 09:31 -0500, Anna Schumaker wrote:
> > On Fri, Jan 7, 2022 at 9:23 AM Anna Schumaker
> > <anna.schumaker@netapp.com> wrote:
> > >
> > > Hi Trond,
> > >
> > > On Tue, Dec 28, 2021 at 3:03 AM <trondmy@kernel.org> wrote:
> > > >
> > > > From: Trond Myklebust <trond.myklebust@primarydata.com>
> > > >
> > > > Add tracking of the NFSv4 'offline' attribute.
> > > >
> > > > Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
> > > > Signed-off-by: Lance Shelton <lance.shelton@hammerspace.com>
> > > > Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com>
> > > > ---
> > > >  fs/nfs/inode.c          | 11 +++++++++++
> > > >  fs/nfs/nfs4proc.c       |  6 ++++++
> > > >  fs/nfs/nfs4trace.h      |  3 ++-
> > > >  fs/nfs/nfs4xdr.c        | 31 ++++++++++++++++++++++++++++++-
> > > >  include/linux/nfs4.h    |  1 +
> > > >  include/linux/nfs_fs.h  |  1 +
> > > >  include/linux/nfs_xdr.h |  5 ++++-
> > > >  7 files changed, 55 insertions(+), 3 deletions(-)
> > > >
> > > > diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c
> > > > index 4673b091ea31..33f4410190b6 100644
> > > > --- a/fs/nfs/inode.c
> > > > +++ b/fs/nfs/inode.c
> > > > @@ -528,6 +528,7 @@ nfs_fhget(struct super_block *sb, struct
> > > > nfs_fh *fh, struct nfs_fattr *fattr)
> > > >                 nfsi->archive = 0;
> > > >                 nfsi->hidden = 0;
> > > >                 nfsi->system = 0;
> > > > +               nfsi->offline = 0;
> > > >                 inode_set_iversion_raw(inode, 0);
> > > >                 inode->i_size = 0;
> > > >                 clear_nlink(inode);
> > > > @@ -606,6 +607,10 @@ nfs_fhget(struct super_block *sb, struct
> > > > nfs_fh *fh, struct nfs_fattr *fattr)
> > > >                 } else if (fattr_supported &
> > > > NFS_ATTR_FATTR_SPACE_USED &&
> > > >                            fattr->size != 0)
> > > >                         nfs_set_cache_invalid(inode,
> > > > NFS_INO_INVALID_BLOCKS);
> > > > +               if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> > > > +                       nfsi->offline = (fattr->hsa_flags &
> > > > NFS_HSA_OFFLINE) ? 1 : 0;
> > > > +               else if (fattr_supported &
> > > > NFS_ATTR_FATTR_OFFLINE)
> > > > +                       nfs_set_cache_invalid(inode,
> > > > NFS_INO_INVALID_WINATTR);
> > > >
> > > >                 nfs_setsecurity(inode, fattr);
> > > >
> > > > @@ -2274,6 +2279,12 @@ static int nfs_update_inode(struct inode
> > > > *inode, struct nfs_fattr *fattr)
> > > >                 nfsi->cache_validity |=
> > > >                         save_cache_validity &
> > > > NFS_INO_INVALID_BLOCKS;
> > > >
> > > > +       if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
> > > > +               nfsi->offline = (fattr->hsa_flags &
> > > > NFS_HSA_OFFLINE) ? 1 : 0;
> > > > +       else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
> > > > +               nfsi->cache_validity |=
> > > > +                       save_cache_validity &
> > > > NFS_INO_INVALID_WINATTR;
> > > > +
> > > >         /* Update attrtimeo value if we're out of the unstable
> > > > period */
> > > >         if (attr_changed) {
> > > >                 nfs_inc_stats(inode, NFSIOS_ATTRINVALIDATE);
> > > > diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
> > > > index 4e6cc54016ba..713a71fb3020 100644
> > > > --- a/fs/nfs/nfs4proc.c
> > > > +++ b/fs/nfs/nfs4proc.c
> > > > @@ -219,6 +219,7 @@ const u32 nfs4_fattr_bitmap[3] = {
> > > >  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> > > >         FATTR4_WORD2_SECURITY_LABEL
> > > >  #endif
> > > > +       | FATTR4_WORD2_OFFLINE
> > > >  };
> > >
> > > This won't compile if CONFIG_NFS_V4_SECURITY_LABEL=n:
> > >
> > > fs/nfs/nfs4proc.c:218:9: error: expected expression before ‘|’
> > > token
> > >   218 |         | FATTR4_WORD2_OFFLINE
> > >       |         ^
> > > make[2]: *** [scripts/Makefile.build:287: fs/nfs/nfs4proc.o] Error
> > > 1
> > > make[1]: *** [scripts/Makefile.build:549: fs/nfs] Error 2
> > > make: *** [Makefile:1846: fs] Error 2
> > > make: *** Waiting for unfinished jobs....
> >
> > I can fix it up by doing:
> >
> > diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
> > index 5d3dd647dbf6..9bd7a785eb1b 100644
> > --- a/fs/nfs/nfs4proc.c
> > +++ b/fs/nfs/nfs4proc.c
> > @@ -212,10 +212,10 @@ const u32 nfs4_fattr_bitmap[3] = {
> >         | FATTR4_WORD1_TIME_METADATA
> >         | FATTR4_WORD1_TIME_MODIFY
> >         | FATTR4_WORD1_MOUNTED_ON_FILEID,
> > +       FATTR4_WORD2_OFFLINE
> >  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> > -       FATTR4_WORD2_SECURITY_LABEL
> > +       | FATTR4_WORD2_SECURITY_LABEL
> >  #endif
> > -       | FATTR4_WORD2_OFFLINE
> >  };
> >
> >  static const u32 nfs4_pnfs_open_bitmap[3] = {
> >
> > If that works for you I can squash it into your patch in my tree
>
> Sure. I was thinking the opposite, but this works too.
>
> i.e. my suggestion was just
>
> #ifdef CONFIG_NFS_V4_SECURITY_LABEL
>         FATTR4_WORD2_SECURITY_LABEL |
> #endif
>         FATTR4_WORD2_OFFLINE
>
> However there is no functional difference between the two, so let's go
> with yours.

Thanks for taking a quick look!

Anna

>
> >
> > Anna
> >
> > >
> > > Anna
> > >
> > > >
> > > >  static const u32 nfs4_pnfs_open_bitmap[3] = {
> > > > @@ -245,6 +246,7 @@ static const u32 nfs4_pnfs_open_bitmap[3] = {
> > > >  #ifdef CONFIG_NFS_V4_SECURITY_LABEL
> > > >         | FATTR4_WORD2_SECURITY_LABEL
> > > >  #endif
> > > > +       | FATTR4_WORD2_OFFLINE
> > > >  };
> > > >
> > > >  static const u32 nfs4_open_noattr_bitmap[3] = {
> > > > @@ -325,6 +327,7 @@ static void nfs4_bitmap_copy_adjust(__u32
> > > > *dst, const __u32 *src,
> > > >         if (!(cache_validity & NFS_INO_INVALID_WINATTR)) {
> > > >                 dst[0] &= ~(FATTR4_WORD0_ARCHIVE |
> > > > FATTR4_WORD0_HIDDEN);
> > > >                 dst[1] &= ~(FATTR4_WORD1_SYSTEM |
> > > > FATTR4_WORD1_TIME_BACKUP);
> > > > +               dst[2] &= ~FATTR4_WORD2_OFFLINE;
> > > >         }
> > > >  }
> > > >
> > > > @@ -3927,6 +3930,8 @@ static int _nfs4_server_capabilities(struct
> > > > nfs_server *server, struct nfs_fh *f
> > > >                 memcpy(server->attr_bitmask_nl, res.attr_bitmask,
> > > >                                 sizeof(server->attr_bitmask));
> > > >                 server->attr_bitmask_nl[2] &=
> > > > ~FATTR4_WORD2_SECURITY_LABEL;
> > > > +               if (!(res.attr_bitmask[2] &
> > > > FATTR4_WORD2_OFFLINE))
> > > > +                       server->fattr_valid &=
> > > > ~NFS_ATTR_FATTR_OFFLINE;
> > > >
> > > >                 memcpy(server->cache_consistency_bitmask,
> > > > res.attr_bitmask, sizeof(server->cache_consistency_bitmask));
> > > >                 server->cache_consistency_bitmask[0] &=
> > > > FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE;
> > > > @@ -5486,6 +5491,7 @@ static void nfs4_bitmask_set(__u32
> > > > bitmask[NFS4_BITMASK_SZ], const __u32 *src,
> > > >         if (cache_validity & NFS_INO_INVALID_WINATTR) {
> > > >                 bitmask[0] |= FATTR4_WORD0_ARCHIVE |
> > > > FATTR4_WORD0_HIDDEN;
> > > >                 bitmask[1] |= FATTR4_WORD1_SYSTEM |
> > > > FATTR4_WORD1_TIME_BACKUP;
> > > > +               bitmask[2] |= FATTR4_WORD2_OFFLINE;
> > > >         }
> > > >
> > > >         if (cache_validity & NFS_INO_INVALID_SIZE)
> > > > diff --git a/fs/nfs/nfs4trace.h b/fs/nfs/nfs4trace.h
> > > > index 5e72639b469e..02c78d66c36d 100644
> > > > --- a/fs/nfs/nfs4trace.h
> > > > +++ b/fs/nfs/nfs4trace.h
> > > > @@ -35,7 +35,8 @@
> > > >                 { NFS_ATTR_FATTR_HIDDEN, "HIDDEN" }, \
> > > >                 { NFS_ATTR_FATTR_SYSTEM, "SYSTEM" }, \
> > > >                 { NFS_ATTR_FATTR_ARCHIVE, "ARCHIVE" }, \
> > > > -               { NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" })
> > > > +               { NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" }, \
> > > > +               { NFS_ATTR_FATTR_OFFLINE, "OFFLINE" })
> > > >
> > > >  DECLARE_EVENT_CLASS(nfs4_clientid_event,
> > > >                 TP_PROTO(
> > > > diff --git a/fs/nfs/nfs4xdr.c b/fs/nfs/nfs4xdr.c
> > > > index 280df43c5bf2..a45872f860ec 100644
> > > > --- a/fs/nfs/nfs4xdr.c
> > > > +++ b/fs/nfs/nfs4xdr.c
> > > > @@ -1623,7 +1623,7 @@ static void encode_readdir(struct
> > > > xdr_stream *xdr, const struct nfs4_readdir_arg
> > > >                             FATTR4_WORD1_TIME_CREATE |
> > > >                             FATTR4_WORD1_TIME_METADATA |
> > > >                             FATTR4_WORD1_TIME_MODIFY;
> > > > -               attrs[2] |= FATTR4_WORD2_SECURITY_LABEL;
> > > > +               attrs[2] |= FATTR4_WORD2_SECURITY_LABEL |
> > > > FATTR4_WORD2_OFFLINE;
> > > >                 dircount >>= 1;
> > > >         }
> > > >         /* Use mounted_on_fileid only if the server supports it
> > > > */
> > > > @@ -4354,6 +4354,29 @@ static int decode_attr_xattrsupport(struct
> > > > xdr_stream *xdr, uint32_t *bitmap,
> > > >         return 0;
> > > >  }
> > > >
> > > > +static int decode_attr_offline(struct xdr_stream *xdr, uint32_t
> > > > *bitmap,
> > > > +                              uint32_t *res, uint64_t *flags)
> > > > +{
> > > > +       int status = 0;
> > > > +       __be32 *p;
> > > > +
> > > > +       if (unlikely(bitmap[2] & (FATTR4_WORD2_OFFLINE - 1U)))
> > > > +               return -EIO;
> > > > +       if (likely(bitmap[2] & FATTR4_WORD2_OFFLINE)) {
> > > > +               p = xdr_inline_decode(xdr, 4);
> > > > +               if (unlikely(!p))
> > > > +                       return -EIO;
> > > > +               if (be32_to_cpup(p))
> > > > +                       *res |= NFS_HSA_OFFLINE;
> > > > +               else
> > > > +                       *res &= ~NFS_HSA_OFFLINE;
> > > > +               bitmap[2] &= ~FATTR4_WORD2_OFFLINE;
> > > > +               *flags |= NFS_ATTR_FATTR_OFFLINE;
> > > > +       }
> > > > +       dprintk("%s: system file: =%s\n", __func__, (*res &
> > > > NFS_HSA_OFFLINE) == 0 ? "false" : "true");
> > > > +       return status;
> > > > +}
> > > > +
> > > >  static int verify_attr_len(struct xdr_stream *xdr, unsigned int
> > > > savep, uint32_t attrlen)
> > > >  {
> > > >         unsigned int attrwords = XDR_QUADLEN(attrlen);
> > > > @@ -4842,6 +4865,12 @@ static int decode_getfattr_attrs(struct
> > > > xdr_stream *xdr, uint32_t *bitmap,
> > > >                 fattr->valid |= status;
> > > >         }
> > > >
> > > > +       status = decode_attr_offline(xdr, bitmap, &fattr-
> > > > >hsa_flags,
> > > > +                                    &fattr->valid);
> > > > +       if (status < 0)
> > > > +               goto xdr_error;
> > > > +
> > > > +       status = 0;
> > > >  xdr_error:
> > > >         dprintk("%s: xdr returned %d\n", __func__, -status);
> > > >         return status;
> > > > diff --git a/include/linux/nfs4.h b/include/linux/nfs4.h
> > > > index 5662d8be04eb..817b349c24ca 100644
> > > > --- a/include/linux/nfs4.h
> > > > +++ b/include/linux/nfs4.h
> > > > @@ -460,6 +460,7 @@ enum lock_type4 {
> > > >  #define FATTR4_WORD2_SECURITY_LABEL     (1UL << 16)
> > > >  #define FATTR4_WORD2_MODE_UMASK                (1UL << 17)
> > > >  #define FATTR4_WORD2_XATTR_SUPPORT     (1UL << 18)
> > > > +#define FATTR4_WORD2_OFFLINE           (1UL << 19)
> > > >
> > > >  /* MDS threshold bitmap bits */
> > > >  #define THRESHOLD_RD                    (1UL << 0)
> > > > diff --git a/include/linux/nfs_fs.h b/include/linux/nfs_fs.h
> > > > index 1839fed16b18..61f852ebb0a4 100644
> > > > --- a/include/linux/nfs_fs.h
> > > > +++ b/include/linux/nfs_fs.h
> > > > @@ -148,6 +148,7 @@ struct nfs_inode {
> > > >         unsigned char           archive : 1;
> > > >         unsigned char           hidden : 1;
> > > >         unsigned char           system : 1;
> > > > +       unsigned char           offline : 1;
> > > >
> > > >         /*
> > > >          * read_cache_jiffies is when we started read-caching
> > > > this inode.
> > > > diff --git a/include/linux/nfs_xdr.h b/include/linux/nfs_xdr.h
> > > > index dedc7e0bf2b9..f5588bc22cb1 100644
> > > > --- a/include/linux/nfs_xdr.h
> > > > +++ b/include/linux/nfs_xdr.h
> > > > @@ -21,6 +21,7 @@
> > > >  #define NFS_HSA_HIDDEN         BIT(0)
> > > >  #define NFS_HSA_SYSTEM         BIT(1)
> > > >  #define NFS_HSA_ARCHIVE                BIT(2)
> > > > +#define NFS_HSA_OFFLINE                BIT(3)
> > > >
> > > >  struct nfs4_string {
> > > >         unsigned int len;
> > > > @@ -119,6 +120,7 @@ struct nfs_fattr {
> > > >  #define NFS_ATTR_FATTR_SYSTEM           BIT_ULL(28)
> > > >  #define NFS_ATTR_FATTR_ARCHIVE          BIT_ULL(29)
> > > >  #define NFS_ATTR_FATTR_TIME_BACKUP      BIT_ULL(30)
> > > > +#define NFS_ATTR_FATTR_OFFLINE          BIT_ULL(31)
> > > >
> > > >  #define NFS_ATTR_FATTR (NFS_ATTR_FATTR_TYPE \
> > > >                 | NFS_ATTR_FATTR_MODE \
> > > > @@ -144,7 +146,8 @@ struct nfs_fattr {
> > > >                 | NFS_ATTR_FATTR_SYSTEM \
> > > >                 | NFS_ATTR_FATTR_ARCHIVE \
> > > >                 | NFS_ATTR_FATTR_TIME_BACKUP \
> > > > -               | NFS_ATTR_FATTR_V4_SECURITY_LABEL)
> > > > +               | NFS_ATTR_FATTR_V4_SECURITY_LABEL \
> > > > +               | NFS_ATTR_FATTR_OFFLINE)
> > > >
> > > >  /*
> > > >   * Maximal number of supported layout drivers.
> > > > --
> > > > 2.33.1
> > > >
>
> --
> Trond Myklebust
> Linux NFS client maintainer, Hammerspace
> trond.myklebust@hammerspace.com
>
>
diff mbox series

Patch

diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c
index 4673b091ea31..33f4410190b6 100644
--- a/fs/nfs/inode.c
+++ b/fs/nfs/inode.c
@@ -528,6 +528,7 @@  nfs_fhget(struct super_block *sb, struct nfs_fh *fh, struct nfs_fattr *fattr)
 		nfsi->archive = 0;
 		nfsi->hidden = 0;
 		nfsi->system = 0;
+		nfsi->offline = 0;
 		inode_set_iversion_raw(inode, 0);
 		inode->i_size = 0;
 		clear_nlink(inode);
@@ -606,6 +607,10 @@  nfs_fhget(struct super_block *sb, struct nfs_fh *fh, struct nfs_fattr *fattr)
 		} else if (fattr_supported & NFS_ATTR_FATTR_SPACE_USED &&
 			   fattr->size != 0)
 			nfs_set_cache_invalid(inode, NFS_INO_INVALID_BLOCKS);
+		if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
+			nfsi->offline = (fattr->hsa_flags & NFS_HSA_OFFLINE) ? 1 : 0;
+		else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
+			nfs_set_cache_invalid(inode, NFS_INO_INVALID_WINATTR);
 
 		nfs_setsecurity(inode, fattr);
 
@@ -2274,6 +2279,12 @@  static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr)
 		nfsi->cache_validity |=
 			save_cache_validity & NFS_INO_INVALID_BLOCKS;
 
+	if (fattr->valid & NFS_ATTR_FATTR_OFFLINE)
+		nfsi->offline = (fattr->hsa_flags & NFS_HSA_OFFLINE) ? 1 : 0;
+	else if (fattr_supported & NFS_ATTR_FATTR_OFFLINE)
+		nfsi->cache_validity |=
+			save_cache_validity & NFS_INO_INVALID_WINATTR;
+
 	/* Update attrtimeo value if we're out of the unstable period */
 	if (attr_changed) {
 		nfs_inc_stats(inode, NFSIOS_ATTRINVALIDATE);
diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
index 4e6cc54016ba..713a71fb3020 100644
--- a/fs/nfs/nfs4proc.c
+++ b/fs/nfs/nfs4proc.c
@@ -219,6 +219,7 @@  const u32 nfs4_fattr_bitmap[3] = {
 #ifdef CONFIG_NFS_V4_SECURITY_LABEL
 	FATTR4_WORD2_SECURITY_LABEL
 #endif
+	| FATTR4_WORD2_OFFLINE
 };
 
 static const u32 nfs4_pnfs_open_bitmap[3] = {
@@ -245,6 +246,7 @@  static const u32 nfs4_pnfs_open_bitmap[3] = {
 #ifdef CONFIG_NFS_V4_SECURITY_LABEL
 	| FATTR4_WORD2_SECURITY_LABEL
 #endif
+	| FATTR4_WORD2_OFFLINE
 };
 
 static const u32 nfs4_open_noattr_bitmap[3] = {
@@ -325,6 +327,7 @@  static void nfs4_bitmap_copy_adjust(__u32 *dst, const __u32 *src,
 	if (!(cache_validity & NFS_INO_INVALID_WINATTR)) {
 		dst[0] &= ~(FATTR4_WORD0_ARCHIVE | FATTR4_WORD0_HIDDEN);
 		dst[1] &= ~(FATTR4_WORD1_SYSTEM | FATTR4_WORD1_TIME_BACKUP);
+		dst[2] &= ~FATTR4_WORD2_OFFLINE;
 	}
 }
 
@@ -3927,6 +3930,8 @@  static int _nfs4_server_capabilities(struct nfs_server *server, struct nfs_fh *f
 		memcpy(server->attr_bitmask_nl, res.attr_bitmask,
 				sizeof(server->attr_bitmask));
 		server->attr_bitmask_nl[2] &= ~FATTR4_WORD2_SECURITY_LABEL;
+		if (!(res.attr_bitmask[2] & FATTR4_WORD2_OFFLINE))
+			server->fattr_valid &= ~NFS_ATTR_FATTR_OFFLINE;
 
 		memcpy(server->cache_consistency_bitmask, res.attr_bitmask, sizeof(server->cache_consistency_bitmask));
 		server->cache_consistency_bitmask[0] &= FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE;
@@ -5486,6 +5491,7 @@  static void nfs4_bitmask_set(__u32 bitmask[NFS4_BITMASK_SZ], const __u32 *src,
 	if (cache_validity & NFS_INO_INVALID_WINATTR) {
 		bitmask[0] |= FATTR4_WORD0_ARCHIVE | FATTR4_WORD0_HIDDEN;
 		bitmask[1] |= FATTR4_WORD1_SYSTEM | FATTR4_WORD1_TIME_BACKUP;
+		bitmask[2] |= FATTR4_WORD2_OFFLINE;
 	}
 
 	if (cache_validity & NFS_INO_INVALID_SIZE)
diff --git a/fs/nfs/nfs4trace.h b/fs/nfs/nfs4trace.h
index 5e72639b469e..02c78d66c36d 100644
--- a/fs/nfs/nfs4trace.h
+++ b/fs/nfs/nfs4trace.h
@@ -35,7 +35,8 @@ 
 		{ NFS_ATTR_FATTR_HIDDEN, "HIDDEN" }, \
 		{ NFS_ATTR_FATTR_SYSTEM, "SYSTEM" }, \
 		{ NFS_ATTR_FATTR_ARCHIVE, "ARCHIVE" }, \
-		{ NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" })
+		{ NFS_ATTR_FATTR_TIME_BACKUP, "TIME_BACKUP" }, \
+		{ NFS_ATTR_FATTR_OFFLINE, "OFFLINE" })
 
 DECLARE_EVENT_CLASS(nfs4_clientid_event,
 		TP_PROTO(
diff --git a/fs/nfs/nfs4xdr.c b/fs/nfs/nfs4xdr.c
index 280df43c5bf2..a45872f860ec 100644
--- a/fs/nfs/nfs4xdr.c
+++ b/fs/nfs/nfs4xdr.c
@@ -1623,7 +1623,7 @@  static void encode_readdir(struct xdr_stream *xdr, const struct nfs4_readdir_arg
 			    FATTR4_WORD1_TIME_CREATE |
 			    FATTR4_WORD1_TIME_METADATA |
 			    FATTR4_WORD1_TIME_MODIFY;
-		attrs[2] |= FATTR4_WORD2_SECURITY_LABEL;
+		attrs[2] |= FATTR4_WORD2_SECURITY_LABEL | FATTR4_WORD2_OFFLINE;
 		dircount >>= 1;
 	}
 	/* Use mounted_on_fileid only if the server supports it */
@@ -4354,6 +4354,29 @@  static int decode_attr_xattrsupport(struct xdr_stream *xdr, uint32_t *bitmap,
 	return 0;
 }
 
+static int decode_attr_offline(struct xdr_stream *xdr, uint32_t *bitmap,
+			       uint32_t *res, uint64_t *flags)
+{
+	int status = 0;
+	__be32 *p;
+
+	if (unlikely(bitmap[2] & (FATTR4_WORD2_OFFLINE - 1U)))
+		return -EIO;
+	if (likely(bitmap[2] & FATTR4_WORD2_OFFLINE)) {
+		p = xdr_inline_decode(xdr, 4);
+		if (unlikely(!p))
+			return -EIO;
+		if (be32_to_cpup(p))
+			*res |= NFS_HSA_OFFLINE;
+		else
+			*res &= ~NFS_HSA_OFFLINE;
+		bitmap[2] &= ~FATTR4_WORD2_OFFLINE;
+		*flags |= NFS_ATTR_FATTR_OFFLINE;
+	}
+	dprintk("%s: system file: =%s\n", __func__, (*res & NFS_HSA_OFFLINE) == 0 ? "false" : "true");
+	return status;
+}
+
 static int verify_attr_len(struct xdr_stream *xdr, unsigned int savep, uint32_t attrlen)
 {
 	unsigned int attrwords = XDR_QUADLEN(attrlen);
@@ -4842,6 +4865,12 @@  static int decode_getfattr_attrs(struct xdr_stream *xdr, uint32_t *bitmap,
 		fattr->valid |= status;
 	}
 
+	status = decode_attr_offline(xdr, bitmap, &fattr->hsa_flags,
+				     &fattr->valid);
+	if (status < 0)
+		goto xdr_error;
+
+	status = 0;
 xdr_error:
 	dprintk("%s: xdr returned %d\n", __func__, -status);
 	return status;
diff --git a/include/linux/nfs4.h b/include/linux/nfs4.h
index 5662d8be04eb..817b349c24ca 100644
--- a/include/linux/nfs4.h
+++ b/include/linux/nfs4.h
@@ -460,6 +460,7 @@  enum lock_type4 {
 #define FATTR4_WORD2_SECURITY_LABEL     (1UL << 16)
 #define FATTR4_WORD2_MODE_UMASK		(1UL << 17)
 #define FATTR4_WORD2_XATTR_SUPPORT	(1UL << 18)
+#define FATTR4_WORD2_OFFLINE		(1UL << 19)
 
 /* MDS threshold bitmap bits */
 #define THRESHOLD_RD                    (1UL << 0)
diff --git a/include/linux/nfs_fs.h b/include/linux/nfs_fs.h
index 1839fed16b18..61f852ebb0a4 100644
--- a/include/linux/nfs_fs.h
+++ b/include/linux/nfs_fs.h
@@ -148,6 +148,7 @@  struct nfs_inode {
 	unsigned char		archive : 1;
 	unsigned char		hidden : 1;
 	unsigned char		system : 1;
+	unsigned char		offline : 1;
 
 	/*
 	 * read_cache_jiffies is when we started read-caching this inode.
diff --git a/include/linux/nfs_xdr.h b/include/linux/nfs_xdr.h
index dedc7e0bf2b9..f5588bc22cb1 100644
--- a/include/linux/nfs_xdr.h
+++ b/include/linux/nfs_xdr.h
@@ -21,6 +21,7 @@ 
 #define NFS_HSA_HIDDEN		BIT(0)
 #define NFS_HSA_SYSTEM		BIT(1)
 #define NFS_HSA_ARCHIVE		BIT(2)
+#define NFS_HSA_OFFLINE		BIT(3)
 
 struct nfs4_string {
 	unsigned int len;
@@ -119,6 +120,7 @@  struct nfs_fattr {
 #define NFS_ATTR_FATTR_SYSTEM           BIT_ULL(28)
 #define NFS_ATTR_FATTR_ARCHIVE          BIT_ULL(29)
 #define NFS_ATTR_FATTR_TIME_BACKUP      BIT_ULL(30)
+#define NFS_ATTR_FATTR_OFFLINE          BIT_ULL(31)
 
 #define NFS_ATTR_FATTR (NFS_ATTR_FATTR_TYPE \
 		| NFS_ATTR_FATTR_MODE \
@@ -144,7 +146,8 @@  struct nfs_fattr {
 		| NFS_ATTR_FATTR_SYSTEM \
 		| NFS_ATTR_FATTR_ARCHIVE \
 		| NFS_ATTR_FATTR_TIME_BACKUP \
-		| NFS_ATTR_FATTR_V4_SECURITY_LABEL)
+		| NFS_ATTR_FATTR_V4_SECURITY_LABEL \
+		| NFS_ATTR_FATTR_OFFLINE)
 
 /*
  * Maximal number of supported layout drivers.