Commit c1e15eeb authored by Duncan Sands's avatar Duncan Sands Committed by Greg Kroah-Hartman

[PATCH] usb: extract sensible strings from buggy string descriptors

The Freebox is a USB modem popular in France.  It returns bogus string
descriptors: while the string part is there, the length and type bytes
are both zero.  This patch detects that case and tries to recover a
sensible string by scanning for printable Latin characters.  This not
only causes the modem to spring to life (because usbnet gets a valid
MAC address) - it also means you get the correct Product and Manufacturer
strings in sysfs and elsewhere.  This patch is in the "mostly harmless"
category.
Signed-off-by: default avatarDuncan Sands <baldrick@free.fr>
Signed-off-by: default avatarGreg Kroah-Hartman <greg@kroah.com>
parent 957f22f3
...@@ -17,6 +17,8 @@ ...@@ -17,6 +17,8 @@
#include <linux/init.h> #include <linux/init.h>
#include <linux/mm.h> #include <linux/mm.h>
#include <linux/timer.h> #include <linux/timer.h>
#include <linux/ctype.h>
#include <linux/device.h>
#include <asm/byteorder.h> #include <asm/byteorder.h>
#include "hcd.h" /* for usbcore internals */ #include "hcd.h" /* for usbcore internals */
...@@ -623,6 +625,20 @@ int usb_get_string(struct usb_device *dev, unsigned short langid, ...@@ -623,6 +625,20 @@ int usb_get_string(struct usb_device *dev, unsigned short langid,
return result; return result;
} }
static void usb_try_string_workarounds(unsigned char *buf, int *length)
{
int newlength, oldlength = *length;
for (newlength = 2; newlength + 1 < oldlength; newlength += 2)
if (!isprint(buf[newlength]) || buf[newlength + 1])
break;
if (newlength > 2) {
buf[0] = newlength;
*length = newlength;
}
}
static int usb_string_sub(struct usb_device *dev, unsigned int langid, static int usb_string_sub(struct usb_device *dev, unsigned int langid,
unsigned int index, unsigned char *buf) unsigned int index, unsigned char *buf)
{ {
...@@ -634,19 +650,26 @@ static int usb_string_sub(struct usb_device *dev, unsigned int langid, ...@@ -634,19 +650,26 @@ static int usb_string_sub(struct usb_device *dev, unsigned int langid,
/* If that failed try to read the descriptor length, then /* If that failed try to read the descriptor length, then
* ask for just that many bytes */ * ask for just that many bytes */
if (rc < 0) { if (rc < 2) {
rc = usb_get_string(dev, langid, index, buf, 2); rc = usb_get_string(dev, langid, index, buf, 2);
if (rc == 2) if (rc == 2)
rc = usb_get_string(dev, langid, index, buf, buf[0]); rc = usb_get_string(dev, langid, index, buf, buf[0]);
} }
if (rc >= 0) { if (rc >= 2) {
if (!buf[0] && !buf[1])
usb_try_string_workarounds(buf, &rc);
/* There might be extra junk at the end of the descriptor */ /* There might be extra junk at the end of the descriptor */
if (buf[0] < rc) if (buf[0] < rc)
rc = buf[0]; rc = buf[0];
if (rc < 2)
rc = -EINVAL; rc = rc - (rc & 1); /* force a multiple of two */
} }
if (rc < 2)
rc = (rc < 0 ? rc : -EINVAL);
return rc; return rc;
} }
...@@ -724,6 +747,9 @@ int usb_string(struct usb_device *dev, int index, char *buf, size_t size) ...@@ -724,6 +747,9 @@ int usb_string(struct usb_device *dev, int index, char *buf, size_t size)
buf[idx] = 0; buf[idx] = 0;
err = idx; err = idx;
if (tbuf[1] != USB_DT_STRING)
dev_dbg(&dev->dev, "wrong descriptor type %02x for string %d (\"%s\")\n", tbuf[1], index, buf);
errout: errout:
kfree(tbuf); kfree(tbuf);
return err; return err;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment