]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
USB: whiteheat: Added bounds checking for bulk command response
authorJames Forshaw <forshaw@google.com>
Wed, 17 Sep 2014 09:21:53 +0000 (10:21 +0100)
committerWilly Tarreau <w@1wt.eu>
Sat, 13 Dec 2014 14:16:14 +0000 (15:16 +0100)
commit c5fd4126151855330280ea9382684980afcfdd03 upstream

This patch fixes a potential security issue in the whiteheat USB driver
which might allow a local attacker to cause kernel memory corrpution. This
is due to an unchecked memcpy into a fixed size buffer (of 64 bytes). On
EHCI and XHCI busses it's possible to craft responses greater than 64
bytes leading a buffer overflow.

Signed-off-by: James Forshaw <forshaw@google.com>
Cc: stable <stable@vger.kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
(backported from commit 6817ae225cd650fb1c3295d769298c38b1eba818)
CVE-2014-3185
BugLink: http://bugs.launchpad.net/bugs/1370036
Signed-off-by: Luis Henriques <luis.henriques@canonical.com>
Signed-off-by: Tim Gardner <tim.gardner@canonical.com>
Signed-off-by: Willy Tarreau <w@1wt.eu>
drivers/usb/serial/whiteheat.c

index 1247be1262ae0cfe9967bca47def1e2cbd58dadf..748c62716f4947b8f39c8c0356585c451a020a21 100644 (file)
@@ -1012,6 +1012,10 @@ static void command_port_read_callback(struct urb *urb)
                dbg("%s - command_info is NULL, exiting.", __func__);
                return;
        }
+       if (!urb->actual_length) {
+               dev_dbg(&urb->dev->dev, "%s - empty response, exiting.\n", __func__);
+               return;
+       }
        if (status) {
                dbg("%s - nonzero urb status: %d", __func__, status);
                if (status != -ENOENT)
@@ -1033,7 +1037,8 @@ static void command_port_read_callback(struct urb *urb)
                /* These are unsolicited reports from the firmware, hence no
                   waiting command to wakeup */
                dbg("%s - event received", __func__);
-       } else if (data[0] == WHITEHEAT_GET_DTR_RTS) {
+       } else if ((data[0] == WHITEHEAT_GET_DTR_RTS) &&
+               (urb->actual_length - 1 <= sizeof(command_info->result_buffer))) {
                memcpy(command_info->result_buffer, &data[1],
                                                urb->actual_length - 1);
                command_info->command_finished = WHITEHEAT_CMD_COMPLETE;