Hey Dominique,

sorry for the delay, I've been quite busy these days.

The patches looks good to me and should indeed speed up the code a bit.
I quickly tested them against Syzkaller tuned for the 9p subsystem and
everything seems fine.

And by the way, which refcount races?

Cheers,
Tomas

On 12/11/18 1:41 PM, Dominique Martinet wrote:
> From: Dominique Martinet <dominique.marti...@cea.fr>
> 
> Add p9_client_async_rpc that will let us send an rpc without waiting
> for the reply, as well as an async handler hook.
> 
> The work done in the hook could mostly be done in the client callback,
> but I prefered not to for a couple of reasons:
>  - the callback can be called in an IRQ for some transports, and the
> handler needs to call p9_tag_remove which takes the client lock that has
> recently been reworked to not be irq-compatible (as it was never taken
> in IRQs until now)
>  - the handled replies can be handled anytime, there is nothing the
> userspace would care about and want to be notified of quickly
>  - the async request list and this function would be needed anyway for
> when we disconnect the client, in order to not leak async requests that
> haven't been replied to
> 
> Signed-off-by: Dominique Martinet <dominique.marti...@cea.fr>
> Cc: Eric Van Hensbergen <eri...@gmail.com>
> Cc: Latchesar Ionkov <lu...@ionkov.net>
> Cc: Tomas Bortoli <tomasbort...@gmail.com>
> Cc: Dmitry Vyukov <dvyu...@google.com>
> ---
> 
> I've been sitting on these patches for almost a month now because I
> wanted to fix the cancel race, but I think it's what the most recent
> syzbot email is about so if it could find it without this it probably
> won't hurt to at least get some reviews for these three patches first.
> 
> I won't submit these for next cycle, so will only put them into -next
> after 4.20 is released; hopefully I'll have time to look at the two
> pending refcount races around that time.

