smsusb.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658
  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 SmsMsgHdr_ST *phdr = (struct SmsMsgHdr_ST *)surb->cb->p;
  74. smsendian_handle_message_header(phdr);
  75. if (urb->actual_length >= phdr->msgLength) {
  76. surb->cb->size = phdr->msgLength;
  77. if (dev->response_alignment &&
  78. (phdr->msgFlags & MSG_HDR_FLAG_SPLIT_MSG)) {
  79. surb->cb->offset =
  80. dev->response_alignment +
  81. ((phdr->msgFlags >> 8) & 3);
  82. /* sanity check */
  83. if (((int) phdr->msgLength +
  84. surb->cb->offset) > urb->actual_length) {
  85. sms_err("invalid response "
  86. "msglen %d offset %d "
  87. "size %d",
  88. phdr->msgLength,
  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 SmsMsgHdr_ST));
  97. } else
  98. surb->cb->offset = 0;
  99. sms_debug("received %s(%d) size: %d",
  100. smscore_translate_msg(phdr->msgType),
  101. phdr->msgType, phdr->msgLength);
  102. smsendian_handle_rx_message((struct SmsMsgData_ST *) 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->msgLength, 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 SmsMsgHdr_ST *phdr = (struct SmsMsgHdr_ST *) 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->msgType), phdr->msgType,
  172. phdr->msgLength);
  173. smsendian_handle_tx_message((struct SmsMsgData_ST *) phdr);
  174. smsendian_handle_message_header((struct SmsMsgHdr_ST *)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 SmsMsgHdr_ST Msg = { MSG_SW_RELOAD_REQ, 0, HIF_TASK,
  249. sizeof(struct SmsMsgHdr_ST), 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. default:
  293. sms_err("Unspecified sms device type!");
  294. /* fall-thru */
  295. case SMS_NOVA_A0:
  296. case SMS_NOVA_B0:
  297. case SMS_VEGA:
  298. case SMS_VENICE:
  299. case SMS_DENVER_1530:
  300. dev->buffer_size = USB2_BUFFER_SIZE;
  301. dev->response_alignment =
  302. le16_to_cpu(dev->udev->ep_in[1]->desc.wMaxPacketSize) -
  303. sizeof(struct SmsMsgHdr_ST);
  304. params.flags |= SMS_DEVICE_FAMILY2;
  305. break;
  306. }
  307. for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++) {
  308. if (intf->cur_altsetting->endpoint[i].desc. bEndpointAddress & USB_DIR_IN)
  309. dev->in_ep = intf->cur_altsetting->endpoint[i].desc.bEndpointAddress;
  310. else
  311. dev->out_ep = intf->cur_altsetting->endpoint[i].desc.bEndpointAddress;
  312. }
  313. sms_info("in_ep = %02x, out_ep = %02x",
  314. dev->in_ep, dev->out_ep);
  315. params.device = &dev->udev->dev;
  316. params.buffer_size = dev->buffer_size;
  317. params.num_buffers = MAX_BUFFERS;
  318. params.sendrequest_handler = smsusb_sendrequest;
  319. params.context = dev;
  320. usb_make_path(dev->udev, params.devpath, sizeof(params.devpath));
  321. /* register in smscore */
  322. rc = smscore_register_device(&params, &dev->coredev);
  323. if (rc < 0) {
  324. sms_err("smscore_register_device(...) failed, rc %d", rc);
  325. smsusb_term_device(intf);
  326. return rc;
  327. }
  328. smscore_set_board_id(dev->coredev, board_id);
  329. /* initialize urbs */
  330. for (i = 0; i < MAX_URBS; i++) {
  331. dev->surbs[i].dev = dev;
  332. usb_init_urb(&dev->surbs[i].urb);
  333. }
  334. sms_info("smsusb_start_streaming(...).");
  335. rc = smsusb_start_streaming(dev);
  336. if (rc < 0) {
  337. sms_err("smsusb_start_streaming(...) failed");
  338. smsusb_term_device(intf);
  339. return rc;
  340. }
  341. dev->state = SMSUSB_ACTIVE;
  342. rc = smscore_start_device(dev->coredev);
  343. if (rc < 0) {
  344. sms_err("smscore_start_device(...) failed");
  345. smsusb_term_device(intf);
  346. return rc;
  347. }
  348. sms_info("device 0x%p created", dev);
  349. return rc;
  350. }
  351. static int smsusb_probe(struct usb_interface *intf,
  352. const struct usb_device_id *id)
  353. {
  354. struct usb_device *udev = interface_to_usbdev(intf);
  355. char devpath[32];
  356. int i, rc;
  357. sms_info("interface number %d",
  358. intf->cur_altsetting->desc.bInterfaceNumber);
  359. if (sms_get_board(id->driver_info)->intf_num !=
  360. intf->cur_altsetting->desc.bInterfaceNumber) {
  361. sms_err("interface number is %d expecting %d",
  362. sms_get_board(id->driver_info)->intf_num,
  363. intf->cur_altsetting->desc.bInterfaceNumber);
  364. return -ENODEV;
  365. }
  366. if (intf->num_altsetting > 1) {
  367. rc = usb_set_interface(udev,
  368. intf->cur_altsetting->desc.bInterfaceNumber,
  369. 0);
  370. if (rc < 0) {
  371. sms_err("usb_set_interface failed, rc %d", rc);
  372. return rc;
  373. }
  374. }
  375. sms_info("smsusb_probe %d",
  376. intf->cur_altsetting->desc.bInterfaceNumber);
  377. for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++) {
  378. sms_info("endpoint %d %02x %02x %d", i,
  379. intf->cur_altsetting->endpoint[i].desc.bEndpointAddress,
  380. intf->cur_altsetting->endpoint[i].desc.bmAttributes,
  381. intf->cur_altsetting->endpoint[i].desc.wMaxPacketSize);
  382. if (intf->cur_altsetting->endpoint[i].desc.bEndpointAddress &
  383. USB_DIR_IN)
  384. rc = usb_clear_halt(udev, usb_rcvbulkpipe(udev,
  385. intf->cur_altsetting->endpoint[i].desc.bEndpointAddress));
  386. else
  387. rc = usb_clear_halt(udev, usb_sndbulkpipe(udev,
  388. intf->cur_altsetting->endpoint[i].desc.bEndpointAddress));
  389. }
  390. if ((udev->actconfig->desc.bNumInterfaces == 2) &&
  391. (intf->cur_altsetting->desc.bInterfaceNumber == 0)) {
  392. sms_err("rom interface 0 is not used");
  393. return -ENODEV;
  394. }
  395. if (id->driver_info == SMS1XXX_BOARD_SIANO_STELLAR_ROM) {
  396. sms_info("stellar device was found.");
  397. snprintf(devpath, sizeof(devpath), "usb\\%d-%s",
  398. udev->bus->busnum, udev->devpath);
  399. sms_info("stellar device was found.");
  400. return smsusb1_load_firmware(
  401. udev, smscore_registry_getmode(devpath),
  402. id->driver_info);
  403. }
  404. rc = smsusb_init_device(intf, id->driver_info);
  405. sms_info("rc %d", rc);
  406. sms_board_load_modules(id->driver_info);
  407. return rc;
  408. }
  409. static void smsusb_disconnect(struct usb_interface *intf)
  410. {
  411. smsusb_term_device(intf);
  412. }
  413. static int smsusb_suspend(struct usb_interface *intf, pm_message_t msg)
  414. {
  415. struct smsusb_device_t *dev = usb_get_intfdata(intf);
  416. printk(KERN_INFO "%s Entering status %d.\n", __func__, msg.event);
  417. dev->state = SMSUSB_SUSPENDED;
  418. /*smscore_set_power_mode(dev, SMS_POWER_MODE_SUSPENDED);*/
  419. smsusb_stop_streaming(dev);
  420. return 0;
  421. }
  422. static int smsusb_resume(struct usb_interface *intf)
  423. {
  424. int rc, i;
  425. struct smsusb_device_t *dev = usb_get_intfdata(intf);
  426. struct usb_device *udev = interface_to_usbdev(intf);
  427. printk(KERN_INFO "%s Entering.\n", __func__);
  428. usb_clear_halt(udev, usb_rcvbulkpipe(udev, dev->in_ep));
  429. usb_clear_halt(udev, usb_sndbulkpipe(udev, dev->out_ep));
  430. for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++)
  431. printk(KERN_INFO "endpoint %d %02x %02x %d\n", i,
  432. intf->cur_altsetting->endpoint[i].desc.bEndpointAddress,
  433. intf->cur_altsetting->endpoint[i].desc.bmAttributes,
  434. intf->cur_altsetting->endpoint[i].desc.wMaxPacketSize);
  435. if (intf->num_altsetting > 0) {
  436. rc = usb_set_interface(udev,
  437. intf->cur_altsetting->desc.
  438. bInterfaceNumber, 0);
  439. if (rc < 0) {
  440. printk(KERN_INFO "%s usb_set_interface failed, "
  441. "rc %d\n", __func__, rc);
  442. return rc;
  443. }
  444. }
  445. smsusb_start_streaming(dev);
  446. return 0;
  447. }
  448. static const struct usb_device_id smsusb_id_table[] = {
  449. { USB_DEVICE(0x187f, 0x0010),
  450. .driver_info = SMS1XXX_BOARD_SIANO_STELLAR },
  451. { USB_DEVICE(0x187f, 0x0100),
  452. .driver_info = SMS1XXX_BOARD_SIANO_STELLAR },
  453. { USB_DEVICE(0x187f, 0x0200),
  454. .driver_info = SMS1XXX_BOARD_SIANO_NOVA_A },
  455. { USB_DEVICE(0x187f, 0x0201),
  456. .driver_info = SMS1XXX_BOARD_SIANO_NOVA_B },
  457. { USB_DEVICE(0x187f, 0x0300),
  458. .driver_info = SMS1XXX_BOARD_SIANO_VEGA },
  459. { USB_DEVICE(0x2040, 0x1700),
  460. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_CATAMOUNT },
  461. { USB_DEVICE(0x2040, 0x1800),
  462. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_OKEMO_A },
  463. { USB_DEVICE(0x2040, 0x1801),
  464. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_OKEMO_B },
  465. { USB_DEVICE(0x2040, 0x2000),
  466. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  467. { USB_DEVICE(0x2040, 0x2009),
  468. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD_R2 },
  469. { USB_DEVICE(0x2040, 0x200a),
  470. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  471. { USB_DEVICE(0x2040, 0x2010),
  472. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  473. { USB_DEVICE(0x2040, 0x2011),
  474. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  475. { USB_DEVICE(0x2040, 0x2019),
  476. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_TIGER_MINICARD },
  477. { USB_DEVICE(0x2040, 0x5500),
  478. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  479. { USB_DEVICE(0x2040, 0x5510),
  480. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  481. { USB_DEVICE(0x2040, 0x5520),
  482. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  483. { USB_DEVICE(0x2040, 0x5530),
  484. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  485. { USB_DEVICE(0x2040, 0x5580),
  486. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  487. { USB_DEVICE(0x2040, 0x5590),
  488. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  489. { USB_DEVICE(0x187f, 0x0202),
  490. .driver_info = SMS1XXX_BOARD_SIANO_NICE },
  491. { USB_DEVICE(0x187f, 0x0301),
  492. .driver_info = SMS1XXX_BOARD_SIANO_VENICE },
  493. { USB_DEVICE(0x2040, 0xb900),
  494. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  495. { USB_DEVICE(0x2040, 0xb910),
  496. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  497. { USB_DEVICE(0x2040, 0xb980),
  498. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  499. { USB_DEVICE(0x2040, 0xb990),
  500. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  501. { USB_DEVICE(0x2040, 0xc000),
  502. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  503. { USB_DEVICE(0x2040, 0xc010),
  504. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  505. { USB_DEVICE(0x2040, 0xc080),
  506. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  507. { USB_DEVICE(0x2040, 0xc090),
  508. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  509. { USB_DEVICE(0x2040, 0xc0a0),
  510. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  511. { USB_DEVICE(0x2040, 0xf5a0),
  512. .driver_info = SMS1XXX_BOARD_HAUPPAUGE_WINDHAM },
  513. { USB_DEVICE(0x187f, 0x0202),
  514. .driver_info = SMS1XXX_BOARD_SIANO_NICE },
  515. { USB_DEVICE(0x187f, 0x0301),
  516. .driver_info = SMS1XXX_BOARD_SIANO_VENICE },
  517. { USB_DEVICE(0x187f, 0x0302),
  518. .driver_info = SMS1XXX_BOARD_SIANO_VENICE },
  519. { USB_DEVICE(0x187f, 0x0310),
  520. .driver_info = SMS1XXX_BOARD_SIANO_MING },
  521. { USB_DEVICE(0x187f, 0x0500),
  522. .driver_info = SMS1XXX_BOARD_SIANO_PELE },
  523. { USB_DEVICE(0x187f, 0x0600),
  524. .driver_info = SMS1XXX_BOARD_SIANO_RIO },
  525. { USB_DEVICE(0x187f, 0x0700),
  526. .driver_info = SMS1XXX_BOARD_SIANO_DENVER_2160 },
  527. { USB_DEVICE(0x187f, 0x0800),
  528. .driver_info = SMS1XXX_BOARD_SIANO_DENVER_1530 },
  529. { USB_DEVICE(0x19D2, 0x0086),
  530. .driver_info = SMS1XXX_BOARD_ZTE_DVB_DATA_CARD },
  531. { USB_DEVICE(0x19D2, 0x0078),
  532. .driver_info = SMS1XXX_BOARD_ONDA_MDTV_DATA_CARD },
  533. { } /* Terminating entry */
  534. };
  535. MODULE_DEVICE_TABLE(usb, smsusb_id_table);
  536. static struct usb_driver smsusb_driver = {
  537. .name = "smsusb",
  538. .probe = smsusb_probe,
  539. .disconnect = smsusb_disconnect,
  540. .id_table = smsusb_id_table,
  541. .suspend = smsusb_suspend,
  542. .resume = smsusb_resume,
  543. };
  544. module_usb_driver(smsusb_driver);
  545. MODULE_DESCRIPTION("Driver for the Siano SMS1xxx USB dongle");
  546. MODULE_AUTHOR("Siano Mobile Silicon, INC. (uris@siano-ms.com)");
  547. MODULE_LICENSE("GPL");