提交 62b58848 编写于 作者: M Matthew Garrett 提交者: Greg Kroah-Hartman

isight_firmware: Avoid crash on loading invalid firmware

Different tools generate slightly different formats of the isight
firmware. Ensure that the firmware buffer is not overrun, while still
ensuring that the correct amount of data is written if trailing data is
present.
Signed-off-by: NMatthew Garrett <mjg@redhat.com>
Report-by: NJustin Mattock <justinmattock@gmail.com>
Tested-by: NJustin Mattock <justinmattock@gmail.com>
Signed-off-by: NAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: NGreg Kroah-Hartman <gregkh@suse.de>
上级 6460a261
...@@ -39,9 +39,12 @@ static int isight_firmware_load(struct usb_interface *intf, ...@@ -39,9 +39,12 @@ static int isight_firmware_load(struct usb_interface *intf,
struct usb_device *dev = interface_to_usbdev(intf); struct usb_device *dev = interface_to_usbdev(intf);
int llen, len, req, ret = 0; int llen, len, req, ret = 0;
const struct firmware *firmware; const struct firmware *firmware;
unsigned char *buf; unsigned char *buf = kmalloc(50, GFP_KERNEL);
unsigned char data[4]; unsigned char data[4];
char *ptr; u8 *ptr;
if (!buf)
return -ENOMEM;
if (request_firmware(&firmware, "isight.fw", &dev->dev) != 0) { if (request_firmware(&firmware, "isight.fw", &dev->dev) != 0) {
printk(KERN_ERR "Unable to load isight firmware\n"); printk(KERN_ERR "Unable to load isight firmware\n");
...@@ -59,7 +62,7 @@ static int isight_firmware_load(struct usb_interface *intf, ...@@ -59,7 +62,7 @@ static int isight_firmware_load(struct usb_interface *intf,
goto out; goto out;
} }
while (1) { while (ptr+4 <= firmware->data+firmware->size) {
memcpy(data, ptr, 4); memcpy(data, ptr, 4);
len = (data[0] << 8 | data[1]); len = (data[0] << 8 | data[1]);
req = (data[2] << 8 | data[3]); req = (data[2] << 8 | data[3]);
...@@ -71,10 +74,14 @@ static int isight_firmware_load(struct usb_interface *intf, ...@@ -71,10 +74,14 @@ static int isight_firmware_load(struct usb_interface *intf,
continue; continue;
for (; len > 0; req += 50) { for (; len > 0; req += 50) {
llen = len > 50 ? 50 : len; llen = min(len, 50);
len -= llen; len -= llen;
if (ptr+llen > firmware->data+firmware->size) {
buf = kmalloc(llen, GFP_KERNEL); printk(KERN_ERR
"Malformed isight firmware");
ret = -ENODEV;
goto out;
}
memcpy(buf, ptr, llen); memcpy(buf, ptr, llen);
ptr += llen; ptr += llen;
...@@ -89,16 +96,18 @@ static int isight_firmware_load(struct usb_interface *intf, ...@@ -89,16 +96,18 @@ static int isight_firmware_load(struct usb_interface *intf,
goto out; goto out;
} }
kfree(buf);
} }
} }
if (usb_control_msg if (usb_control_msg
(dev, usb_sndctrlpipe(dev, 0), 0xa0, 0x40, 0xe600, 0, "\0", 1, (dev, usb_sndctrlpipe(dev, 0), 0xa0, 0x40, 0xe600, 0, "\0", 1,
300) != 1) { 300) != 1) {
printk(KERN_ERR "isight firmware loading completion failed\n"); printk(KERN_ERR "isight firmware loading completion failed\n");
ret = -ENODEV; ret = -ENODEV;
} }
out: out:
kfree(buf);
release_firmware(firmware); release_firmware(firmware);
return ret; return ret;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册