Commit d3b00a80 authored by Anna Schumaker's avatar Anna Schumaker Committed by Trond Myklebust

NFS: Replace the READ_PLUS decoding code

We now take a 2-step process that allows us to place data and hole
segments directly at their final position in the xdr_stream without
needing to do a bunch of redundant copies to expand holes. Due to the
variable lengths of each segment, the xdr metadata might cross page
boundaries which I account for by setting a small scratch buffer so
xdr_inline_decode() won't fail.
Signed-off-by: default avatarAnna Schumaker <Anna.Schumaker@Netapp.com>
Signed-off-by: default avatarTrond Myklebust <trond.myklebust@hammerspace.com>
parent e1bd8760
...@@ -1025,73 +1025,84 @@ static int decode_deallocate(struct xdr_stream *xdr, struct nfs42_falloc_res *re ...@@ -1025,73 +1025,84 @@ static int decode_deallocate(struct xdr_stream *xdr, struct nfs42_falloc_res *re
return decode_op_hdr(xdr, OP_DEALLOCATE); return decode_op_hdr(xdr, OP_DEALLOCATE);
} }
static int decode_read_plus_data(struct xdr_stream *xdr, struct read_plus_segment {
struct nfs_pgio_args *args, enum data_content4 type;
struct nfs_pgio_res *res)
{
uint32_t count, recvd;
uint64_t offset; uint64_t offset;
union {
struct {
uint64_t length;
} hole;
struct {
uint32_t length;
unsigned int from;
} data;
};
};
static inline uint64_t read_plus_segment_length(struct read_plus_segment *seg)
{
return seg->type == NFS4_CONTENT_DATA ? seg->data.length : seg->hole.length;
}
static int decode_read_plus_segment(struct xdr_stream *xdr,
struct read_plus_segment *seg)
{
__be32 *p; __be32 *p;
p = xdr_inline_decode(xdr, 8 + 4); p = xdr_inline_decode(xdr, 4);
if (!p) if (!p)
return 1; return -EIO;
seg->type = be32_to_cpup(p++);
p = xdr_decode_hyper(p, &offset); p = xdr_inline_decode(xdr, seg->type == NFS4_CONTENT_DATA ? 12 : 16);
count = be32_to_cpup(p); if (!p)
recvd = xdr_align_data(xdr, res->count, xdr_align_size(count)); return -EIO;
if (recvd > count) p = xdr_decode_hyper(p, &seg->offset);
recvd = count;
if (res->count + recvd > args->count) { if (seg->type == NFS4_CONTENT_DATA) {
if (args->count > res->count) struct xdr_buf buf;
res->count += args->count - res->count; uint32_t len = be32_to_cpup(p);
return 1;
} seg->data.length = len;
res->count += recvd; seg->data.from = xdr_stream_pos(xdr);
if (count > recvd)
return 1; if (!xdr_stream_subsegment(xdr, &buf, xdr_align_size(len)))
return -EIO;
} else if (seg->type == NFS4_CONTENT_HOLE) {
xdr_decode_hyper(p, &seg->hole.length);
} else
return -EINVAL;
return 0; return 0;
} }
static int decode_read_plus_hole(struct xdr_stream *xdr, static int process_read_plus_segment(struct xdr_stream *xdr,
struct nfs_pgio_args *args, struct nfs_pgio_args *args,
struct nfs_pgio_res *res, uint32_t *eof) struct nfs_pgio_res *res,
struct read_plus_segment *seg)
{ {
uint64_t offset, length, recvd; unsigned long offset = seg->offset;
__be32 *p; unsigned long length = read_plus_segment_length(seg);
unsigned int bufpos;
p = xdr_inline_decode(xdr, 8 + 8); if (offset + length < args->offset)
if (!p) return 0;
return 1; else if (offset > args->offset + args->count) {
res->eof = 0;
p = xdr_decode_hyper(p, &offset); return 0;
p = xdr_decode_hyper(p, &length); } else if (offset < args->offset) {
if (offset != args->offset + res->count) { length -= (args->offset - offset);
/* Server returned an out-of-sequence extent */ offset = args->offset;
if (offset > args->offset + res->count || } else if (offset + length > args->offset + args->count) {
offset + length < args->offset + res->count) { length = (args->offset + args->count) - offset;
dprintk("NFS: server returned out of sequence extent: " res->eof = 0;
"offset/size = %llu/%llu != expected %llu\n",
(unsigned long long)offset,
(unsigned long long)length,
(unsigned long long)(args->offset +
res->count));
return 1;
}
length -= args->offset + res->count - offset;
}
if (length + res->count > args->count) {
*eof = 0;
if (unlikely(res->count >= args->count))
return 1;
length = args->count - res->count;
} }
recvd = xdr_expand_hole(xdr, res->count, length);
res->count += recvd;
if (recvd < length) bufpos = xdr->buf->head[0].iov_len + (offset - args->offset);
return 1; if (seg->type == NFS4_CONTENT_HOLE)
return 0; return xdr_stream_zero(xdr, bufpos, length);
else
return xdr_stream_move_subsegment(xdr, seg->data.from, bufpos, length);
} }
static int decode_read_plus(struct xdr_stream *xdr, struct nfs_pgio_res *res) static int decode_read_plus(struct xdr_stream *xdr, struct nfs_pgio_res *res)
...@@ -1099,8 +1110,10 @@ static int decode_read_plus(struct xdr_stream *xdr, struct nfs_pgio_res *res) ...@@ -1099,8 +1110,10 @@ static int decode_read_plus(struct xdr_stream *xdr, struct nfs_pgio_res *res)
struct nfs_pgio_header *hdr = struct nfs_pgio_header *hdr =
container_of(res, struct nfs_pgio_header, res); container_of(res, struct nfs_pgio_header, res);
struct nfs_pgio_args *args = &hdr->args; struct nfs_pgio_args *args = &hdr->args;
uint32_t eof, segments, type; uint32_t segments;
struct read_plus_segment *segs;
int status, i; int status, i;
char scratch_buf[16];
__be32 *p; __be32 *p;
status = decode_op_hdr(xdr, OP_READ_PLUS); status = decode_op_hdr(xdr, OP_READ_PLUS);
...@@ -1112,38 +1125,31 @@ static int decode_read_plus(struct xdr_stream *xdr, struct nfs_pgio_res *res) ...@@ -1112,38 +1125,31 @@ static int decode_read_plus(struct xdr_stream *xdr, struct nfs_pgio_res *res)
return -EIO; return -EIO;
res->count = 0; res->count = 0;
eof = be32_to_cpup(p++); res->eof = be32_to_cpup(p++);
segments = be32_to_cpup(p++); segments = be32_to_cpup(p++);
if (segments == 0) if (segments == 0)
goto out; return status;
for (i = 0; i < segments; i++) { segs = kmalloc_array(segments, sizeof(*segs), GFP_KERNEL);
p = xdr_inline_decode(xdr, 4); if (!segs)
if (!p) return -ENOMEM;
goto early_out;
type = be32_to_cpup(p++);
if (type == NFS4_CONTENT_DATA)
status = decode_read_plus_data(xdr, args, res);
else if (type == NFS4_CONTENT_HOLE)
status = decode_read_plus_hole(xdr, args, res, &eof);
else
return -EINVAL;
xdr_set_scratch_buffer(xdr, &scratch_buf, 32);
status = -EIO;
for (i = 0; i < segments; i++) {
status = decode_read_plus_segment(xdr, &segs[i]);
if (status < 0) if (status < 0)
return status; goto out;
if (status > 0)
goto early_out;
} }
xdr_set_pagelen(xdr, xdr_align_size(args->count));
for (i = segments; i > 0; i--)
res->count += process_read_plus_segment(xdr, args, res, &segs[i-1]);
status = 0;
out: out:
res->eof = eof; kfree(segs);
return 0; return status;
early_out:
if (unlikely(!i))
return -EIO;
res->eof = 0;
return 0;
} }
static int decode_seek(struct xdr_stream *xdr, struct nfs42_seek_res *res) static int decode_seek(struct xdr_stream *xdr, struct nfs42_seek_res *res)
......
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