diff options
author | Pontus Fuchs <pontusf@broadcom.com> | 2015-03-06 10:18:41 -0500 |
---|---|---|
committer | Kalle Valo <kvalo@codeaurora.org> | 2015-03-07 03:52:30 -0500 |
commit | 3f1615340acea54e21f4b9d4d65921540dca84b2 (patch) | |
tree | 36dd5fa76bba565410b585c491719c8197fc18f9 | |
parent | eb9f75d88ebc0c17c1025972e5298ab965fee31b (diff) |
brcmfmac: Perform bound checking on vendor command buffer
A short or malformed vendor command buffer could cause reads outside
the command buffer.
Cc: stable@vger.kernel.org # v3.19
Signed-off-by: Pontus Fuchs <pontusf@broadcom.com>
[arend@broadcom.com: slightly modified debug trace output]
Signed-off-by: Arend van Spriel <arend@broadcom.com>
Signed-off-by: Kalle Valo <kvalo@codeaurora.org>
-rw-r--r-- | drivers/net/wireless/brcm80211/brcmfmac/vendor.c | 15 |
1 files changed, 12 insertions, 3 deletions
diff --git a/drivers/net/wireless/brcm80211/brcmfmac/vendor.c b/drivers/net/wireless/brcm80211/brcmfmac/vendor.c index 50cdf7090198..8eff2753abad 100644 --- a/drivers/net/wireless/brcm80211/brcmfmac/vendor.c +++ b/drivers/net/wireless/brcm80211/brcmfmac/vendor.c | |||
@@ -39,13 +39,22 @@ static int brcmf_cfg80211_vndr_cmds_dcmd_handler(struct wiphy *wiphy, | |||
39 | void *dcmd_buf = NULL, *wr_pointer; | 39 | void *dcmd_buf = NULL, *wr_pointer; |
40 | u16 msglen, maxmsglen = PAGE_SIZE - 0x100; | 40 | u16 msglen, maxmsglen = PAGE_SIZE - 0x100; |
41 | 41 | ||
42 | brcmf_dbg(TRACE, "cmd %x set %d len %d\n", cmdhdr->cmd, cmdhdr->set, | 42 | if (len < sizeof(*cmdhdr)) { |
43 | cmdhdr->len); | 43 | brcmf_err("vendor command too short: %d\n", len); |
44 | return -EINVAL; | ||
45 | } | ||
44 | 46 | ||
45 | vif = container_of(wdev, struct brcmf_cfg80211_vif, wdev); | 47 | vif = container_of(wdev, struct brcmf_cfg80211_vif, wdev); |
46 | ifp = vif->ifp; | 48 | ifp = vif->ifp; |
47 | 49 | ||
48 | len -= sizeof(struct brcmf_vndr_dcmd_hdr); | 50 | brcmf_dbg(TRACE, "ifidx=%d, cmd=%d\n", ifp->ifidx, cmdhdr->cmd); |
51 | |||
52 | if (cmdhdr->offset > len) { | ||
53 | brcmf_err("bad buffer offset %d > %d\n", cmdhdr->offset, len); | ||
54 | return -EINVAL; | ||
55 | } | ||
56 | |||
57 | len -= cmdhdr->offset; | ||
49 | ret_len = cmdhdr->len; | 58 | ret_len = cmdhdr->len; |
50 | if (ret_len > 0 || len > 0) { | 59 | if (ret_len > 0 || len > 0) { |
51 | if (len > BRCMF_DCMD_MAXLEN) { | 60 | if (len > BRCMF_DCMD_MAXLEN) { |