aboutsummaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
authorJ. Bruce Fields <bfields@redhat.com>2017-05-05 16:17:57 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2017-05-25 09:44:49 -0400
commit51d9c51523ec6927a068ee54280b5a4ff3bf401d (patch)
tree830e943dfa89a535594cd821ecc9cbba499ce18d /fs
parentea465551af30146efea215da58786ff732da70fb (diff)
nfsd: encoders mustn't use unitialized values in error cases
commit f961e3f2acae94b727380c0b74e2d3954d0edf79 upstream. In error cases, lgp->lg_layout_type may be out of bounds; so we shouldn't be using it until after the check of nfserr. This was seen to crash nfsd threads when the server receives a LAYOUTGET request with a large layout type. GETDEVICEINFO has the same problem. Reported-by: Ari Kauppi <Ari.Kauppi@synopsys.com> Reviewed-by: Christoph Hellwig <hch@lst.de> Signed-off-by: J. Bruce Fields <bfields@redhat.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'fs')
-rw-r--r--fs/nfsd/nfs4xdr.c8
1 files changed, 4 insertions, 4 deletions
diff --git a/fs/nfsd/nfs4xdr.c b/fs/nfsd/nfs4xdr.c
index c2d2895a1ec1..2ee80e1f5230 100644
--- a/fs/nfsd/nfs4xdr.c
+++ b/fs/nfsd/nfs4xdr.c
@@ -4081,8 +4081,7 @@ nfsd4_encode_getdeviceinfo(struct nfsd4_compoundres *resp, __be32 nfserr,
4081 struct nfsd4_getdeviceinfo *gdev) 4081 struct nfsd4_getdeviceinfo *gdev)
4082{ 4082{
4083 struct xdr_stream *xdr = &resp->xdr; 4083 struct xdr_stream *xdr = &resp->xdr;
4084 const struct nfsd4_layout_ops *ops = 4084 const struct nfsd4_layout_ops *ops;
4085 nfsd4_layout_ops[gdev->gd_layout_type];
4086 u32 starting_len = xdr->buf->len, needed_len; 4085 u32 starting_len = xdr->buf->len, needed_len;
4087 __be32 *p; 4086 __be32 *p;
4088 4087
@@ -4099,6 +4098,7 @@ nfsd4_encode_getdeviceinfo(struct nfsd4_compoundres *resp, __be32 nfserr,
4099 4098
4100 /* If maxcount is 0 then just update notifications */ 4099 /* If maxcount is 0 then just update notifications */
4101 if (gdev->gd_maxcount != 0) { 4100 if (gdev->gd_maxcount != 0) {
4101 ops = nfsd4_layout_ops[gdev->gd_layout_type];
4102 nfserr = ops->encode_getdeviceinfo(xdr, gdev); 4102 nfserr = ops->encode_getdeviceinfo(xdr, gdev);
4103 if (nfserr) { 4103 if (nfserr) {
4104 /* 4104 /*
@@ -4151,8 +4151,7 @@ nfsd4_encode_layoutget(struct nfsd4_compoundres *resp, __be32 nfserr,
4151 struct nfsd4_layoutget *lgp) 4151 struct nfsd4_layoutget *lgp)
4152{ 4152{
4153 struct xdr_stream *xdr = &resp->xdr; 4153 struct xdr_stream *xdr = &resp->xdr;
4154 const struct nfsd4_layout_ops *ops = 4154 const struct nfsd4_layout_ops *ops;
4155 nfsd4_layout_ops[lgp->lg_layout_type];
4156 __be32 *p; 4155 __be32 *p;
4157 4156
4158 dprintk("%s: err %d\n", __func__, nfserr); 4157 dprintk("%s: err %d\n", __func__, nfserr);
@@ -4175,6 +4174,7 @@ nfsd4_encode_layoutget(struct nfsd4_compoundres *resp, __be32 nfserr,
4175 *p++ = cpu_to_be32(lgp->lg_seg.iomode); 4174 *p++ = cpu_to_be32(lgp->lg_seg.iomode);
4176 *p++ = cpu_to_be32(lgp->lg_layout_type); 4175 *p++ = cpu_to_be32(lgp->lg_layout_type);
4177 4176
4177 ops = nfsd4_layout_ops[lgp->lg_layout_type];
4178 nfserr = ops->encode_layoutget(xdr, lgp); 4178 nfserr = ops->encode_layoutget(xdr, lgp);
4179out: 4179out:
4180 kfree(lgp->lg_content); 4180 kfree(lgp->lg_content);