RE: [PATCH] Bluetooth: ath3k: Add supports for Qualcomm Atheros 3012 composite chip

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



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���)ߣ�

[Index of Archives]     [Bluez Devel]     [Linux Wireless Networking]     [Linux Wireless Personal Area Networking]     [Linux ATH6KL]     [Linux USB Devel]     [Linux Media Drivers]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Big List of Linux Books]

  Powered by Linux