]> www.infradead.org Git - nvme.git/commitdiff
SUNRPC/xprt: async tasks mustn't block waiting for memory
authorNeilBrown <neilb@suse.de>
Sun, 6 Mar 2022 23:41:44 +0000 (10:41 +1100)
committerTrond Myklebust <trond.myklebust@hammerspace.com>
Sun, 13 Mar 2022 16:59:35 +0000 (12:59 -0400)
When memory is short, new worker threads cannot be created and we depend
on the minimum one rpciod thread to be able to handle everything.  So it
must not block waiting for memory.

xprt_dynamic_alloc_slot can block indefinitely.  This can tie up all
workqueue threads and NFS can deadlock.  So when called from a
workqueue, set __GFP_NORETRY.

The rdma alloc_slot already does not block.  However it sets the error
to -EAGAIN suggesting this will trigger a sleep.  It does not.  As we
can see in call_reserveresult(), only -ENOMEM causes a sleep.  -EAGAIN
causes immediate retry.

Signed-off-by: NeilBrown <neilb@suse.de>
Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com>
net/sunrpc/xprt.c
net/sunrpc/xprtrdma/transport.c

index 9f0025e0742c3079e01f48587b4bb480fa80b8d5..2d1f84aea516538be4e59da49360918179b70e2d 100644 (file)
@@ -1687,12 +1687,15 @@ out:
 static struct rpc_rqst *xprt_dynamic_alloc_slot(struct rpc_xprt *xprt)
 {
        struct rpc_rqst *req = ERR_PTR(-EAGAIN);
+       gfp_t gfp_mask = GFP_KERNEL;
 
        if (xprt->num_reqs >= xprt->max_reqs)
                goto out;
        ++xprt->num_reqs;
        spin_unlock(&xprt->reserve_lock);
-       req = kzalloc(sizeof(struct rpc_rqst), GFP_KERNEL);
+       if (current->flags & PF_WQ_WORKER)
+               gfp_mask |= __GFP_NORETRY | __GFP_NOWARN;
+       req = kzalloc(sizeof(*req), gfp_mask);
        spin_lock(&xprt->reserve_lock);
        if (req != NULL)
                goto out;
index 5714bf880e95fa9cf42532c0ee2c76d57dec1ec8..923e4b512ee9ece7dda5985a48d0be27c429a460 100644 (file)
@@ -517,7 +517,7 @@ xprt_rdma_alloc_slot(struct rpc_xprt *xprt, struct rpc_task *task)
        return;
 
 out_sleep:
-       task->tk_status = -EAGAIN;
+       task->tk_status = -ENOMEM;
        xprt_add_backlog(xprt, task);
 }