All of lore.kernel.org
 help / color / mirror / Atom feed
From: Olga Kornievskaia <aglo@umich.edu>
To: Olga Kornievskaia <kolga@netapp.com>
Cc: "J. Bruce Fields" <bfields@redhat.com>,
	linux-nfs <linux-nfs@vger.kernel.org>
Subject: Re: [PATCH v5 05/10] NFSD first draft of async copy
Date: Sat, 14 Oct 2017 09:45:02 -0400	[thread overview]
Message-ID: <CAN-5tyHny-MDhJoo=xqEzsZAZf301p8Z0AAYnE6+yYr6R4xjJQ@mail.gmail.com> (raw)
In-Reply-To: <20171013205412.65532-6-kolga@netapp.com>

On Fri, Oct 13, 2017 at 4:54 PM, Olga Kornievskaia <kolga@netapp.com> wrote:
> Upon receiving a request for async copy, create a new kthread.
> If we get asynchronous request, make sure to copy the needed
> arguments/state from the stack before starting the copy. Then
> start the thread and reply back to the client indicating copy
> is asynchronous.
>
> nfsd_copy_file_range() will copy in a loop over the total
> number of bytes is needed to copy. In case a failure happens
> in the middle, we can return an error as well as how much we
> copied so far. Once done creating a workitem for the callback
> workqueue and send CB_OFFLOAD with the results.
>
> Signed-off-by: Olga Kornievskaia <kolga@netapp.com>
> ---
>  fs/nfsd/nfs4proc.c  | 177 ++++++++++++++++++++++++++++++++++++++++++++++------
>  fs/nfsd/nfs4state.c |   2 +
>  fs/nfsd/state.h     |   2 +
>  fs/nfsd/xdr4.h      |   9 +++
>  4 files changed, 171 insertions(+), 19 deletions(-)
>
> diff --git a/fs/nfsd/nfs4proc.c b/fs/nfsd/nfs4proc.c
> index ea698a4..2787c95 100644
> --- a/fs/nfsd/nfs4proc.c
> +++ b/fs/nfsd/nfs4proc.c
> @@ -35,6 +35,7 @@
>  #include <linux/file.h>
>  #include <linux/falloc.h>
>  #include <linux/slab.h>
> +#include <linux/kthread.h>
>
>  #include "idmap.h"
>  #include "cache.h"
> @@ -1085,39 +1086,177 @@ static int fill_in_write_vector(struct kvec *vec, struct nfsd4_write *write)
>  out:
>         return status;
>  }
> +static void nfsd4_cb_offload_release(struct nfsd4_callback *cb)
> +{
> +       struct nfsd4_copy *copy = container_of(cb, struct nfsd4_copy, cp_cb);
> +
> +       atomic_dec(&copy->cp_clp->cl_refcount);
> +       kfree(copy);
> +}
> +
> +static int nfsd4_cb_offload_done(struct nfsd4_callback *cb,
> +                                struct rpc_task *task)
> +{
> +       return 1;
> +}
> +
> +static const struct nfsd4_callback_ops nfsd4_cb_offload_ops = {
> +       .release = nfsd4_cb_offload_release,
> +       .done = nfsd4_cb_offload_done
> +};
> +
> +static int nfsd4_init_copy_res(struct nfsd4_copy *copy, bool sync)
> +{
> +       memcpy(&copy->cp_res.cb_stateid, &copy->cp_dst_stateid,
> +               sizeof(copy->cp_dst_stateid));
> +       copy->cp_res.wr_stable_how = NFS_UNSTABLE;
> +       copy->cp_consecutive = 1;
> +       copy->cp_synchronous = sync;
> +       gen_boot_verifier(&copy->cp_res.wr_verifier, copy->net);
> +
> +       return nfs_ok;
> +}
> +
> +static int _nfsd_copy_file_range(struct nfsd4_copy *copy)
> +{
> +       ssize_t bytes_copied = 0;
> +       size_t bytes_total = copy->cp_count;
> +       size_t bytes_to_copy;
> +       u64 src_pos = copy->cp_src_pos;
> +       u64 dst_pos = copy->cp_dst_pos;
> +
> +       do {
> +               if (copy->cp_synchronous)
> +                       bytes_to_copy = min_t(u64, bytes_total, MAX_RW_COUNT);

urgh that should have been 4MB value from the vfs.c file not the
do_splice max. I should also remove the nds_copy_file_range() from
vfs.c as it's no longer called. To implement the offload_status, if
you don't object, I'd rather go back for both sync and async to do 4MB
calls into VFS to get some kind of copy progress to report.

> +               else
> +                       bytes_to_copy = bytes_total;
> +               bytes_copied = vfs_copy_file_range(copy->fh_src, src_pos,
> +                               copy->fh_dst, dst_pos, bytes_to_copy, 0);
> +               if (bytes_copied <= 0)
> +                       break;
> +               bytes_total -= bytes_copied;
> +               copy->cp_res.wr_bytes_written += bytes_copied;
> +               src_pos += bytes_copied;
> +               dst_pos += bytes_copied;
> +       } while (bytes_total > 0 && !copy->cp_synchronous);
> +       return bytes_copied;
> +}
> +
> +static int nfsd4_do_copy(struct nfsd4_copy *copy, bool sync)
> +{
> +       __be32 status;
> +       ssize_t bytes;
> +
> +       bytes = _nfsd_copy_file_range(copy);
> +       if (bytes < 0)
> +               status = nfserrno(bytes);
> +       else
> +               status = nfsd4_init_copy_res(copy, sync);
> +
> +       fput(copy->fh_src);
> +       fput(copy->fh_dst);
> +       return status;
> +}
> +
> +static void dup_copy_fields(struct nfsd4_copy *src, struct nfsd4_copy *dst)
> +{
> +       memcpy(&dst->cp_src_stateid, &src->cp_src_stateid, sizeof(stateid_t));
> +       memcpy(&dst->cp_dst_stateid, &src->cp_dst_stateid, sizeof(stateid_t));
> +       dst->cp_src_pos = src->cp_src_pos;
> +       dst->cp_dst_pos = src->cp_dst_pos;
> +       dst->cp_count = src->cp_count;
> +       dst->cp_consecutive = src->cp_consecutive;
> +       dst->cp_synchronous = src->cp_synchronous;
> +       memcpy(&dst->cp_res, &src->cp_res, sizeof(src->cp_res));
> +       /* skipping nfsd4_callback */
> +       memcpy(&dst->fh, &src->fh, sizeof(src->fh));
> +       dst->net = src->net;
> +       dst->cp_clp = src->cp_clp;
> +       atomic_inc(&dst->cp_clp->cl_refcount);
> +       dst->fh_dst = get_file(src->fh_dst);
> +       dst->fh_src = get_file(src->fh_src);
> +}
> +
> +static void cleanup_async_copy(struct nfsd4_copy *copy)
> +{
> +       fput(copy->fh_dst);
> +       fput(copy->fh_src);
> +       spin_lock(&copy->cp_clp->async_lock);
> +       list_del(&copy->copies);
> +       spin_unlock(&copy->cp_clp->async_lock);
> +       atomic_dec(&copy->cp_clp->cl_refcount);
> +       kfree(copy);
> +}
> +
> +static int nfsd4_do_async_copy(void *data)
> +{
> +       struct nfsd4_copy *copy = (struct nfsd4_copy *)data;
> +       struct nfsd4_copy *cb_copy;
> +
> +       copy->nfserr = nfsd4_do_copy(copy, 0);
> +       cb_copy = kzalloc(sizeof(struct nfsd4_copy), GFP_KERNEL);
> +       if (!cb_copy)
> +               goto out;
> +       memcpy(&cb_copy->cp_res, &copy->cp_res, sizeof(copy->cp_res));
> +       cb_copy->cp_clp = copy->cp_clp;
> +       atomic_inc(&cb_copy->cp_clp->cl_refcount);
> +       cb_copy->nfserr = copy->nfserr;
> +       memcpy(&cb_copy->fh, &copy->fh, sizeof(copy->fh));
> +       nfsd4_init_cb(&cb_copy->cp_cb, cb_copy->cp_clp,
> +                       &nfsd4_cb_offload_ops, NFSPROC4_CLNT_CB_OFFLOAD);
> +       nfsd4_run_cb(&cb_copy->cp_cb);
> +out:
> +       cleanup_async_copy(copy);
> +       return 0;
> +}
>
>  static __be32
>  nfsd4_copy(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
>                 union nfsd4_op_u *u)
>  {
>         struct nfsd4_copy *copy = &u->copy;
> -       struct file *src, *dst;
>         __be32 status;
> -       ssize_t bytes;
> +       struct nfsd4_copy *async_copy = NULL;
>
> -       status = nfsd4_verify_copy(rqstp, cstate, &copy->cp_src_stateid, &src,
> -                                  &copy->cp_dst_stateid, &dst);
> +       status = nfsd4_verify_copy(rqstp, cstate, &copy->cp_src_stateid,
> +                                  &copy->fh_src, &copy->cp_dst_stateid,
> +                                  &copy->fh_dst);
>         if (status)
>                 goto out;
>
> -       bytes = nfsd_copy_file_range(src, copy->cp_src_pos,
> -                       dst, copy->cp_dst_pos, copy->cp_count);
> -
> -       if (bytes < 0)
> -               status = nfserrno(bytes);
> -       else {
> -               copy->cp_res.wr_bytes_written = bytes;
> -               copy->cp_res.wr_stable_how = NFS_UNSTABLE;
> -               copy->cp_consecutive = 1;
> -               copy->cp_synchronous = 1;
> -               gen_boot_verifier(&copy->cp_res.wr_verifier, SVC_NET(rqstp));
> -               status = nfs_ok;
> +       copy->cp_clp = cstate->clp;
> +       memcpy(&copy->fh, &cstate->current_fh.fh_handle,
> +               sizeof(struct knfsd_fh));
> +       copy->net = SVC_NET(rqstp);
> +       if (!copy->cp_synchronous) {
> +               status = nfsd4_init_copy_res(copy, 0);
> +               async_copy = kzalloc(sizeof(struct nfsd4_copy), GFP_KERNEL);
> +               if (!async_copy) {
> +                       status = nfserrno(-ENOMEM);
> +                       goto out;
> +               }
> +               dup_copy_fields(copy, async_copy);
> +               memcpy(&copy->cp_res.cb_stateid, &copy->cp_dst_stateid,
> +                       sizeof(copy->cp_dst_stateid));
> +               spin_lock(&async_copy->cp_clp->async_lock);
> +               list_add(&async_copy->copies,
> +                               &async_copy->cp_clp->async_copies);
> +               spin_unlock(&async_copy->cp_clp->async_lock);
> +               async_copy->copy_task = kthread_create(nfsd4_do_async_copy,
> +                               async_copy, "%s", "copy thread");
> +               if (IS_ERR(async_copy->copy_task)) {
> +                       status = PTR_ERR(async_copy->copy_task);
> +                       goto out_err_dec;
> +               }
> +               wake_up_process(async_copy->copy_task);
> +       } else {
> +               status = nfsd4_do_copy(copy, 1);
>         }
> -
> -       fput(src);
> -       fput(dst);
>  out:
>         return status;
> +out_err_dec:
> +       cleanup_async_copy(async_copy);
> +       goto out;
>  }
>
>  static __be32
> diff --git a/fs/nfsd/nfs4state.c b/fs/nfsd/nfs4state.c
> index 0c04f81..d7767a1 100644
> --- a/fs/nfsd/nfs4state.c
> +++ b/fs/nfsd/nfs4state.c
> @@ -1774,6 +1774,8 @@ static struct nfs4_client *alloc_client(struct xdr_netobj name)
>  #ifdef CONFIG_NFSD_PNFS
>         INIT_LIST_HEAD(&clp->cl_lo_states);
>  #endif
> +       INIT_LIST_HEAD(&clp->async_copies);
> +       spin_lock_init(&clp->async_lock);
>         spin_lock_init(&clp->cl_lock);
>         rpc_init_wait_queue(&clp->cl_cb_waitq, "Backchannel slot table");
>         return clp;
> diff --git a/fs/nfsd/state.h b/fs/nfsd/state.h
> index f8b0210..9189062 100644
> --- a/fs/nfsd/state.h
> +++ b/fs/nfsd/state.h
> @@ -352,6 +352,8 @@ struct nfs4_client {
>         struct rpc_wait_queue   cl_cb_waitq;    /* backchannel callers may */
>                                                 /* wait here for slots */
>         struct net              *net;
> +       struct list_head        async_copies;   /* list of async copies */
> +       spinlock_t              async_lock;     /* lock for async copies */
>  };
>
>  /* struct nfs4_client_reset
> diff --git a/fs/nfsd/xdr4.h b/fs/nfsd/xdr4.h
> index 9b0c099..0a19954 100644
> --- a/fs/nfsd/xdr4.h
> +++ b/fs/nfsd/xdr4.h
> @@ -529,6 +529,15 @@ struct nfsd4_copy {
>         struct nfsd4_callback   cp_cb;
>         __be32                  nfserr;
>         struct knfsd_fh         fh;
> +
> +       struct nfs4_client      *cp_clp;
> +
> +       struct file             *fh_src;
> +       struct file             *fh_dst;
> +       struct net              *net;
> +
> +       struct list_head        copies;
> +       struct task_struct      *copy_task;
>  };
>
>  struct nfsd4_seek {
> --
> 1.8.3.1
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-nfs" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html

  reply	other threads:[~2017-10-14 13:45 UTC|newest]

Thread overview: 21+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2017-10-13 20:54 [PATCH v5 00/10] NFSD support for asynchronous COPY Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 01/10] NFSD CB_OFFLOAD xdr Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 02/10] NFSD OFFLOAD_STATUS xdr Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 03/10] NFSD OFFLOAD_CANCEL xdr Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 04/10] NFSD xdr callback stateid in async COPY reply Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 05/10] NFSD first draft of async copy Olga Kornievskaia
2017-10-14 13:45   ` Olga Kornievskaia [this message]
2017-10-13 20:54 ` [PATCH v5 06/10] NFSD return nfs4_stid in nfs4_preprocess_stateid_op Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 07/10] NFSD create new stateid for async copy Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 08/10] NFSD handle OFFLOAD_CANCEL op Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 09/10] NFSD support OFFLOAD_STATUS Olga Kornievskaia
2017-10-13 20:54 ` [PATCH v5 10/10] NFSD stop queued async copies on client shutdown Olga Kornievskaia
2017-10-13 21:26 ` [PATCH v5 00/10] NFSD support for asynchronous COPY J. Bruce Fields
2017-10-14  0:09   ` Olga Kornievskaia
2017-10-16 13:13     ` Anna Schumaker
2017-10-16 13:37       ` Mauricio Tavares
2017-10-16 16:49       ` J. Bruce Fields
2017-10-16 19:25         ` Olga Kornievskaia
2017-10-23 21:48           ` Olga Kornievskaia
2017-10-23 22:39             ` Olga Kornievskaia
2017-10-24 13:35               ` J. Bruce Fields

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to='CAN-5tyHny-MDhJoo=xqEzsZAZf301p8Z0AAYnE6+yYr6R4xjJQ@mail.gmail.com' \
    --to=aglo@umich.edu \
    --cc=bfields@redhat.com \
    --cc=kolga@netapp.com \
    --cc=linux-nfs@vger.kernel.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.