SUNRPC: Add helpers for decoding list discriminators symbolically

Use these helpers in a few spots to demonstrate their use.

The remaining open-coded discriminator checks in rpcrdma will be
addressed in subsequent patches.

Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
This commit is contained in:
Chuck Lever 2020-03-28 13:43:22 -04:00
parent 0b8dc1b699
commit 07e9a6325a
3 changed files with 40 additions and 15 deletions

View File

@ -474,6 +474,32 @@ xdr_stream_encode_uint32_array(struct xdr_stream *xdr,
return ret; return ret;
} }
/**
* xdr_item_is_absent - symbolically handle XDR discriminators
* @p: pointer to undecoded discriminator
*
* Return values:
* %true if the following XDR item is absent
* %false if the following XDR item is present
*/
static inline bool xdr_item_is_absent(const __be32 *p)
{
return *p == xdr_zero;
}
/**
* xdr_item_is_present - symbolically handle XDR discriminators
* @p: pointer to undecoded discriminator
*
* Return values:
* %true if the following XDR item is present
* %false if the following XDR item is absent
*/
static inline bool xdr_item_is_present(const __be32 *p)
{
return *p != xdr_zero;
}
/** /**
* xdr_stream_decode_u32 - Decode a 32-bit integer * xdr_stream_decode_u32 - Decode a 32-bit integer
* @xdr: pointer to xdr_stream * @xdr: pointer to xdr_stream

View File

@ -1133,11 +1133,11 @@ rpcrdma_is_bcall(struct rpcrdma_xprt *r_xprt, struct rpcrdma_rep *rep)
p = xdr_inline_decode(xdr, 0); p = xdr_inline_decode(xdr, 0);
/* Chunk lists */ /* Chunk lists */
if (*p++ != xdr_zero) if (xdr_item_is_present(p++))
return false; return false;
if (*p++ != xdr_zero) if (xdr_item_is_present(p++))
return false; return false;
if (*p++ != xdr_zero) if (xdr_item_is_present(p++))
return false; return false;
/* RPC header */ /* RPC header */
@ -1215,7 +1215,7 @@ static int decode_read_list(struct xdr_stream *xdr)
p = xdr_inline_decode(xdr, sizeof(*p)); p = xdr_inline_decode(xdr, sizeof(*p));
if (unlikely(!p)) if (unlikely(!p))
return -EIO; return -EIO;
if (unlikely(*p != xdr_zero)) if (unlikely(xdr_item_is_present(p)))
return -EIO; return -EIO;
return 0; return 0;
} }
@ -1234,7 +1234,7 @@ static int decode_write_list(struct xdr_stream *xdr, u32 *length)
p = xdr_inline_decode(xdr, sizeof(*p)); p = xdr_inline_decode(xdr, sizeof(*p));
if (unlikely(!p)) if (unlikely(!p))
return -EIO; return -EIO;
if (*p == xdr_zero) if (xdr_item_is_absent(p))
break; break;
if (!first) if (!first)
return -EIO; return -EIO;
@ -1256,7 +1256,7 @@ static int decode_reply_chunk(struct xdr_stream *xdr, u32 *length)
return -EIO; return -EIO;
*length = 0; *length = 0;
if (*p != xdr_zero) if (xdr_item_is_present(p))
if (decode_write_chunk(xdr, length)) if (decode_write_chunk(xdr, length))
return -EIO; return -EIO;
return 0; return 0;

View File

@ -419,7 +419,7 @@ static bool xdr_check_read_list(struct svc_rdma_recv_ctxt *rctxt)
len = 0; len = 0;
first = true; first = true;
while (*p != xdr_zero) { while (xdr_item_is_present(p)) {
p = xdr_inline_decode(&rctxt->rc_stream, p = xdr_inline_decode(&rctxt->rc_stream,
rpcrdma_readseg_maxsz * sizeof(*p)); rpcrdma_readseg_maxsz * sizeof(*p));
if (!p) if (!p)
@ -500,7 +500,7 @@ static bool xdr_check_write_list(struct svc_rdma_recv_ctxt *rctxt)
if (!p) if (!p)
return false; return false;
rctxt->rc_write_list = p; rctxt->rc_write_list = p;
while (*p != xdr_zero) { while (xdr_item_is_present(p)) {
if (!xdr_check_write_chunk(rctxt, MAX_BYTES_WRITE_CHUNK)) if (!xdr_check_write_chunk(rctxt, MAX_BYTES_WRITE_CHUNK))
return false; return false;
++chcount; ++chcount;
@ -532,12 +532,11 @@ static bool xdr_check_reply_chunk(struct svc_rdma_recv_ctxt *rctxt)
p = xdr_inline_decode(&rctxt->rc_stream, sizeof(*p)); p = xdr_inline_decode(&rctxt->rc_stream, sizeof(*p));
if (!p) if (!p)
return false; return false;
rctxt->rc_reply_chunk = p; rctxt->rc_reply_chunk = NULL;
if (*p != xdr_zero) { if (xdr_item_is_present(p)) {
if (!xdr_check_write_chunk(rctxt, MAX_BYTES_SPECIAL_CHUNK)) if (!xdr_check_write_chunk(rctxt, MAX_BYTES_SPECIAL_CHUNK))
return false; return false;
} else { rctxt->rc_reply_chunk = p;
rctxt->rc_reply_chunk = NULL;
} }
return true; return true;
} }
@ -568,7 +567,7 @@ static void svc_rdma_get_inv_rkey(struct svcxprt_rdma *rdma,
p += rpcrdma_fixed_maxsz; p += rpcrdma_fixed_maxsz;
/* Read list */ /* Read list */
while (*p++ != xdr_zero) { while (xdr_item_is_present(p++)) {
p++; /* position */ p++; /* position */
if (inv_rkey == xdr_zero) if (inv_rkey == xdr_zero)
inv_rkey = *p; inv_rkey = *p;
@ -578,7 +577,7 @@ static void svc_rdma_get_inv_rkey(struct svcxprt_rdma *rdma,
} }
/* Write list */ /* Write list */
while (*p++ != xdr_zero) { while (xdr_item_is_present(p++)) {
segcount = be32_to_cpup(p++); segcount = be32_to_cpup(p++);
for (i = 0; i < segcount; i++) { for (i = 0; i < segcount; i++) {
if (inv_rkey == xdr_zero) if (inv_rkey == xdr_zero)
@ -590,7 +589,7 @@ static void svc_rdma_get_inv_rkey(struct svcxprt_rdma *rdma,
} }
/* Reply chunk */ /* Reply chunk */
if (*p++ != xdr_zero) { if (xdr_item_is_present(p++)) {
segcount = be32_to_cpup(p++); segcount = be32_to_cpup(p++);
for (i = 0; i < segcount; i++) { for (i = 0; i < segcount; i++) {
if (inv_rkey == xdr_zero) if (inv_rkey == xdr_zero)