aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorQuentin Casasnovas <quentin.casasnovas@oracle.com>2015-04-14 05:25:43 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2015-04-28 06:53:16 -0400
commit0d3bba0287d4e284c3ec7d3397e81eec920d5e7e (patch)
treeb41e9c9a3992a140837df671e9cef3fe64e49667
parentbb304b71f8dbcb284f0f876dfb4eecbadf079773 (diff)
cdc-acm: prevent infinite loop when parsing CDC headers.
Phil and I found out a problem with commit: 7e860a6e7aa6 ("cdc-acm: add sanity checks") It added some sanity checks to ignore potential garbage in CDC headers but also introduced a potential infinite loop. This can happen at the first loop iteration (elength = 0 in that case) if the description isn't a DT_CS_INTERFACE or later if 'buffer[0]' is zero. It should also be noted that the wrong length was being added to 'buffer' in case 'buffer[1]' was not a DT_CS_INTERFACE descriptor, since elength was assigned after that check in the loop. A specially crafted USB device could be used to trigger this infinite loop. Fixes: 7e860a6e7aa6 ("cdc-acm: add sanity checks") Signed-off-by: Phil Turnbull <phil.turnbull@oracle.com> Signed-off-by: Quentin Casasnovas <quentin.casasnovas@oracle.com> CC: Sergei Shtylyov <sergei.shtylyov@cogentembedded.com> CC: Oliver Neukum <oneukum@suse.de> CC: Adam Lee <adam8157@gmail.com> CC: <stable@vger.kernel.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r--drivers/usb/class/cdc-acm.c7
1 files changed, 6 insertions, 1 deletions
diff --git a/drivers/usb/class/cdc-acm.c b/drivers/usb/class/cdc-acm.c
index 3e15add665e2..5c8f58114677 100644
--- a/drivers/usb/class/cdc-acm.c
+++ b/drivers/usb/class/cdc-acm.c
@@ -1142,11 +1142,16 @@ static int acm_probe(struct usb_interface *intf,
1142 } 1142 }
1143 1143
1144 while (buflen > 0) { 1144 while (buflen > 0) {
1145 elength = buffer[0];
1146 if (!elength) {
1147 dev_err(&intf->dev, "skipping garbage byte\n");
1148 elength = 1;
1149 goto next_desc;
1150 }
1145 if (buffer[1] != USB_DT_CS_INTERFACE) { 1151 if (buffer[1] != USB_DT_CS_INTERFACE) {
1146 dev_err(&intf->dev, "skipping garbage\n"); 1152 dev_err(&intf->dev, "skipping garbage\n");
1147 goto next_desc; 1153 goto next_desc;
1148 } 1154 }
1149 elength = buffer[0];
1150 1155
1151 switch (buffer[2]) { 1156 switch (buffer[2]) {
1152 case USB_CDC_UNION_TYPE: /* we've found it */ 1157 case USB_CDC_UNION_TYPE: /* we've found it */