bcm203x.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. /*
  2. *
  3. * Broadcom Blutonium firmware driver
  4. *
  5. * Copyright (C) 2003 Maxim Krasnyansky <maxk@qualcomm.com>
  6. * Copyright (C) 2003 Marcel Holtmann <marcel@holtmann.org>
  7. *
  8. *
  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.
  13. *
  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.
  18. *
  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
  22. *
  23. */
  24. #include <linux/module.h>
  25. #include <linux/kernel.h>
  26. #include <linux/init.h>
  27. #include <linux/slab.h>
  28. #include <linux/types.h>
  29. #include <linux/errno.h>
  30. #include <linux/device.h>
  31. #include <linux/firmware.h>
  32. #include <linux/usb.h>
  33. #include <net/bluetooth/bluetooth.h>
  34. #define VERSION "1.2"
  35. static const struct usb_device_id bcm203x_table[] = {
  36. /* Broadcom Blutonium (BCM2033) */
  37. { USB_DEVICE(0x0a5c, 0x2033) },
  38. { } /* Terminating entry */
  39. };
  40. MODULE_DEVICE_TABLE(usb, bcm203x_table);
  41. #define BCM203X_ERROR 0
  42. #define BCM203X_RESET 1
  43. #define BCM203X_LOAD_MINIDRV 2
  44. #define BCM203X_SELECT_MEMORY 3
  45. #define BCM203X_CHECK_MEMORY 4
  46. #define BCM203X_LOAD_FIRMWARE 5
  47. #define BCM203X_CHECK_FIRMWARE 6
  48. #define BCM203X_IN_EP 0x81
  49. #define BCM203X_OUT_EP 0x02
  50. struct bcm203x_data {
  51. struct usb_device *udev;
  52. unsigned long state;
  53. struct work_struct work;
  54. struct urb *urb;
  55. unsigned char *buffer;
  56. unsigned char *fw_data;
  57. unsigned int fw_size;
  58. unsigned int fw_sent;
  59. };
  60. static void bcm203x_complete(struct urb *urb)
  61. {
  62. struct bcm203x_data *data = urb->context;
  63. struct usb_device *udev = urb->dev;
  64. int len;
  65. BT_DBG("udev %p urb %p", udev, urb);
  66. if (urb->status) {
  67. BT_ERR("URB failed with status %d", urb->status);
  68. data->state = BCM203X_ERROR;
  69. return;
  70. }
  71. switch (data->state) {
  72. case BCM203X_LOAD_MINIDRV:
  73. memcpy(data->buffer, "#", 1);
  74. usb_fill_bulk_urb(urb, udev, usb_sndbulkpipe(udev, BCM203X_OUT_EP),
  75. data->buffer, 1, bcm203x_complete, data);
  76. data->state = BCM203X_SELECT_MEMORY;
  77. schedule_work(&data->work);
  78. break;
  79. case BCM203X_SELECT_MEMORY:
  80. usb_fill_int_urb(urb, udev, usb_rcvintpipe(udev, BCM203X_IN_EP),
  81. data->buffer, 32, bcm203x_complete, data, 1);
  82. data->state = BCM203X_CHECK_MEMORY;
  83. if (usb_submit_urb(data->urb, GFP_ATOMIC) < 0)
  84. BT_ERR("Can't submit URB");
  85. break;
  86. case BCM203X_CHECK_MEMORY:
  87. if (data->buffer[0] != '#') {
  88. BT_ERR("Memory select failed");
  89. data->state = BCM203X_ERROR;
  90. break;
  91. }
  92. data->state = BCM203X_LOAD_FIRMWARE;
  93. case BCM203X_LOAD_FIRMWARE:
  94. if (data->fw_sent == data->fw_size) {
  95. usb_fill_int_urb(urb, udev, usb_rcvintpipe(udev, BCM203X_IN_EP),
  96. data->buffer, 32, bcm203x_complete, data, 1);
  97. data->state = BCM203X_CHECK_FIRMWARE;
  98. } else {
  99. len = min_t(uint, data->fw_size - data->fw_sent, 4096);
  100. usb_fill_bulk_urb(urb, udev, usb_sndbulkpipe(udev, BCM203X_OUT_EP),
  101. data->fw_data + data->fw_sent, len, bcm203x_complete, data);
  102. data->fw_sent += len;
  103. }
  104. if (usb_submit_urb(data->urb, GFP_ATOMIC) < 0)
  105. BT_ERR("Can't submit URB");
  106. break;
  107. case BCM203X_CHECK_FIRMWARE:
  108. if (data->buffer[0] != '.') {
  109. BT_ERR("Firmware loading failed");
  110. data->state = BCM203X_ERROR;
  111. break;
  112. }
  113. data->state = BCM203X_RESET;
  114. break;
  115. }
  116. }
  117. static void bcm203x_work(struct work_struct *work)
  118. {
  119. struct bcm203x_data *data =
  120. container_of(work, struct bcm203x_data, work);
  121. if (usb_submit_urb(data->urb, GFP_ATOMIC) < 0)
  122. BT_ERR("Can't submit URB");
  123. }
  124. static int bcm203x_probe(struct usb_interface *intf, const struct usb_device_id *id)
  125. {
  126. const struct firmware *firmware;
  127. struct usb_device *udev = interface_to_usbdev(intf);
  128. struct bcm203x_data *data;
  129. int size;
  130. BT_DBG("intf %p id %p", intf, id);
  131. if (intf->cur_altsetting->desc.bInterfaceNumber != 0)
  132. return -ENODEV;
  133. data = kzalloc(sizeof(*data), GFP_KERNEL);
  134. if (!data) {
  135. BT_ERR("Can't allocate memory for data structure");
  136. return -ENOMEM;
  137. }
  138. data->udev = udev;
  139. data->state = BCM203X_LOAD_MINIDRV;
  140. data->urb = usb_alloc_urb(0, GFP_KERNEL);
  141. if (!data->urb) {
  142. BT_ERR("Can't allocate URB");
  143. kfree(data);
  144. return -ENOMEM;
  145. }
  146. if (request_firmware(&firmware, "BCM2033-MD.hex", &udev->dev) < 0) {
  147. BT_ERR("Mini driver request failed");
  148. usb_free_urb(data->urb);
  149. kfree(data);
  150. return -EIO;
  151. }
  152. BT_DBG("minidrv data %p size %zu", firmware->data, firmware->size);
  153. size = max_t(uint, firmware->size, 4096);
  154. data->buffer = kmalloc(size, GFP_KERNEL);
  155. if (!data->buffer) {
  156. BT_ERR("Can't allocate memory for mini driver");
  157. release_firmware(firmware);
  158. usb_free_urb(data->urb);
  159. kfree(data);
  160. return -ENOMEM;
  161. }
  162. memcpy(data->buffer, firmware->data, firmware->size);
  163. usb_fill_bulk_urb(data->urb, udev, usb_sndbulkpipe(udev, BCM203X_OUT_EP),
  164. data->buffer, firmware->size, bcm203x_complete, data);
  165. release_firmware(firmware);
  166. if (request_firmware(&firmware, "BCM2033-FW.bin", &udev->dev) < 0) {
  167. BT_ERR("Firmware request failed");
  168. usb_free_urb(data->urb);
  169. kfree(data->buffer);
  170. kfree(data);
  171. return -EIO;
  172. }
  173. BT_DBG("firmware data %p size %zu", firmware->data, firmware->size);
  174. data->fw_data = kmemdup(firmware->data, firmware->size, GFP_KERNEL);
  175. if (!data->fw_data) {
  176. BT_ERR("Can't allocate memory for firmware image");
  177. release_firmware(firmware);
  178. usb_free_urb(data->urb);
  179. kfree(data->buffer);
  180. kfree(data);
  181. return -ENOMEM;
  182. }
  183. data->fw_size = firmware->size;
  184. data->fw_sent = 0;
  185. release_firmware(firmware);
  186. INIT_WORK(&data->work, bcm203x_work);
  187. usb_set_intfdata(intf, data);
  188. schedule_work(&data->work);
  189. return 0;
  190. }
  191. static void bcm203x_disconnect(struct usb_interface *intf)
  192. {
  193. struct bcm203x_data *data = usb_get_intfdata(intf);
  194. BT_DBG("intf %p", intf);
  195. usb_kill_urb(data->urb);
  196. usb_set_intfdata(intf, NULL);
  197. usb_free_urb(data->urb);
  198. kfree(data->fw_data);
  199. kfree(data->buffer);
  200. kfree(data);
  201. }
  202. static struct usb_driver bcm203x_driver = {
  203. .name = "bcm203x",
  204. .probe = bcm203x_probe,
  205. .disconnect = bcm203x_disconnect,
  206. .id_table = bcm203x_table,
  207. };
  208. static int __init bcm203x_init(void)
  209. {
  210. int err;
  211. BT_INFO("Broadcom Blutonium firmware driver ver %s", VERSION);
  212. err = usb_register(&bcm203x_driver);
  213. if (err < 0)
  214. BT_ERR("Failed to register USB driver");
  215. return err;
  216. }
  217. static void __exit bcm203x_exit(void)
  218. {
  219. usb_deregister(&bcm203x_driver);
  220. }
  221. module_init(bcm203x_init);
  222. module_exit(bcm203x_exit);
  223. MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>");
  224. MODULE_DESCRIPTION("Broadcom Blutonium firmware driver ver " VERSION);
  225. MODULE_VERSION(VERSION);
  226. MODULE_LICENSE("GPL");
  227. MODULE_FIRMWARE("BCM2033-MD.hex");
  228. MODULE_FIRMWARE("BCM2033-FW.bin");