Re: [PATCH 3/4] libfuse: add statx support to the lower level library

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Fri, Jul 18, 2025 at 1:39 AM Darrick J. Wong <djwong@xxxxxxxxxx> wrote:
>
> From: Darrick J. Wong <djwong@xxxxxxxxxx>
>
> Add statx support to the lower level fuse library.

This looked familiar.
Merged 3 days ago:
https://github.com/libfuse/libfuse/pull/1026

>
> Signed-off-by: "Darrick J. Wong" <djwong@xxxxxxxxxx>
> ---
>  include/fuse_lowlevel.h |   37 ++++++++++++++++++
>  lib/fuse_lowlevel.c     |   97 +++++++++++++++++++++++++++++++++++++++++++++++
>  lib/fuse_versionscript  |    2 +
>  3 files changed, 136 insertions(+)
>
>
> diff --git a/include/fuse_lowlevel.h b/include/fuse_lowlevel.h
> index 77685e433e4f7d..f4d62cee22870a 100644
> --- a/include/fuse_lowlevel.h
> +++ b/include/fuse_lowlevel.h
> @@ -1416,6 +1416,26 @@ struct fuse_lowlevel_ops {
>          * @param ino the inode number
>          */
>         void (*syncfs) (fuse_req_t req, fuse_ino_t ino);
> +
> +       /**
> +        * Fetch extended stat information about a file
> +        *
> +        * If this request is answered with an error code of ENOSYS, this is
> +        * treated as a permanent failure, i.e. all future statx() requests
> +        * will fail with the same error code without being sent to the
> +        * filesystem process.
> +        *
> +        * Valid replies:
> +        *   fuse_reply_statx
> +        *   fuse_reply_err
> +        *
> +        * @param req request handle
> +        * @param statx_flags AT_STATX_* flags
> +        * @param statx_mask desired STATX_* attribute mask
> +        * @param fi file information
> +        */
> +       void (*statx) (fuse_req_t req, fuse_ino_t ino, uint32_t statx_flags,
> +                      uint32_t statx_mask, struct fuse_file_info *fi);
>  #endif /* FUSE_USE_VERSION >= 318 */
>  };
>
> @@ -1897,6 +1917,23 @@ int fuse_reply_iomap_begin(fuse_req_t req, const struct fuse_iomap *read_iomap,
>   * @return zero for success, -errno for failure to send reply
>   */
>  int fuse_reply_iomap_config(fuse_req_t req, const struct fuse_iomap_config *cfg);
> +
> +struct statx;
> +
> +/**
> + * Reply with statx attributes
> + *
> + * Possible requests:
> + *   statx
> + *
> + * @param req request handle
> + * @param statx the attributes
> + * @param size the size of the statx structure
> + * @param attr_timeout validity timeout (in seconds) for the attributes
> + * @return zero for success, -errno for failure to send reply
> + */
> +int fuse_reply_statx(fuse_req_t req, const struct statx *statx, size_t size,
> +                    double attr_timeout);
>  #endif /* FUSE_USE_VERSION >= 318 */
>
>  /* ----------------------------------------------------------- *
> diff --git a/lib/fuse_lowlevel.c b/lib/fuse_lowlevel.c
> index ec30ebc4cdd074..8eeb6a8547da91 100644
> --- a/lib/fuse_lowlevel.c
> +++ b/lib/fuse_lowlevel.c
> @@ -144,6 +144,43 @@ static void convert_attr(const struct fuse_setattr_in *attr, struct stat *stbuf)
>         ST_CTIM_NSEC_SET(stbuf, attr->ctimensec);
>  }
>
> +#ifdef STATX_BASIC_STATS
> +static int convert_statx(struct fuse_statx *stbuf, const struct statx *stx,
> +                        size_t size)
> +{
> +       if (sizeof(struct statx) != size)
> +               return EOPNOTSUPP;
> +
> +       stbuf->mask = stx->stx_mask & (STATX_BASIC_STATS | STATX_BTIME);
> +       stbuf->blksize          = stx->stx_blksize;
> +       stbuf->attributes       = stx->stx_attributes;
> +       stbuf->nlink            = stx->stx_nlink;
> +       stbuf->uid              = stx->stx_uid;
> +       stbuf->gid              = stx->stx_gid;
> +       stbuf->mode             = stx->stx_mode;
> +       stbuf->ino              = stx->stx_ino;
> +       stbuf->size             = stx->stx_size;
> +       stbuf->blocks           = stx->stx_blocks;
> +       stbuf->attributes_mask  = stx->stx_attributes_mask;
> +       stbuf->rdev_major       = stx->stx_rdev_major;
> +       stbuf->rdev_minor       = stx->stx_rdev_minor;
> +       stbuf->dev_major        = stx->stx_dev_major;
> +       stbuf->dev_minor        = stx->stx_dev_minor;
> +
> +       stbuf->atime.tv_sec     = stx->stx_atime.tv_sec;
> +       stbuf->btime.tv_sec     = stx->stx_btime.tv_sec;
> +       stbuf->ctime.tv_sec     = stx->stx_ctime.tv_sec;
> +       stbuf->mtime.tv_sec     = stx->stx_mtime.tv_sec;
> +
> +       stbuf->atime.tv_nsec    = stx->stx_atime.tv_nsec;
> +       stbuf->btime.tv_nsec    = stx->stx_btime.tv_nsec;
> +       stbuf->ctime.tv_nsec    = stx->stx_ctime.tv_nsec;
> +       stbuf->mtime.tv_nsec    = stx->stx_mtime.tv_nsec;
> +
> +       return 0;
> +}
> +#endif
> +

Why is this conversion not needed in the merged version?
What am I missing?

Thanks,
Amir.

>  static size_t iov_length(const struct iovec *iov, size_t count)
>  {
>         size_t seg;
> @@ -2653,6 +2690,64 @@ static void do_syncfs(fuse_req_t req, const fuse_ino_t nodeid, const void *inarg
>         _do_syncfs(req, nodeid, inarg, NULL);
>  }
>
> +#ifdef STATX_BASIC_STATS
> +int fuse_reply_statx(fuse_req_t req, const struct statx *statx, size_t size,
> +                    double attr_timeout)
> +{
> +       struct fuse_statx_out arg = {
> +               .attr_valid = calc_timeout_sec(attr_timeout),
> +               .attr_valid_nsec = calc_timeout_nsec(attr_timeout),
> +       };
> +
> +       int err = convert_statx(&arg.stat, statx, size);
> +       if (err) {
> +               fuse_reply_err(req, err);
> +               return err;
> +       }
> +
> +       return send_reply_ok(req, &arg, sizeof(arg));
> +}
> +
> +static void _do_statx(fuse_req_t req, const fuse_ino_t nodeid,
> +                     const void *op_in, const void *in_payload)
> +{
> +       (void)in_payload;
> +       const struct fuse_statx_in *arg = op_in;
> +       struct fuse_file_info *fip = NULL;
> +       struct fuse_file_info fi;
> +
> +       if (arg->getattr_flags & FUSE_GETATTR_FH) {
> +               memset(&fi, 0, sizeof(fi));
> +               fi.fh = arg->fh;
> +               fip = &fi;
> +       }
> +
> +       if (req->se->op.statx)
> +               req->se->op.statx(req, nodeid, arg->sx_flags, arg->sx_mask,
> +                                 fip);
> +       else
> +               fuse_reply_err(req, ENOSYS);
> +}
> +#else
> +int fuse_reply_statx(fuse_req_t req, const struct statx *statx,
> +                    double attr_timeout)
> +{
> +       fuse_reply_err(req, ENOSYS);
> +       return -ENOSYS;
> +}
> +
> +static void _do_statx(fuse_req_t req, const fuse_ino_t nodeid,
> +                     const void *op_in, const void *in_payload)
> +{
> +       fuse_reply_err(req, ENOSYS);
> +}
> +#endif /* STATX_BASIC_STATS */
> +
> +static void do_statx(fuse_req_t req, const fuse_ino_t nodeid, const void *inarg)
> +{
> +       _do_statx(req, nodeid, inarg, NULL);
> +}
> +
>  static bool want_flags_valid(uint64_t capable, uint64_t want)
>  {
>         uint64_t unknown_flags = want & (~capable);
> @@ -3627,6 +3722,7 @@ static struct {
>         [FUSE_COPY_FILE_RANGE] = { do_copy_file_range, "COPY_FILE_RANGE" },
>         [FUSE_LSEEK]       = { do_lseek,       "LSEEK"       },
>         [FUSE_SYNCFS]      = { do_syncfs,       "SYNCFS"     },
> +       [FUSE_STATX]       = { do_statx,       "STATX"       },
>         [FUSE_IOMAP_CONFIG]= { do_iomap_config, "IOMAP_CONFIG" },
>         [FUSE_IOMAP_BEGIN] = { do_iomap_begin,  "IOMAP_BEGIN" },
>         [FUSE_IOMAP_END]   = { do_iomap_end,    "IOMAP_END" },
> @@ -3686,6 +3782,7 @@ static struct {
>         [FUSE_COPY_FILE_RANGE]  = { _do_copy_file_range, "COPY_FILE_RANGE" },
>         [FUSE_LSEEK]            = { _do_lseek,          "LSEEK" },
>         [FUSE_SYNCFS]           = { _do_syncfs,         "SYNCFS" },
> +       [FUSE_STATX]            = { _do_statx,          "STATX" },
>         [FUSE_IOMAP_CONFIG]     = { _do_iomap_config,   "IOMAP_CONFIG" },
>         [FUSE_IOMAP_BEGIN]      = { _do_iomap_begin,    "IOMAP_BEGIN" },
>         [FUSE_IOMAP_END]        = { _do_iomap_end,      "IOMAP_END" },
> diff --git a/lib/fuse_versionscript b/lib/fuse_versionscript
> index dc9fa2428b5325..a67b1802770335 100644
> --- a/lib/fuse_versionscript
> +++ b/lib/fuse_versionscript
> @@ -223,6 +223,8 @@ FUSE_3.18 {
>                 fuse_reply_iomap_config;
>                 fuse_lowlevel_notify_iomap_upsert;
>                 fuse_lowlevel_notify_iomap_inval;
> +
> +               fuse_reply_statx;
>  } FUSE_3.17;
>
>  # Local Variables:
>
>





[Index of Archives]     [Linux Ext4 Filesystem]     [Union Filesystem]     [Filesystem Testing]     [Ceph Users]     [Ecryptfs]     [NTFS 3]     [AutoFS]     [Kernel Newbies]     [Share Photos]     [Security]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux Cachefs]     [Reiser Filesystem]     [Linux RAID]     [NTFS 3]     [Samba]     [Device Mapper]     [CEPH Development]

  Powered by Linux