> Until then, comments please!
> 
> 
>  include/net/9p/client.h |  9 +++++-
>  net/9p/client.c         | 64 +++++++++++++++++++++++++++++++++++++++++
>  2 files changed, 72 insertions(+), 1 deletion(-)
> 
> diff --git a/include/net/9p/client.h b/include/net/9p/client.h
> index 947a570307a6..a4ded7666c73 100644
> --- a/include/net/9p/client.h
> +++ b/include/net/9p/client.h
> @@ -89,7 +89,8 @@ enum p9_req_status_t {
>   * @tc: the request fcall structure
>   * @rc: the response fcall structure
>   * @aux: transport specific data (provided for trans_fd migration)
> - * @req_list: link for higher level objects to chain requests
> + * @req_list: link used by trans_fd
> + * @async_list: link used to check on async requests
>   */
>  struct p9_req_t {
>       int status;
> @@ -100,6 +101,7 @@ struct p9_req_t {
>       struct p9_fcall rc;
>       void *aux;
>       struct list_head req_list;
> +     struct list_head async_list;
>  };
>  
>  /**
> @@ -110,6 +112,9 @@ struct p9_req_t {
>   * @trans_mod: module API instantiated with this client
>   * @status: connection state
>   * @trans: tranport instance state and API
> + * @fcall_cache: kmem cache for client request data (msize-specific)
> + * @async_req_lock: protects @async_req_list
> + * @async_req_list: list of requests waiting a reply
>   * @fids: All active FID handles
>   * @reqs: All active requests.
>   * @name: node name used as client id
> @@ -125,6 +130,8 @@ struct p9_client {
>       enum p9_trans_status status;
>       void *trans;
>       struct kmem_cache *fcall_cache;
> +     spinlock_t async_req_lock;
> +     struct list_head async_req_list;
>  
>       union {
>               struct {
> diff --git a/net/9p/client.c b/net/9p/client.c
> index 357214a51f13..0a67c3ccd4fd 100644
> --- a/net/9p/client.c
> +++ b/net/9p/client.c
> @@ -729,6 +729,62 @@ static struct p9_req_t *p9_client_prepare_req(struct 
> p9_client *c,
>       return ERR_PTR(err);
>  }
>  
> +static struct p9_req_t *
> +p9_client_async_rpc(struct p9_client *c, int8_t type, const char *fmt, ...)
> +{
> +     va_list ap;
> +     int err;
> +     struct p9_req_t *req;
> +
> +     va_start(ap, fmt);
> +     req = p9_client_prepare_req(c, type, c->msize, fmt, ap);
> +     va_end(ap);
> +     if (IS_ERR(req))
> +             return req;
> +
> +     err = c->trans_mod->request(c, req);
> +     if (err < 0) {
> +             /* bail out without flushing... */
> +             p9_req_put(req);
> +             p9_tag_remove(c, req);
> +             if (err != -ERESTARTSYS && err != -EFAULT)
> +                     c->status = Disconnected;
> +             return ERR_PTR(safe_errno(err));
> +     }
> +
> +     return req;
> +}
> +
> +static void p9_client_handle_async(struct p9_client *c, bool free_all)
> +{
> +     struct p9_req_t *req, *nreq;
> +
> +     /* it's ok to miss some async replies here, do a quick check without
> +      * lock first unless called with free_all
> +      */
> +     if (!free_all && list_empty(&c->async_req_list))
> +             return;
> +
> +     spin_lock_irq(&c->async_req_lock);
> +     list_for_each_entry_safe(req, nreq, &c->async_req_list, async_list) {
> +             if (free_all && req->status < REQ_STATUS_RCVD) {
> +                     p9_debug(P9_DEBUG_ERROR,
> +                              "final async handler found req tag %d type %d 
> status %d\n",
> +                              req->tc.tag, req->tc.id, req->status);
> +                     if (c->trans_mod->cancelled)
> +                             c->trans_mod->cancelled(c, req);
> +                     /* won't receive reply now */
> +                     p9_req_put(req);
> +             }
> +             if (free_all || req->status >= REQ_STATUS_RCVD) {
> +                     /* Put old refs whatever reqs actually returned */
> +                     list_del(&req->async_list);
> +                     p9_tag_remove(c, req);
> +             }
> +     }
> +     spin_unlock_irq(&c->async_req_lock);
> +}
> +
>  /**
>   * p9_client_rpc - issue a request and wait for a response
>   * @c: client session
> @@ -746,6 +802,8 @@ p9_client_rpc(struct p9_client *c, int8_t type, const 
> char *fmt, ...)
>       unsigned long flags;
>       struct p9_req_t *req;
>  
> +     p9_client_handle_async(c, 0);
> +
>       va_start(ap, fmt);
>       req = p9_client_prepare_req(c, type, c->msize, fmt, ap);
>       va_end(ap);
> @@ -841,6 +899,8 @@ static struct p9_req_t *p9_client_zc_rpc(struct p9_client 
> *c, int8_t type,
>       unsigned long flags;
>       struct p9_req_t *req;
>  
> +     p9_client_handle_async(c, 0);
> +
>       va_start(ap, fmt);
>       /*
>        * We allocate a inline protocol data of only 4k bytes.
> @@ -1030,6 +1090,8 @@ struct p9_client *p9_client_create(const char 
> *dev_name, char *options)
>       memcpy(clnt->name, client_id, strlen(client_id) + 1);
>  
>       spin_lock_init(&clnt->lock);
> +     spin_lock_init(&clnt->async_req_lock);
> +     INIT_LIST_HEAD(&clnt->async_req_list);
>       idr_init(&clnt->fids);
>       idr_init(&clnt->reqs);
>  
> @@ -1101,6 +1163,8 @@ void p9_client_destroy(struct p9_client *clnt)
>  
>       v9fs_put_trans(clnt->trans_mod);
>  
> +     p9_client_handle_async(clnt, 1);
> +
>       idr_for_each_entry(&clnt->fids, fid, id) {
>               pr_info("Found fid %d not clunked\n", fid->fid);
>               p9_fid_destroy(fid);
> 

Reply via email to