smsusb.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656
  1. /****************************************************************
  2. Siano Mobile Silicon, Inc.
  3. MDTV receiver kernel modules.
  4. Copyright (C) 2005-2009, Uri Shkolnik, Anatoly Greenblat
  5. This program is free software: you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation, either version 2 of the License, or
  8. (at your option) any later version.
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. ****************************************************************/
  16. #include <linux/kernel.h>
  17. #include <linux/init.h>
  18. #include <linux/usb.h>
  19. #include <linux/firmware.h>
  20. #include <linux/slab.h>
  21. #include <linux/module.h>
  22. #include "smscoreapi.h"
  23. #include "sms-cards.h"
  24. #include "smsendian.h"
  25. static int sms_dbg;
  26. module_param_named(debug, sms_dbg, int, 0644);
  27. MODULE_PARM_DESC(debug, "set debug level (info=1, adv=2 (or-able))");
  28. #define USB1_BUFFER_SIZE 0x1000
  29. #define USB2_BUFFER_SIZE 0x2000
  30. #define MAX_BUFFERS 50
  31. #define MAX_URBS 10
  32. struct smsusb_device_t;
  33. enum smsusb_state {
  34. SMSUSB_DISCONNECTED,
  35. SMSUSB_SUSPENDED,
  36. SMSUSB_ACTIVE
  37. };
  38. struct smsusb_urb_t {
  39. struct list_head entry;
  40. struct smscore_buffer_t *cb;
  41. struct smsusb_device_t *dev;
  42. struct urb urb;
  43. };
  44. struct smsusb_device_t {
  45. struct usb_device *udev;
  46. struct smscore_device_t *coredev;
  47. struct smsusb_urb_t surbs[MAX_URBS];
  48. int response_alignment;
  49. int buffer_size;
  50. unsigned char in_ep;
  51. unsigned char out_ep;
  52. enum smsusb_state state;
  53. };
  54. static int smsusb_submit_urb(struct smsusb_device_t *dev,
  55. struct smsusb_urb_t *surb);
  56. /**
  57. * Completing URB's callback handler - top half (interrupt context)
  58. * adds completing sms urb to the global surbs list and activtes the worker
  59. * thread the surb
  60. * IMPORTANT - blocking functions must not be called from here !!!
  61. * @param urb pointer to a completing urb object
  62. */
  63. static void smsusb_onresponse(struct urb *urb)
  64. {
  65. struct smsusb_urb_t *surb = (struct smsusb_urb_t *) urb->context;
  66. struct smsusb_device_t *dev = surb->dev;
  67. if (urb->status == -ESHUTDOWN) {
  68. sms_err("error, urb status %d (-ESHUTDOWN), %d bytes",
  69. urb->status, urb->actual_length);
  70. return;
  71. }
  72. if ((urb->actual_length > 0) && (urb->status == 0)) {
  73. struct sms_msg_hdr *phdr = (struct sms_msg_hdr *)surb->cb->p;
  74. smsendian_handle_message_header(phdr);
  75. if (urb->actual_length >= phdr->msg_length) {
  76. surb->cb->size = phdr->msg_length;
  77. if (dev->response_alignment &&
  78. (phdr->msg_flags & MSG_HDR_FLAG_SPLIT_MSG)) {
  79. surb->cb->offset =
  80. dev->response_alignment +
  81. ((phdr->msg_flags >> 8) & 3);
  82. /* sanity check */
  83. if (((int) phdr->msg_length +
  84. surb->cb->offset) > urb->actual_length) {
  85. sms_err("invalid response "
  86. "msglen %d offset %d "
  87. "size %d",
  88. phdr->msg_length,
  89. surb->cb->offset,
  90. urb->actual_length);
  91. goto exit_and_resubmit;
  92. }
  93. /* move buffer pointer and
  94. * copy header to its new location */
  95. memcpy((char *) phdr + surb->cb->offset,
  96. phdr, sizeof(struct sms_msg_hdr));
  97. } else
  98. surb->cb->offset = 0;
  99. sms_debug("received %s(%d) size: %d",
  100. smscore_translate_msg(phdr->msg_type),
  101. phdr->msg_type, phdr->msg_length);
  102. smsendian_handle_rx_message((struct sms_msg_data *) phdr);
  103. smscore_onresponse(dev->coredev, surb->cb);
  104. surb->cb = NULL;
  105. } else {
  106. sms_err("invalid response "
  107. "msglen %d actual %d",
  108. phdr->msg_length, urb->actual_length);
  109. }
  110. } else
  111. sms_err("error, urb status %d, %d bytes",
  112. urb->status, urb->actual_length);
  113. exit_and_resubmit:
  114. smsusb_submit_urb(dev, surb);
  115. }
  116. static int smsusb_submit_urb(struct smsusb_device_t *dev,
  117. struct smsusb_urb_t *surb)
  118. {
  119. if (!surb->cb) {
  120. surb->cb = smscore_getbuffer(dev->coredev);
  121. if (!surb->cb) {
  122. sms_err("smscore_getbuffer(...) returned NULL");
  123. return -ENOMEM;
  124. }
  125. }
  126. usb_fill_bulk_urb(
  127. &surb->urb,
  128. dev->udev,
  129. usb_rcvbulkpipe(dev->udev, dev->in_ep),
  130. surb->cb->p,
  131. dev->buffer_size,
  132. smsusb_onresponse,
  133. surb
  134. );
  135. surb->urb.transfer_dma = surb->cb->phys;
  136. surb->urb.transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
  137. return usb_submit_urb(&surb->urb, GFP_ATOMIC);
  138. }
  139. static void smsusb_stop_streaming(struct smsusb_device_t *dev)
  140. {
  141. int i;
  142. for (i = 0; i < MAX_URBS; i++) {
  143. usb_kill_urb(&dev->surbs[i].urb);
  144. if (dev->surbs[i].cb) {
  145. smscore_putbuffer(dev->coredev, dev->surbs[i].cb);
  146. dev->surbs[i].cb = NULL;
  147. }
  148. }
  149. }
  150. static int smsusb_start_streaming(struct smsusb_device_t *dev)
  151. {
  152. int i, rc;
  153. for (i = 0; i < MAX_URBS; i++) {
  154. rc = smsusb_submit_urb(dev, &dev->surbs[i]);
  155. if (rc < 0) {
  156. sms_err("smsusb_submit_urb(...) failed");
  157. smsusb_stop_streaming(dev);
  158. break;
  159. }
  160. }
  161. return rc;
  162. }
  163. static int smsusb_sendrequest(void *context, void *buffer, size_t size)
  164. {
  165. struct smsusb_device_t *dev = (struct smsusb_device_t *) context;
  166. struct sms_msg_hdr *phdr = (struct sms_msg_hdr *) buffer;
  167. int dummy;
  168. if (dev->state != SMSUSB_ACTIVE)
  169. return -ENOENT;
  170. sms_debug("sending %s(%d) size: %d",
  171. smscore_translate_msg(phdr->msg_type), phdr->msg_type,
  172. phdr->msg_length);
  173. smsendian_handle_tx_message((struct sms_msg_data *) phdr);
  174. smsendian_handle_message_header((struct sms_msg_hdr *)buffer);
  175. return usb_bulk_msg(dev->udev, usb_sndbulkpipe(dev->udev, 2),
  176. buffer, size, &dummy, 1000);
  177. }
  178. static char *smsusb1_fw_lkup[] = {
  179. "dvbt_stellar_usb.inp",
  180. "dvbh_stellar_usb.inp",
  181. "tdmb_stellar_usb.inp",
  182. "none",
  183. "dvbt_bda_stellar_usb.inp",
  184. };
  185. static inline char *sms_get_fw_name(int mode, int board_id)
  186. {
  187. char **fw = sms_get_board(board_id)->fw;
  188. return (fw && fw[mode]) ? fw[mode] : smsusb1_fw_lkup[mode];
  189. }
  190. static int smsusb1_load_firmware(struct usb_device *udev, int id, int board_id)
  191. {
  192. const struct firmware *fw;
  193. u8 *fw_buffer;
  194. int rc, dummy;
  195. char *fw_filename;
  196. if (id < DEVICE_MODE_DVBT || id > DEVICE_MODE_DVBT_BDA) {
  197. sms_err("invalid firmware id specified %d", id);
  198. return -EINVAL;
  199. }
  200. fw_filename = sms_get_fw_name(id, board_id);
  201. rc = request_firmware(&fw, fw_filename, &udev->dev);
  202. if (rc < 0) {
  203. sms_warn("failed to open \"%s\" mode %d, "
  204. "trying again with default firmware", fw_filename, id);
  205. fw_filename = smsusb1_fw_lkup[id];
  206. rc = request_firmware(&fw, fw_filename, &udev->dev);
  207. if (rc < 0) {
  208. sms_warn("failed to open \"%s\" mode %d",
  209. fw_filename, id);
  210. return rc;
  211. }
  212. }
  213. fw_buffer = kmalloc(fw->size, GFP_KERNEL);
  214. if (fw_buffer) {
  215. memcpy(fw_buffer, fw->data, fw->size);
  216. rc = usb_bulk_msg(udev, usb_sndbulkpipe(udev, 2),
  217. fw_buffer, fw->size, &dummy, 1000);
  218. sms_info("sent %zd(%d) bytes, rc %d", fw->size, dummy, rc);
  219. kfree(fw_buffer);
  220. } else {
  221. sms_err("failed to allocate firmware buffer");
  222. rc = -ENOMEM;
  223. }
  224. sms_info("read FW %s, size=%zd", fw_filename, fw->size);
  225. release_firmware(fw);
  226. return rc;
  227. }
  228. static void smsusb1_detectmode(void *context, int *mode)
  229. {
  230. char *product_string =
  231. ((struct smsusb_device_t *) context)->udev->product;
  232. *mode = DEVICE_MODE_NONE;
  233. if (!product_string) {
  234. product_string = "none";
  235. sms_err("product string not found");
  236. } else if (strstr(product_string, "DVBH"))
  237. *mode = 1;
  238. else if (strstr(product_string, "BDA"))
  239. *mode = 4;
  240. else if (strstr(product_string, "DVBT"))
  241. *mode = 0;
  242. else if (strstr(product_string, "TDMB"))
  243. *mode = 2;
  244. sms_info("%d \"%s\"", *mode, product_string);
  245. }
  246. static int smsusb1_setmode(void *context, int mode)
  247. {
  248. struct sms_msg_hdr msg = { MSG_SW_RELOAD_REQ, 0, HIF_TASK,
  249. sizeof(struct sms_msg_hdr), 0 };
  250. if (mode < DEVICE_MODE_DVBT || mode > DEVICE_MODE_DVBT_BDA) {
  251. sms_err("invalid firmware id specified %d", mode);
  252. return -EINVAL;
  253. }
  254. return smsusb_sendrequest(context, &msg, sizeof(msg));
  255. }
  256. static void smsusb_term_device(struct usb_interface *intf)
  257. {
  258. struct smsusb_device_t *dev = usb_get_intfdata(intf);
  259. if (dev) {
  260. dev->state = SMSUSB_DISCONNECTED;
  261. smsusb_stop_streaming(dev);
  262. /* unregister from smscore */
  263. if (dev->coredev)
  264. smscore_unregister_device(dev->coredev);
  265. sms_info("device 0x%p destroyed", dev);
  266. kfree(dev);
  267. }
  268. usb_set_intfdata(intf, NULL);
  269. }
  270. static int smsusb_init_device(struct usb_interface *intf, int board_id)
  271. {
  272. struct smsdevice_params_t params;
  273. struct smsusb_device_t *dev;
  274. int i, rc;
  275. /* create device object */
  276. dev = kzalloc(sizeof(struct smsusb_device_t), GFP_KERNEL);
  277. if (!dev) {
  278. sms_err("kzalloc(sizeof(struct smsusb_device_t) failed");
  279. return -ENOMEM;
  280. }
  281. memset(&params, 0, sizeof(params));
  282. usb_set_intfdata(intf, dev);
  283. dev->udev = interface_to_usbdev(intf);
  284. dev->state = SMSUSB_DISCONNECTED;
  285. params.device_type = sms_get_board(board_id)->type;
  286. switch (params.device_type) {
  287. case SMS_STELLAR:
  288. dev->buffer_size = USB1_BUFFER_SIZE;
  289. params.setmode_handler = smsusb1_setmode;
  290. params.detectmode_handler = smsusb1_detectmode;
  291. break;
  292. case SMS_UNKNOWN_TYPE:
  293. sms_err("Unspecified sms device type!");
  294. /* fall-thru */
  295. default:
  296. dev->buffer_size = USB2_BUFFER_SIZE;
  297. dev->response_alignment =
  298. le16_to_cpu(dev->udev->ep_in[1]->desc.wMaxPacketSize) -
  299. sizeof(struct sms_msg_hdr);
  300. params.flags |= SMS_DEVICE_FAMILY2;
  301. break;
  302. }
  303. for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++) {
  304. if (intf->cur_altsetting->endpoint[i].desc. bEndpointAddress & USB_DIR_IN)
  305. dev->in_ep = intf->cur_altsetting->endpoint[i].desc.bEndpointAddress;
  306. else
  307. dev->out_ep = intf->cur_altsetting->endpoint[i].desc.bEndpointAddress;
  308. }
  309. sms_info("in_ep = %02x, out_ep = %02x",
  310. dev->in_ep, dev->out_ep);
  311. params.device = &dev->udev->dev;
  312. params.buffer_size = dev->buffer_size;
  313. params.num_buffers = MAX_BUFFERS;
  314. params.sendrequest_handler = smsusb_sendrequest;
  315. params.context = dev;
  316. usb_make_path(dev->udev, params.devpath, sizeof(params.devpath));
  317. /* register in smscore */
  318. rc = smscore_register_device(&params, &dev->coredev);
  319. if (rc < 0) {
  320. sms_err("smscore_register_device(...) failed, rc %d", rc);
  321. smsusb_term_device(intf);
  322. return rc;
  323. }
  324. smscore_set_board_id(dev->coredev, board_id);
  325. dev->coredev->is_usb_device = true;
  326. /* initialize urbs */
  327. for (i = 0; i < MAX_URBS; i++) {
  328. dev->surbs[i].dev = dev;
  329. usb_init_urb(&dev->surbs[i].urb);
  330. }
  331. sms_info("smsusb_start_streaming(...).");
  332. rc = smsusb_start_streaming(dev);
  333. if (rc < 0) {
  334. sms_err("smsusb_start_streaming(...) failed");
  335. smsusb_term_device(intf);
  336. return rc;
  337. }
  338. dev->state = SMSUSB_ACTIVE;
  339. rc = smscore_start_device(dev->coredev);
  340. if (rc < 0) {
  341. sms_err("smscore_start_device(...) failed");
  342. smsusb_term_device(intf);
  343. return rc;
  344. }
  345. sms_info("device 0x%p created", dev);
  346. return rc;
  347. }
  348. static int smsusb_probe(struct usb_interface *intf,
  349. const struct usb_device_id *id)
  350. {
  351. struct usb_device *udev = interface_to_usbdev(intf);
  352. char devpath[32];
  353. int i, rc;
  354. sms_info("interface number %d",
  355. intf->cur_altsetting->desc.bInterfaceNumber);
  356. if (sms_get_board(id->driver_info)->intf_num !=
  357. intf->cur_altsetting->desc.bInterfaceNumber) {
  358. sms_err("interface number is %d expecting %d",
  359. sms_get_board(id->driver_info)->intf_num,
  360. intf->cur_altsetting->desc.bInterfaceNumber);
  361. return -ENODEV;
  362. }
  363. if (intf->num_altsetting > 1) {
  364. rc = usb_set_interface(udev,
  365. intf->cur_altsetting->desc.bInterfaceNumber,
  366. 0);
  367. if (rc < 0) {
  368. sms_err("usb_set_interface failed, rc %d", rc);
  369. return rc;
  370. }
  371. }
  372. sms_info("smsusb_probe %d",
  373. intf->cur_altsetting->desc.bInterfaceNumber);
  374. for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++) {
  375. sms_info("endpoint %d %02x %02x %d", i,
  376. intf->cur_altsetting->endpoint[i].desc.bEndpointAddress,
  377. intf->cur_altsetting->endpoint[i].desc.bmAttributes,
  378. intf->cur_altsetting->endpoint[i].desc.wMaxPacketSize);
  379. if (intf->cur_altsetting->endpoint[i].desc.bEndpointAddress &
  380. USB_DIR_IN)
  381. rc = usb_clear_halt(udev, usb_rcvbulkpipe(udev,
  382. intf->cur_altsetting->endpoint[i].desc.bEndpointAddress));
  383. else
  384. rc = usb_clear_halt(udev, usb_sndbulkpipe(udev,
  385. intf->cur_altsetting->endpoint[i].desc.bEndpointAddress));
  386. }
  387. if ((udev->actconfig->desc.bNumInterfaces == 2) &&
  388. (intf->cur_altsetting->desc.bInterfaceNumber == 0)) {
  389. sms_err("rom interface 0 is not used");
  390. return -ENODEV;
  391. }
  392. if (id->driver_info == SMS1XXX_BOARD_SIANO_STELLAR_ROM) {
  393. sms_info("stellar device was found.");
  394. snprintf(devpath, sizeof(devpath), "usb\\%d-%s",
  395. udev->bus->busnum, udev->devpath);
  396. sms_info("stellar device was found.");
  397. return smsusb1_load_firmware(
  398. udev, smscore_registry_getmode(devpath),
  399. id->driver_info);
  400. }
  401. rc = smsusb_init_device(intf, id->driver_info);
  402. sms_info("rc %d", rc);
  403. sms_board_load_modules(id->driver_info);
  404. return rc;
  405. }
  406. static void smsusb_disconnect(struct usb_interface *intf)
  407. {
  408. smsusb_term_device(intf);
  409. }
  410. static int smsusb_suspend(struct usb_interface *intf, pm_message_t msg)
  411. {
  412. struct smsusb_device_t *dev = usb_get_intfdata(intf);
  413. printk(KERN_INFO "%s Entering status %d.\n", __func__, msg.event);
  414. dev->state = SMSUSB_SUSPENDED;
  415. /*smscore_set_power_mode(dev, SMS_POWER_MODE_SUSPENDED);*/
  416. smsusb_stop_streaming(dev);
  417. return 0;
  418. }
  419. static int smsusb_resume(struct usb_interface *intf)
  420. {
  421. int rc, i;
  422. struct smsusb_device_t *dev = usb_get_intfdata(intf);
  423. struct usb_device *udev = interface_to_usbdev(intf);
  424. printk(KERN_INFO "%s Entering.\n", __func__);
  425. usb_clear_halt(udev, usb_rcvbulkpipe(udev, dev->in_ep));
  426. usb_clear_halt(udev, usb_sndbulkpipe(udev, dev->out_ep));
  427. for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++)
  428. printk(KERN_INFO "endpoint %d %02x %02x %d\n", i,
  429. intf->cur_altsetting->endpoint[i].desc.bEndpointAddress,
  430. intf->cur_altsetting->endpoint[i].desc.bmAttributes,
  431. intf->cur_altsetting->endpoint[i].desc.wMaxPacketSize);
  432. if (intf->num_altsetting > 0) {
  433. rc = usb_set_interface(udev,
  434. intf->cur_altsetting->desc.
  435. bInterfaceNumber, 0);
  436. if (rc < 0) {
  437. printk(KERN_INFO "%s usb_set_interface failed, "
  438. "rc %d\n", __func__, rc);
  439. return rc;
  440. }
  441. }
  442. smsusb_start_streaming(dev);
  443. return 0;
  444. }
  445. static const struct usb_device_id smsusb_id_table[] = {
  446. { USB_DEVICE(0x187f, 0x0010),
  447. .driver_info = SMS1XXX_BOARD_SIANO_STELLAR },
  448. { USB_DEVICE(0x187f, 0x0100),
  449. .driver_info = SMS1XXX_BOARD_SIANO_STELLAR },
  450. { USB_DEVICE(0x187f, 0x0200),
  451. .driver_info = SMS1XXX_BOARD_SIANO_NOVA_A },
  452. { USB_DEVICE(0x187f, 0x0201),
  453. .driver_info = SMS1XXX_BOARD_SIANO_NOVA_B },
  454. { USB_DEVICE(0x187f, 0x0300),
  455. .driver_info = SMS1XXX_BOARD_SIANO_VEGA },
  456. { USB_DEVICE(0x2040, 0x1700),
  457. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_CATAMOUNT },
  458. { USB_DEVICE(0x2040, 0x1800),
  459. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_OKEMO_A },
  460. { USB_DEVICE(0x2040, 0x1801),
  461. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_OKEMO_B },
  462. { USB_DEVICE(0x2040, 0x2000),
  463. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  464. { USB_DEVICE(0x2040, 0x2009),
  465. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD_R2 },
  466. { USB_DEVICE(0x2040, 0x200a),
  467. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  468. { USB_DEVICE(0x2040, 0x2010),
  469. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  470. { USB_DEVICE(0x2040, 0x2011),
  471. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  472. { USB_DEVICE(0x2040, 0x2019),
  473. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  474. { USB_DEVICE(0x2040, 0x5500),
  475. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  476. { USB_DEVICE(0x2040, 0x5510),
  477. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  478. { USB_DEVICE(0x2040, 0x5520),
  479. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  480. { USB_DEVICE(0x2040, 0x5530),
  481. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  482. { USB_DEVICE(0x2040, 0x5580),
  483. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  484. { USB_DEVICE(0x2040, 0x5590),
  485. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  486. { USB_DEVICE(0x187f, 0x0202),
  487. .driver_info = SMS1XXX_BOARD_SIANO_NICE },
  488. { USB_DEVICE(0x187f, 0x0301),
  489. .driver_info = SMS1XXX_BOARD_SIANO_VENICE },
  490. { USB_DEVICE(0x2040, 0xb900),
  491. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  492. { USB_DEVICE(0x2040, 0xb910),
  493. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  494. { USB_DEVICE(0x2040, 0xb980),
  495. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  496. { USB_DEVICE(0x2040, 0xb990),
  497. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  498. { USB_DEVICE(0x2040, 0xc000),
  499. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  500. { USB_DEVICE(0x2040, 0xc010),
  501. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  502. { USB_DEVICE(0x2040, 0xc080),
  503. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  504. { USB_DEVICE(0x2040, 0xc090),
  505. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  506. { USB_DEVICE(0x2040, 0xc0a0),
  507. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  508. { USB_DEVICE(0x2040, 0xf5a0),
  509. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  510. { USB_DEVICE(0x187f, 0x0202),
  511. .driver_info = SMS1XXX_BOARD_SIANO_NICE },
  512. { USB_DEVICE(0x187f, 0x0301),
  513. .driver_info = SMS1XXX_BOARD_SIANO_VENICE },
  514. { USB_DEVICE(0x187f, 0x0302),
  515. .driver_info = SMS1XXX_BOARD_SIANO_VENICE },
  516. { USB_DEVICE(0x187f, 0x0310),
  517. .driver_info = SMS1XXX_BOARD_SIANO_MING },
  518. { USB_DEVICE(0x187f, 0x0500),
  519. .driver_info = SMS1XXX_BOARD_SIANO_PELE },
  520. { USB_DEVICE(0x187f, 0x0600),
  521. .driver_info = SMS1XXX_BOARD_SIANO_RIO },
  522. { USB_DEVICE(0x187f, 0x0700),
  523. .driver_info = SMS1XXX_BOARD_SIANO_DENVER_2160 },
  524. { USB_DEVICE(0x187f, 0x0800),
  525. .driver_info = SMS1XXX_BOARD_SIANO_DENVER_1530 },
  526. { USB_DEVICE(0x19D2, 0x0086),
  527. .driver_info = SMS1XXX_BOARD_ZTE_DVB_DATA_CARD },
  528. { USB_DEVICE(0x19D2, 0x0078),
  529. .driver_info = SMS1XXX_BOARD_ONDA_MDTV_DATA_CARD },
  530. { } /* Terminating entry */
  531. };
  532. MODULE_DEVICE_TABLE(usb, smsusb_id_table);
  533. static struct usb_driver smsusb_driver = {
  534. .name = "smsusb",
  535. .probe = smsusb_probe,
  536. .disconnect = smsusb_disconnect,
  537. .id_table = smsusb_id_table,
  538. .suspend = smsusb_suspend,
  539. .resume = smsusb_resume,
  540. };
  541. module_usb_driver(smsusb_driver);
  542. MODULE_DESCRIPTION("Driver for the Siano SMS1xxx USB dongle");
  543. MODULE_AUTHOR("Siano Mobile Silicon, INC. (uris@siano-ms.com)");
  544. MODULE_LICENSE("GPL");