Commit b1818412 authored by Chuck Lever's avatar Chuck Lever

svcrdma: Start moving fields out of struct svc_rdma_read_info

Since the request's svc_rdma_recv_ctxt will stay around for the
duration of the RDMA Read operation, the contents of struct
svc_rdma_read_info can reside in the request's svc_rdma_recv_ctxt
rather than being allocated separately. This will eventually save a
call to kmalloc() in a hot path.

Start this clean-up by moving the Read chunk's svc_rdma_chunk_ctxt.
Signed-off-by: default avatarChuck Lever <chuck.lever@oracle.com>
parent 6a04a434
...@@ -156,6 +156,10 @@ struct svc_rdma_recv_ctxt { ...@@ -156,6 +156,10 @@ struct svc_rdma_recv_ctxt {
u32 rc_inv_rkey; u32 rc_inv_rkey;
__be32 rc_msgtype; __be32 rc_msgtype;
/* State for pulling a Read chunk */
unsigned int rc_readbytes;
struct svc_rdma_chunk_ctxt rc_cc;
struct svc_rdma_pcl rc_call_pcl; struct svc_rdma_pcl rc_call_pcl;
struct svc_rdma_pcl rc_read_pcl; struct svc_rdma_pcl rc_read_pcl;
......
...@@ -294,9 +294,6 @@ struct svc_rdma_read_info { ...@@ -294,9 +294,6 @@ struct svc_rdma_read_info {
struct svc_rdma_recv_ctxt *ri_readctxt; struct svc_rdma_recv_ctxt *ri_readctxt;
unsigned int ri_pageno; unsigned int ri_pageno;
unsigned int ri_pageoff; unsigned int ri_pageoff;
unsigned int ri_totalbytes;
struct svc_rdma_chunk_ctxt ri_cc;
}; };
static struct svc_rdma_read_info * static struct svc_rdma_read_info *
...@@ -304,20 +301,13 @@ svc_rdma_read_info_alloc(struct svcxprt_rdma *rdma) ...@@ -304,20 +301,13 @@ svc_rdma_read_info_alloc(struct svcxprt_rdma *rdma)
{ {
struct svc_rdma_read_info *info; struct svc_rdma_read_info *info;
info = kmalloc_node(sizeof(*info), GFP_KERNEL, return kmalloc_node(sizeof(*info), GFP_KERNEL,
ibdev_to_node(rdma->sc_cm_id->device)); ibdev_to_node(rdma->sc_cm_id->device));
if (!info)
return info;
svc_rdma_cc_init(rdma, &info->ri_cc);
info->ri_cc.cc_cqe.done = svc_rdma_wc_read_done;
return info;
} }
static void svc_rdma_read_info_free(struct svcxprt_rdma *rdma, static void svc_rdma_read_info_free(struct svcxprt_rdma *rdma,
struct svc_rdma_read_info *info) struct svc_rdma_read_info *info)
{ {
svc_rdma_cc_release(rdma, &info->ri_cc, DMA_FROM_DEVICE);
kfree(info); kfree(info);
} }
...@@ -333,12 +323,12 @@ static void svc_rdma_wc_read_done(struct ib_cq *cq, struct ib_wc *wc) ...@@ -333,12 +323,12 @@ static void svc_rdma_wc_read_done(struct ib_cq *cq, struct ib_wc *wc)
struct ib_cqe *cqe = wc->wr_cqe; struct ib_cqe *cqe = wc->wr_cqe;
struct svc_rdma_chunk_ctxt *cc = struct svc_rdma_chunk_ctxt *cc =
container_of(cqe, struct svc_rdma_chunk_ctxt, cc_cqe); container_of(cqe, struct svc_rdma_chunk_ctxt, cc_cqe);
struct svc_rdma_read_info *info; struct svc_rdma_recv_ctxt *ctxt;
switch (wc->status) { switch (wc->status) {
case IB_WC_SUCCESS: case IB_WC_SUCCESS:
info = container_of(cc, struct svc_rdma_read_info, ri_cc); ctxt = container_of(cc, struct svc_rdma_recv_ctxt, rc_cc);
trace_svcrdma_wc_read(wc, &cc->cc_cid, info->ri_totalbytes, trace_svcrdma_wc_read(wc, &cc->cc_cid, ctxt->rc_readbytes,
cc->cc_posttime); cc->cc_posttime);
break; break;
case IB_WC_WR_FLUSH_ERR: case IB_WC_WR_FLUSH_ERR:
...@@ -708,7 +698,7 @@ static int svc_rdma_build_read_segment(struct svcxprt_rdma *rdma, ...@@ -708,7 +698,7 @@ static int svc_rdma_build_read_segment(struct svcxprt_rdma *rdma,
const struct svc_rdma_segment *segment) const struct svc_rdma_segment *segment)
{ {
struct svc_rdma_recv_ctxt *head = info->ri_readctxt; struct svc_rdma_recv_ctxt *head = info->ri_readctxt;
struct svc_rdma_chunk_ctxt *cc = &info->ri_cc; struct svc_rdma_chunk_ctxt *cc = &head->rc_cc;
struct svc_rqst *rqstp = info->ri_rqst; struct svc_rqst *rqstp = info->ri_rqst;
unsigned int sge_no, seg_len, len; unsigned int sge_no, seg_len, len;
struct svc_rdma_rw_ctxt *ctxt; struct svc_rdma_rw_ctxt *ctxt;
...@@ -778,6 +768,7 @@ static int svc_rdma_build_read_chunk(struct svcxprt_rdma *rdma, ...@@ -778,6 +768,7 @@ static int svc_rdma_build_read_chunk(struct svcxprt_rdma *rdma,
struct svc_rdma_read_info *info, struct svc_rdma_read_info *info,
const struct svc_rdma_chunk *chunk) const struct svc_rdma_chunk *chunk)
{ {
struct svc_rdma_recv_ctxt *head = info->ri_readctxt;
const struct svc_rdma_segment *segment; const struct svc_rdma_segment *segment;
int ret; int ret;
...@@ -786,7 +777,7 @@ static int svc_rdma_build_read_chunk(struct svcxprt_rdma *rdma, ...@@ -786,7 +777,7 @@ static int svc_rdma_build_read_chunk(struct svcxprt_rdma *rdma,
ret = svc_rdma_build_read_segment(rdma, info, segment); ret = svc_rdma_build_read_segment(rdma, info, segment);
if (ret < 0) if (ret < 0)
break; break;
info->ri_totalbytes += segment->rs_length; head->rc_readbytes += segment->rs_length;
} }
return ret; return ret;
} }
...@@ -828,7 +819,7 @@ static int svc_rdma_copy_inline_range(struct svc_rdma_read_info *info, ...@@ -828,7 +819,7 @@ static int svc_rdma_copy_inline_range(struct svc_rdma_read_info *info,
dst = page_address(rqstp->rq_pages[info->ri_pageno]); dst = page_address(rqstp->rq_pages[info->ri_pageno]);
memcpy(dst + info->ri_pageno, src + offset, page_len); memcpy(dst + info->ri_pageno, src + offset, page_len);
info->ri_totalbytes += page_len; head->rc_readbytes += page_len;
info->ri_pageoff += page_len; info->ri_pageoff += page_len;
if (info->ri_pageoff == PAGE_SIZE) { if (info->ri_pageoff == PAGE_SIZE) {
info->ri_pageno++; info->ri_pageno++;
...@@ -883,7 +874,7 @@ static noinline int svc_rdma_read_multiple_chunks(struct svcxprt_rdma *rdma, ...@@ -883,7 +874,7 @@ static noinline int svc_rdma_read_multiple_chunks(struct svcxprt_rdma *rdma,
break; break;
start += length; start += length;
length = next->ch_position - info->ri_totalbytes; length = next->ch_position - head->rc_readbytes;
ret = svc_rdma_copy_inline_range(info, start, length); ret = svc_rdma_copy_inline_range(info, start, length);
if (ret < 0) if (ret < 0)
return ret; return ret;
...@@ -895,13 +886,13 @@ static noinline int svc_rdma_read_multiple_chunks(struct svcxprt_rdma *rdma, ...@@ -895,13 +886,13 @@ static noinline int svc_rdma_read_multiple_chunks(struct svcxprt_rdma *rdma,
if (ret < 0) if (ret < 0)
return ret; return ret;
buf->len += info->ri_totalbytes; buf->len += head->rc_readbytes;
buf->buflen += info->ri_totalbytes; buf->buflen += head->rc_readbytes;
buf->head[0].iov_base = page_address(info->ri_rqst->rq_pages[0]); buf->head[0].iov_base = page_address(info->ri_rqst->rq_pages[0]);
buf->head[0].iov_len = min_t(size_t, PAGE_SIZE, info->ri_totalbytes); buf->head[0].iov_len = min_t(size_t, PAGE_SIZE, head->rc_readbytes);
buf->pages = &info->ri_rqst->rq_pages[1]; buf->pages = &info->ri_rqst->rq_pages[1];
buf->page_len = info->ri_totalbytes - buf->head[0].iov_len; buf->page_len = head->rc_readbytes - buf->head[0].iov_len;
return 0; return 0;
} }
...@@ -985,6 +976,7 @@ static int svc_rdma_read_chunk_range(struct svcxprt_rdma *rdma, ...@@ -985,6 +976,7 @@ static int svc_rdma_read_chunk_range(struct svcxprt_rdma *rdma,
const struct svc_rdma_chunk *chunk, const struct svc_rdma_chunk *chunk,
unsigned int offset, unsigned int length) unsigned int offset, unsigned int length)
{ {
struct svc_rdma_recv_ctxt *head = info->ri_readctxt;
const struct svc_rdma_segment *segment; const struct svc_rdma_segment *segment;
int ret; int ret;
...@@ -1005,7 +997,7 @@ static int svc_rdma_read_chunk_range(struct svcxprt_rdma *rdma, ...@@ -1005,7 +997,7 @@ static int svc_rdma_read_chunk_range(struct svcxprt_rdma *rdma,
if (ret < 0) if (ret < 0)
break; break;
info->ri_totalbytes += dummy.rs_length; head->rc_readbytes += dummy.rs_length;
length -= dummy.rs_length; length -= dummy.rs_length;
offset = 0; offset = 0;
} }
...@@ -1055,7 +1047,7 @@ static int svc_rdma_read_call_chunk(struct svcxprt_rdma *rdma, ...@@ -1055,7 +1047,7 @@ static int svc_rdma_read_call_chunk(struct svcxprt_rdma *rdma,
break; break;
start += length; start += length;
length = next->ch_position - info->ri_totalbytes; length = next->ch_position - head->rc_readbytes;
ret = svc_rdma_read_chunk_range(rdma, info, call_chunk, ret = svc_rdma_read_chunk_range(rdma, info, call_chunk,
start, length); start, length);
if (ret < 0) if (ret < 0)
...@@ -1089,6 +1081,7 @@ static int svc_rdma_read_call_chunk(struct svcxprt_rdma *rdma, ...@@ -1089,6 +1081,7 @@ static int svc_rdma_read_call_chunk(struct svcxprt_rdma *rdma,
static noinline int svc_rdma_read_special(struct svcxprt_rdma *rdma, static noinline int svc_rdma_read_special(struct svcxprt_rdma *rdma,
struct svc_rdma_read_info *info) struct svc_rdma_read_info *info)
{ {
struct svc_rdma_recv_ctxt *head = info->ri_readctxt;
struct xdr_buf *buf = &info->ri_rqst->rq_arg; struct xdr_buf *buf = &info->ri_rqst->rq_arg;
int ret; int ret;
...@@ -1096,13 +1089,13 @@ static noinline int svc_rdma_read_special(struct svcxprt_rdma *rdma, ...@@ -1096,13 +1089,13 @@ static noinline int svc_rdma_read_special(struct svcxprt_rdma *rdma,
if (ret < 0) if (ret < 0)
goto out; goto out;
buf->len += info->ri_totalbytes; buf->len += head->rc_readbytes;
buf->buflen += info->ri_totalbytes; buf->buflen += head->rc_readbytes;
buf->head[0].iov_base = page_address(info->ri_rqst->rq_pages[0]); buf->head[0].iov_base = page_address(info->ri_rqst->rq_pages[0]);
buf->head[0].iov_len = min_t(size_t, PAGE_SIZE, info->ri_totalbytes); buf->head[0].iov_len = min_t(size_t, PAGE_SIZE, head->rc_readbytes);
buf->pages = &info->ri_rqst->rq_pages[1]; buf->pages = &info->ri_rqst->rq_pages[1];
buf->page_len = info->ri_totalbytes - buf->head[0].iov_len; buf->page_len = head->rc_readbytes - buf->head[0].iov_len;
out: out:
return ret; return ret;
...@@ -1135,19 +1128,20 @@ int svc_rdma_process_read_list(struct svcxprt_rdma *rdma, ...@@ -1135,19 +1128,20 @@ int svc_rdma_process_read_list(struct svcxprt_rdma *rdma,
struct svc_rqst *rqstp, struct svc_rqst *rqstp,
struct svc_rdma_recv_ctxt *head) struct svc_rdma_recv_ctxt *head)
{ {
struct svc_rdma_chunk_ctxt *cc = &head->rc_cc;
struct svc_rdma_read_info *info; struct svc_rdma_read_info *info;
struct svc_rdma_chunk_ctxt *cc;
int ret; int ret;
info = svc_rdma_read_info_alloc(rdma); info = svc_rdma_read_info_alloc(rdma);
if (!info) if (!info)
return -ENOMEM; return -ENOMEM;
cc = &info->ri_cc;
info->ri_rqst = rqstp; info->ri_rqst = rqstp;
info->ri_readctxt = head; info->ri_readctxt = head;
info->ri_pageno = 0; info->ri_pageno = 0;
info->ri_pageoff = 0; info->ri_pageoff = 0;
info->ri_totalbytes = 0; svc_rdma_cc_init(rdma, cc);
cc->cc_cqe.done = svc_rdma_wc_read_done;
head->rc_readbytes = 0;
if (pcl_is_empty(&head->rc_call_pcl)) { if (pcl_is_empty(&head->rc_call_pcl)) {
if (head->rc_read_pcl.cl_count == 1) if (head->rc_read_pcl.cl_count == 1)
...@@ -1178,6 +1172,7 @@ int svc_rdma_process_read_list(struct svcxprt_rdma *rdma, ...@@ -1178,6 +1172,7 @@ int svc_rdma_process_read_list(struct svcxprt_rdma *rdma,
head->rc_page_count = 0; head->rc_page_count = 0;
out_err: out_err:
svc_rdma_cc_release(rdma, cc, DMA_FROM_DEVICE);
svc_rdma_read_info_free(rdma, info); svc_rdma_read_info_free(rdma, info);
return ret; return ret;
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment