Commit e07896e6 authored by Anssi Hannula's avatar Anssi Hannula Committed by Greg Kroah-Hartman

USB: qcserial: Add support for Qualcomm Gobi 2000 devices

Add ids for Qualcomm Gobi 2000 QDL and Modem modes. Gobi 2000 has a
single altsetting in QDL mode, so adapt code to handle that.

Firmware upload protocol is also slightly different, with an
additional firmware file. However, qcserial doesn't handle firmware
uploading.

Tested on Lenovo Thinkpad T510.
Signed-off-by: default avatarAnssi Hannula <anssi.hannula@gmail.com>
Signed-off-by: default avatarMatthew Garrett <mjg@redhat.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 3d7e59ad
...@@ -78,6 +78,8 @@ static const struct usb_device_id id_table[] = { ...@@ -78,6 +78,8 @@ static const struct usb_device_id id_table[] = {
{USB_DEVICE(0x1199, 0x900a)}, /* Sierra Wireless Gobi 2000 Modem device (VT773) */ {USB_DEVICE(0x1199, 0x900a)}, /* Sierra Wireless Gobi 2000 Modem device (VT773) */
{USB_DEVICE(0x16d8, 0x8001)}, /* CMDTech Gobi 2000 QDL device (VU922) */ {USB_DEVICE(0x16d8, 0x8001)}, /* CMDTech Gobi 2000 QDL device (VU922) */
{USB_DEVICE(0x16d8, 0x8002)}, /* CMDTech Gobi 2000 Modem device (VU922) */ {USB_DEVICE(0x16d8, 0x8002)}, /* CMDTech Gobi 2000 Modem device (VU922) */
{USB_DEVICE(0x05c6, 0x9204)}, /* Gobi 2000 QDL device */
{USB_DEVICE(0x05c6, 0x9205)}, /* Gobi 2000 Modem device */
{ } /* Terminating entry */ { } /* Terminating entry */
}; };
MODULE_DEVICE_TABLE(usb, id_table); MODULE_DEVICE_TABLE(usb, id_table);
...@@ -95,6 +97,7 @@ static struct usb_driver qcdriver = { ...@@ -95,6 +97,7 @@ static struct usb_driver qcdriver = {
static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id) static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id)
{ {
struct usb_wwan_intf_private *data; struct usb_wwan_intf_private *data;
struct usb_host_interface *intf = serial->interface->cur_altsetting;
int retval = -ENODEV; int retval = -ENODEV;
__u8 nintf; __u8 nintf;
__u8 ifnum; __u8 ifnum;
...@@ -103,7 +106,7 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id) ...@@ -103,7 +106,7 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id)
nintf = serial->dev->actconfig->desc.bNumInterfaces; nintf = serial->dev->actconfig->desc.bNumInterfaces;
dbg("Num Interfaces = %d", nintf); dbg("Num Interfaces = %d", nintf);
ifnum = serial->interface->cur_altsetting->desc.bInterfaceNumber; ifnum = intf->desc.bInterfaceNumber;
dbg("This Interface = %d", ifnum); dbg("This Interface = %d", ifnum);
data = serial->private = kzalloc(sizeof(struct usb_wwan_intf_private), data = serial->private = kzalloc(sizeof(struct usb_wwan_intf_private),
...@@ -116,27 +119,32 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id) ...@@ -116,27 +119,32 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id)
switch (nintf) { switch (nintf) {
case 1: case 1:
/* QDL mode */ /* QDL mode */
if (serial->interface->num_altsetting == 2) { /* Gobi 2000 has a single altsetting, older ones have two */
struct usb_host_interface *intf; if (serial->interface->num_altsetting == 2)
intf = &serial->interface->altsetting[1]; intf = &serial->interface->altsetting[1];
if (intf->desc.bNumEndpoints == 2) { else if (serial->interface->num_altsetting > 2)
if (usb_endpoint_is_bulk_in(&intf->endpoint[0].desc) && break;
usb_endpoint_is_bulk_out(&intf->endpoint[1].desc)) {
dbg("QDL port found"); if (intf->desc.bNumEndpoints == 2 &&
retval = usb_set_interface(serial->dev, ifnum, 1); usb_endpoint_is_bulk_in(&intf->endpoint[0].desc) &&
if (retval < 0) { usb_endpoint_is_bulk_out(&intf->endpoint[1].desc)) {
dev_err(&serial->dev->dev, dbg("QDL port found");
"Could not set interface, error %d\n",
retval); if (serial->interface->num_altsetting == 1)
retval = -ENODEV; return 0;
}
return retval; retval = usb_set_interface(serial->dev, ifnum, 1);
} if (retval < 0) {
dev_err(&serial->dev->dev,
"Could not set interface, error %d\n",
retval);
retval = -ENODEV;
} }
return retval;
} }
break; break;
case 3:
case 4: case 4:
/* Composite mode */ /* Composite mode */
if (ifnum == 2) { if (ifnum == 2) {
......
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