diff options
author | J. Bruce Fields <bfields@redhat.com> | 2017-05-05 16:17:57 -0400 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2017-05-25 09:44:49 -0400 |
commit | 51d9c51523ec6927a068ee54280b5a4ff3bf401d (patch) | |
tree | 830e943dfa89a535594cd821ecc9cbba499ce18d /fs | |
parent | ea465551af30146efea215da58786ff732da70fb (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.c | 8 |
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); |
4179 | out: | 4179 | out: |
4180 | kfree(lgp->lg_content); | 4180 | kfree(lgp->lg_content); |