3  *  Broadcom Blutonium firmware driver
 
   5  *  Copyright (C) 2003  Maxim Krasnyansky <maxk@qualcomm.com>
 
   6  *  Copyright (C) 2003  Marcel Holtmann <marcel@holtmann.org>
 
   9  *  This program is free software; you can redistribute it and/or modify
 
  10  *  it under the terms of the GNU General Public License as published by
 
  11  *  the Free Software Foundation; either version 2 of the License, or
 
  12  *  (at your option) any later version.
 
  14  *  This program is distributed in the hope that it will be useful,
 
  15  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  16  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
  17  *  GNU General Public License for more details.
 
  19  *  You should have received a copy of the GNU General Public License
 
  20  *  along with this program; if not, write to the Free Software
 
  21  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
  25 #include <linux/config.h>
 
  26 #include <linux/module.h>
 
  28 #include <linux/kernel.h>
 
  29 #include <linux/init.h>
 
  30 #include <linux/slab.h>
 
  31 #include <linux/types.h>
 
  32 #include <linux/errno.h>
 
  33 #include <linux/timer.h>
 
  35 #include <linux/device.h>
 
  36 #include <linux/firmware.h>
 
  38 #include <linux/usb.h>
 
  40 #include <net/bluetooth/bluetooth.h>
 
  42 #ifndef CONFIG_BT_HCIBCM203X_DEBUG
 
  49 static int ignore = 0;
 
  51 static struct usb_device_id bcm203x_table[] = {
 
  52         /* Broadcom Blutonium (BCM2033) */
 
  53         { USB_DEVICE(0x0a5c, 0x2033) },
 
  55         { }     /* Terminating entry */
 
  58 MODULE_DEVICE_TABLE(usb, bcm203x_table);
 
  60 #define BCM203X_ERROR           0
 
  61 #define BCM203X_RESET           1
 
  62 #define BCM203X_LOAD_MINIDRV    2
 
  63 #define BCM203X_SELECT_MEMORY   3
 
  64 #define BCM203X_CHECK_MEMORY    4
 
  65 #define BCM203X_LOAD_FIRMWARE   5
 
  66 #define BCM203X_CHECK_FIRMWARE  6
 
  68 #define BCM203X_IN_EP           0x81
 
  69 #define BCM203X_OUT_EP          0x02
 
  72         struct usb_device       *udev;
 
  76         struct timer_list       timer;
 
  79         unsigned char           *buffer;
 
  81         unsigned char           *fw_data;
 
  86 static void bcm203x_complete(struct urb *urb, struct pt_regs *regs)
 
  88         struct bcm203x_data *data = urb->context;
 
  89         struct usb_device *udev = urb->dev;
 
  92         BT_DBG("udev %p urb %p", udev, urb);
 
  95                 BT_ERR("URB failed with status %d", urb->status);
 
  96                 data->state = BCM203X_ERROR;
 
 100         switch (data->state) {
 
 101         case BCM203X_LOAD_MINIDRV:
 
 102                 memcpy(data->buffer, "#", 1);
 
 104                 usb_fill_bulk_urb(urb, udev, usb_sndbulkpipe(udev, BCM203X_OUT_EP),
 
 105                                 data->buffer, 1, bcm203x_complete, data);
 
 107                 data->state = BCM203X_SELECT_MEMORY;
 
 109                 mod_timer(&data->timer, jiffies + (HZ / 10));
 
 112         case BCM203X_SELECT_MEMORY:
 
 113                 usb_fill_int_urb(urb, udev, usb_rcvintpipe(udev, BCM203X_IN_EP),
 
 114                                 data->buffer, 32, bcm203x_complete, data, 1);
 
 116                 data->state = BCM203X_CHECK_MEMORY;
 
 118                 if (usb_submit_urb(data->urb, GFP_ATOMIC) < 0)
 
 119                         BT_ERR("Can't submit URB");
 
 122         case BCM203X_CHECK_MEMORY:
 
 123                 if (data->buffer[0] != '#') {
 
 124                         BT_ERR("Memory select failed");
 
 125                         data->state = BCM203X_ERROR;
 
 129                 data->state = BCM203X_LOAD_FIRMWARE;
 
 131         case BCM203X_LOAD_FIRMWARE:
 
 132                 if (data->fw_sent == data->fw_size) {
 
 133                         usb_fill_int_urb(urb, udev, usb_rcvintpipe(udev, BCM203X_IN_EP),
 
 134                                 data->buffer, 32, bcm203x_complete, data, 1);
 
 136                         data->state = BCM203X_CHECK_FIRMWARE;
 
 138                         len = min_t(uint, data->fw_size - data->fw_sent, 4096);
 
 140                         usb_fill_bulk_urb(urb, udev, usb_sndbulkpipe(udev, BCM203X_OUT_EP),
 
 141                                 data->fw_data + data->fw_sent, len, bcm203x_complete, data);
 
 143                         data->fw_sent += len;
 
 146                 if (usb_submit_urb(data->urb, GFP_ATOMIC) < 0)
 
 147                         BT_ERR("Can't submit URB");
 
 150         case BCM203X_CHECK_FIRMWARE:
 
 151                 if (data->buffer[0] != '.') {
 
 152                         BT_ERR("Firmware loading failed");
 
 153                         data->state = BCM203X_ERROR;
 
 157                 data->state = BCM203X_RESET;
 
 162 static void bcm203x_timer(unsigned long user_data)
 
 164         struct bcm203x_data *data = (struct bcm203x_data *) user_data;
 
 166         if (usb_submit_urb(data->urb, GFP_ATOMIC) < 0)
 
 167                 BT_ERR("Can't submit URB");
 
 170 static int bcm203x_probe(struct usb_interface *intf, const struct usb_device_id *id)
 
 172         const struct firmware *firmware;
 
 173         struct usb_device *udev = interface_to_usbdev(intf);
 
 174         struct bcm203x_data *data;
 
 177         BT_DBG("intf %p id %p", intf, id);
 
 179         if (ignore || (intf->cur_altsetting->desc.bInterfaceNumber != 0))
 
 182         data = kzalloc(sizeof(*data), GFP_KERNEL);
 
 184                 BT_ERR("Can't allocate memory for data structure");
 
 189         data->state = BCM203X_LOAD_MINIDRV;
 
 191         data->urb = usb_alloc_urb(0, GFP_KERNEL);
 
 193                 BT_ERR("Can't allocate URB");
 
 198         if (request_firmware(&firmware, "BCM2033-MD.hex", &udev->dev) < 0) {
 
 199                 BT_ERR("Mini driver request failed");
 
 200                 usb_free_urb(data->urb);
 
 205         BT_DBG("minidrv data %p size %d", firmware->data, firmware->size);
 
 207         size = max_t(uint, firmware->size, 4096);
 
 209         data->buffer = kmalloc(size, GFP_KERNEL);
 
 211                 BT_ERR("Can't allocate memory for mini driver");
 
 212                 release_firmware(firmware);
 
 213                 usb_free_urb(data->urb);
 
 218         memcpy(data->buffer, firmware->data, firmware->size);
 
 220         usb_fill_bulk_urb(data->urb, udev, usb_sndbulkpipe(udev, BCM203X_OUT_EP),
 
 221                         data->buffer, firmware->size, bcm203x_complete, data);
 
 223         release_firmware(firmware);
 
 225         if (request_firmware(&firmware, "BCM2033-FW.bin", &udev->dev) < 0) {
 
 226                 BT_ERR("Firmware request failed");
 
 227                 usb_free_urb(data->urb);
 
 233         BT_DBG("firmware data %p size %d", firmware->data, firmware->size);
 
 235         data->fw_data = kmalloc(firmware->size, GFP_KERNEL);
 
 236         if (!data->fw_data) {
 
 237                 BT_ERR("Can't allocate memory for firmware image");
 
 238                 usb_free_urb(data->urb);
 
 244         memcpy(data->fw_data, firmware->data, firmware->size);
 
 245         data->fw_size = firmware->size;
 
 248         release_firmware(firmware);
 
 250         init_timer(&data->timer);
 
 251         data->timer.function = bcm203x_timer;
 
 252         data->timer.data = (unsigned long) data;
 
 254         usb_set_intfdata(intf, data);
 
 256         mod_timer(&data->timer, jiffies + HZ);
 
 261 static void bcm203x_disconnect(struct usb_interface *intf)
 
 263         struct bcm203x_data *data = usb_get_intfdata(intf);
 
 265         BT_DBG("intf %p", intf);
 
 267         usb_kill_urb(data->urb);
 
 269         usb_set_intfdata(intf, NULL);
 
 271         usb_free_urb(data->urb);
 
 272         kfree(data->fw_data);
 
 277 static struct usb_driver bcm203x_driver = {
 
 279         .probe          = bcm203x_probe,
 
 280         .disconnect     = bcm203x_disconnect,
 
 281         .id_table       = bcm203x_table,
 
 284 static int __init bcm203x_init(void)
 
 288         BT_INFO("Broadcom Blutonium firmware driver ver %s", VERSION);
 
 290         err = usb_register(&bcm203x_driver);
 
 292                 BT_ERR("Failed to register USB driver");
 
 297 static void __exit bcm203x_exit(void)
 
 299         usb_deregister(&bcm203x_driver);
 
 302 module_init(bcm203x_init);
 
 303 module_exit(bcm203x_exit);
 
 305 module_param(ignore, bool, 0644);
 
 306 MODULE_PARM_DESC(ignore, "Ignore devices from the matching table");
 
 308 MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>");
 
 309 MODULE_DESCRIPTION("Broadcom Blutonium firmware driver ver " VERSION);
 
 310 MODULE_VERSION(VERSION);
 
 311 MODULE_LICENSE("GPL");