1 Use USB_DEVICE_AND_INTERFACE_INFO to do device matching with PID/VID 0x3006/0x0cf3. 2 Delete desc.bInterfaceNumber != 0 check. 3 Download firmware to hardware for PID=0x3006 Signed-off-by: Costa Yao <cqyao@xxxxxxxxxxxxxxxx> --- drivers/bluetooth/ath3k.c | 69 +++++++++++++++++++++++++++++++++++++++++---- 1 files changed, 63 insertions(+), 6 deletions(-) diff --git a/drivers/bluetooth/ath3k.c b/drivers/bluetooth/ath3k.c index 1622772..77f040d 100644 --- a/drivers/bluetooth/ath3k.c +++ b/drivers/bluetooth/ath3k.c @@ -40,6 +40,7 @@ #define ATH3K_MODE_MASK 0x3F #define ATH3K_NORMAL_MODE 0x0E +#define ATH3K_PREBOOT_MODE 0x0D #define ATH3K_PATCH_UPDATE 0x80 #define ATH3K_SYSCFG_UPDATE 0x40 @@ -49,6 +50,9 @@ #define ATH3K_XTAL_FREQ_19P2 0x02 #define ATH3K_NAME_LEN 0xFF +#define ATH3K_3012 0x80 +#define ATH3K_3006 0x0100 + struct ath3k_version { unsigned int rom_version; unsigned int build_version; @@ -71,6 +75,7 @@ static struct usb_device_id ath3k_table[] = { /* Atheros AR3012 with sflash firmware*/ { USB_DEVICE(0x0CF3, 0x3004) }, + { USB_DEVICE_AND_INTERFACE_INFO(0x0CF3, 0x3006, 0xe0, 0x01, 0x01) }, /* Atheros AR5BBU12 with sflash firmware */ { USB_DEVICE(0x0489, 0xE02C) }, @@ -80,13 +85,13 @@ static struct usb_device_id ath3k_table[] = { MODULE_DEVICE_TABLE(usb, ath3k_table); -#define BTUSB_ATH3012 0x80 /* This table is to load patch and sysconfig files * for AR3012 */ static struct usb_device_id ath3k_blist_tbl[] = { /* Atheros AR3012 with sflash firmware*/ - { USB_DEVICE(0x0cf3, 0x3004), .driver_info = BTUSB_ATH3012 }, + { USB_DEVICE(0x0cf3, 0x3004), .driver_info = ATH3K_3012 }, + { USB_DEVICE(0x0CF3, 0x3006), .driver_info = ATH3K_3006 }, { } /* Terminating entry */ }; @@ -355,6 +360,52 @@ static int ath3k_load_syscfg(struct usb_device *udev) return ret; } +static int ath3k_download_3006(struct usb_device *udev) +{ + int ret; + unsigned char fw_state; + + if (le16_to_cpu(udev->descriptor.bcdDevice) > 0x0001) { + BT_ERR("ath3k_probe: udev->descriptor.bcdDevice"); + return -ENODEV; + } + + ret = ath3k_get_state(udev, &fw_state); + if (ret < 0) { + BT_ERR("ath3k_probe: ath3k_get_state err"); + return ret; + } + + if ((fw_state & ATH3K_MODE_MASK) == ATH3K_PREBOOT_MODE) { + BT_ERR("ath3k_probe: firmware are in preboot mode now"); + BT_ERR("ath3k_probe: try to switch to Normal Mode"); + ret = ath3k_set_normal_mode(udev); + if (ret < 0) { + BT_ERR("ath3k_probe: set normal mode failed"); + return ret; + } + } + + /* Note we should wait for a while */ + mdelay(100); + ret = ath3k_load_patch(udev); + if (ret < 0) { + BT_ERR("ath3k_probe: Load patch file failed"); + return ret; + } + ret = ath3k_load_syscfg(udev); + if (ret < 0) { + BT_ERR("ath3k_probe: Load sysconfig file failed"); + return ret; + } + ret = ath3k_switch_pid(udev); + if (ret < 0) { + BT_ERR("ath3k_probe: switch pid failed"); + return ret; + } + return 0; +} + static int ath3k_probe(struct usb_interface *intf, const struct usb_device_id *id) { @@ -364,9 +415,6 @@ static int ath3k_probe(struct usb_interface *intf, BT_DBG("intf %p id %p", intf, id); - if (intf->cur_altsetting->desc.bInterfaceNumber != 0) - return -ENODEV; - /* match device ID in ath3k blacklist table */ if (!id->driver_info) { const struct usb_device_id *match; @@ -376,7 +424,7 @@ static int ath3k_probe(struct usb_interface *intf, } /* load patch and sysconfig files for AR3012 */ - if (id->driver_info & BTUSB_ATH3012) { + if (id->driver_info & ATH3K_3012) { /* New firmware with patch and sysconfig files already loaded */ if (le16_to_cpu(udev->descriptor.bcdDevice) > 0x0001) @@ -401,6 +449,15 @@ static int ath3k_probe(struct usb_interface *intf, return 0; } + if (id->driver_info & ATH3K_3006) { + ret = ath3k_download_3006(udev); + if (ret < 0) { + BT_ERR("ath3k_probe: download_3006 failed"); + return ret; + } + return 0; + } + ret = request_firmware(&firmware, ATH3K_FIRMWARE, &udev->dev); if (ret < 0) { if (ret == -ENOENT) -- 1.7.4.1 > -----Original Message----- > From: linux-bluetooth-owner@xxxxxxxxxxxxxxx > [mailto:linux-bluetooth-owner@xxxxxxxxxxxxxxx] On Behalf Of Marcel Holtmann > Sent: 2011年11月18日 20:07 > To: Yao, Costa > Cc: padovan@xxxxxxxxxxxxxx; linux-bluetooth@xxxxxxxxxxxxxxx > Subject: Re: [PATCH] Bluetooth: ath3k: Add supports for Qualcomm Atherose > 3012 composite chip > > Hi Costa, > > > 1 Use USB_DEVICE_AND_INTERFACE_INFO to do device matching with > PID/VID 0x3006/0x0cf3. > > 2 Delete desc.bInterfaceNumber != 0 check. > > 3 Add function ath3k_download_3006 to download firmware to hardware > > for 0x3006 > > > > Signed-off-by: Costa Yao <cqyao@xxxxxxxxxxxxxxxx> > > --- > > drivers/bluetooth/ath3k.c | 65 > ++++++++++++++++++++++++++++++++++++++++++--- > > 1 files changed, 61 insertions(+), 4 deletions(-) > > > > diff --git a/drivers/bluetooth/ath3k.c b/drivers/bluetooth/ath3k.c > > index 1622772..b38cd16 100644 > > --- a/drivers/bluetooth/ath3k.c > > +++ b/drivers/bluetooth/ath3k.c > > @@ -40,6 +40,7 @@ > > > > #define ATH3K_MODE_MASK 0x3F > > #define ATH3K_NORMAL_MODE 0x0E > > +#define ATH3K_PREBOOT_MODE 0x0D > > > > #define ATH3K_PATCH_UPDATE 0x80 > > #define ATH3K_SYSCFG_UPDATE 0x40 > > @@ -49,6 +50,9 @@ > > #define ATH3K_XTAL_FREQ_19P2 0x02 > > #define ATH3K_NAME_LEN 0xFF > > > > +#define BTUSB_ATH3012 0x80 > > +#define BTUSB_ATH3006 0x0100 > > + > > this is ath3k.c specific. So do not use BTUSB_ prefux. > > > struct ath3k_version { > > unsigned int rom_version; > > unsigned int build_version; > > @@ -71,6 +75,7 @@ static struct usb_device_id ath3k_table[] = { > > > > /* Atheros AR3012 with sflash firmware*/ > > { USB_DEVICE(0x0CF3, 0x3004) }, > > + { USB_DEVICE_AND_INTERFACE_INFO(0x0CF3, 0x3006, 0xe0, 0x01, > 0x01) }, > > > > /* Atheros AR5BBU12 with sflash firmware */ > > { USB_DEVICE(0x0489, 0xE02C) }, > > @@ -80,13 +85,13 @@ static struct usb_device_id ath3k_table[] = { > > > > MODULE_DEVICE_TABLE(usb, ath3k_table); > > > > -#define BTUSB_ATH3012 0x80 > > /* This table is to load patch and sysconfig files > > * for AR3012 */ > > static struct usb_device_id ath3k_blist_tbl[] = { > > > > /* Atheros AR3012 with sflash firmware*/ > > { USB_DEVICE(0x0cf3, 0x3004), .driver_info = BTUSB_ATH3012 }, > > + { USB_DEVICE(0x0CF3, 0x3006), .driver_info = BTUSB_ATH3006 }, > > > > { } /* Terminating entry */ > > }; > > @@ -355,6 +360,52 @@ static int ath3k_load_syscfg(struct usb_device > *udev) > > return ret; > > } > > > > +static int ath3k_download_3006(struct usb_device *udev) { > > + int ret; > > + unsigned char fw_state; > > + > > + if (le16_to_cpu(udev->descriptor.bcdDevice) > 0x0001) { > > + BT_ERR("ath3k_probe: udev->descriptor.bcdDevice"); > > + return -ENODEV; > > + } > > + > > + ret = ath3k_get_state(udev, &fw_state); > > + if (ret < 0) { > > + BT_ERR("ath3k_probe: ath3k_get_state err"); > > + return ret; > > + } > > + > > + if ((fw_state & ATH3K_MODE_MASK) == ATH3K_PREBOOT_MODE) { > > + BT_ERR("ath3k_probe: firmware are in preboot mode now"); > > + BT_ERR("ath3k_probe: try to switch to Normal Mode"); > > + ret = ath3k_set_normal_mode(udev); > > + if (ret < 0) { > > + BT_ERR("ath3k_probe: set normal mode failed"); > > + return ret; > > + } > > + } > > + > > + /* Note we should wait for a while */ > > + mdelay(100); > > + ret = ath3k_load_patch(udev); > > + if (ret < 0) { > > + BT_ERR("ath3k_probe: Load patch file failed"); > > + return ret; > > + } > > + ret = ath3k_load_syscfg(udev); > > + if (ret < 0) { > > + BT_ERR("ath3k_probe: Load sysconfig file failed"); > > + return ret; > > + } > > + ret = ath3k_switch_pid(udev); > > + if (ret < 0) { > > + BT_ERR("ath3k_probe: switch pid failed"); > > + return ret; > > + } > > + return 0; > > +} > > + > > static int ath3k_probe(struct usb_interface *intf, > > const struct usb_device_id *id) > > { > > @@ -364,9 +415,6 @@ static int ath3k_probe(struct usb_interface *intf, > > > > BT_DBG("intf %p id %p", intf, id); > > > > - if (intf->cur_altsetting->desc.bInterfaceNumber != 0) > > - return -ENODEV; > > - > > /* match device ID in ath3k blacklist table */ > > if (!id->driver_info) { > > const struct usb_device_id *match; > > @@ -401,6 +449,15 @@ static int ath3k_probe(struct usb_interface *intf, > > return 0; > > } > > > > + if (id->driver_info & BTUSB_ATH3006) { > > + ret = ath3k_download_3006(udev); > > + if (ret < 0) { > > + BT_ERR("ath3k_probe: download_3006 failed"); > > + return ret; > > + } > > + return 0; > > + } > > + > > ret = request_firmware(&firmware, ATH3K_FIRMWARE, &udev->dev); > > if (ret < 0) { > > if (ret == -ENOENT) > > The rest is fine with me. > > Regards > > Marcel > > > -- > To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in the > body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at > http://vger.kernel.org/majordomo-info.html ��.n��������+%������w��{.n�����{����^n�r������&��z�ޗ�zf���h���~����������_��+v���)ߣ