SHARE
TWEET

tbs-qbox.c

a guest Jul 3rd, 2012 57 Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
  1. /* DVB USB framework compliant Linux driver for the
  2. *       TBS QBOX
  3. *
  4. * Copyright (C) 2008 Bob Liu (Bob@Turbosight.com)
  5. * Igor M. Liplianin (liplianin@me.by)
  6. *
  7. * Konstantin Dimitrov <kosio.dimitrov@gmail.com>
  8. *       June 2009
  9. *       Some fixes and improvements.
  10. *
  11. *       This program is free software; you can redistribute it and/or modify it
  12. *       under the terms of the GNU General Public License as published by the
  13. *       Free Software Foundation, version 2.
  14. *
  15. * see Documentation/dvb/README.dvb-usb for more information
  16. */
  17. #include <linux/version.h>
  18. #include "tbs-qbox.h"
  19. #include "stv0299.h"
  20. #include "stv0288.h"
  21. #include "stb6000.h"
  22.  
  23.  
  24. #ifndef USB_PID_TBSQBOX
  25. #define USB_PID_TBSQBOX 0x2601
  26. #endif
  27.  
  28. #ifndef USB_PID_TBSQBOX_1
  29. #define USB_PID_TBSQBOX_1 0x2601
  30. #endif
  31.  
  32. #define TBSQBOX_READ_MSG 0
  33. #define TBSQBOX_WRITE_MSG 1
  34.  
  35. /* on my own*/
  36. #define TBSQBOX_VOLTAGE_CTRL (0x1800)
  37. #define TBSQBOX_RC_QUERY (0x1a00)
  38.  
  39. struct tbsqboxs1_state {
  40.         u32 last_key_pressed;
  41. };
  42. struct tbsqboxs1_rc_keys {
  43.         u32 keycode;
  44.         u32 event;
  45. };
  46.  
  47. /* debug */
  48. static int dvb_usb_tbsqboxs1_debug;
  49. module_param_named(debug, dvb_usb_tbsqboxs1_debug, int, 0644);
  50. MODULE_PARM_DESC(debug, "set debugging level (1=info 2=xfer (or-able))." DVB_USB_DEBUG_STATUS);
  51.  
  52. DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr);
  53.  
  54. static int tbsqboxs1_op_rw(struct usb_device *dev, u8 request, u16 value,
  55.                         u16 index, u8 * data, u16 len, int flags)
  56. {
  57.         int ret;
  58.         u8 u8buf[len];
  59.  
  60.         unsigned int pipe = (flags == TBSQBOX_READ_MSG) ?
  61.                                 usb_rcvctrlpipe(dev, 0) : usb_sndctrlpipe(dev, 0);
  62.         u8 request_type = (flags == TBSQBOX_READ_MSG) ? USB_DIR_IN : USB_DIR_OUT;
  63.  
  64.         if (flags == TBSQBOX_WRITE_MSG)
  65.                 memcpy(u8buf, data, len);
  66.         ret = usb_control_msg(dev, pipe, request, request_type | USB_TYPE_VENDOR,
  67.                                 value, index , u8buf, len, 2000);
  68.  
  69.         if (flags == TBSQBOX_READ_MSG)
  70.                 memcpy(data, u8buf, len);
  71.         return ret;
  72. }
  73.  
  74. /* I2C */
  75. static int tbsqboxs1_i2c_transfer(struct i2c_adapter *adap, struct i2c_msg msg[],
  76.                 int num)
  77. {
  78. struct dvb_usb_device *d = i2c_get_adapdata(adap);
  79.         int i = 0;
  80.         u8 buf6[20];
  81.         u8 inbuf[20];
  82.  
  83.         if (!d)
  84.                 return -ENODEV;
  85.         if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  86.                 return -EAGAIN;
  87.  
  88.         switch (num) {
  89.         case 2:
  90.                
  91.                 buf6[0]=msg[1].len;//lenth
  92.                 buf6[1]=msg[0].addr<<1;//demod addr
  93.                 buf6[2]=msg[0].buf[0];//register
  94.                
  95.                 tbsqboxs1_op_rw(d->udev, 0x90, 0, 0,
  96.                                         buf6, 3, TBSQBOX_WRITE_MSG);
  97.                 msleep(5);
  98.                 tbsqboxs1_op_rw(d->udev, 0x91, 0, 0,
  99.                                         inbuf, 1, TBSQBOX_READ_MSG);
  100.                 memcpy(msg[1].buf, inbuf, msg[1].len);
  101.                 break;
  102.         case 1:
  103.                 switch (msg[0].addr) {
  104.                 case 0x68:
  105.                         /* write to stv0299 register */
  106.                         buf6[0] = msg[0].len+1;//lenth
  107.                         buf6[1] = msg[0].addr<<1;//demod addr
  108.                         for(i=0;i<msg[0].len;i++)
  109.                                 {
  110.                                 buf6[2+i] = msg[0].buf[i];//register
  111.                                 }
  112.                         tbsqboxs1_op_rw(d->udev, 0x80, 0, 0,
  113.                                         buf6, msg[0].len+2, TBSQBOX_WRITE_MSG);
  114.                         msleep(3);
  115.                         break;
  116.                 case 0x61:
  117.                 case 0x60:
  118.                         if (msg[0].flags == 0) {
  119.                         /* write to tuner pll */
  120.                                 buf6[0] = msg[0].len+1;//lenth
  121.                                 buf6[1] = msg[0].addr<<1;//tuner addr
  122.                                 for(i=0;i<msg[0].len;i++)
  123.                                 {
  124.                                 buf6[2+i] = msg[0].buf[i];//register
  125.                                 }
  126.                                 tbsqboxs1_op_rw(d->udev, 0x80, 0, 0,
  127.                                                 buf6, msg[0].len+2, TBSQBOX_WRITE_MSG);
  128.                         }
  129.                         msleep(3);
  130.                         break;
  131.                 case (TBSQBOX_RC_QUERY):
  132.                         tbsqboxs1_op_rw(d->udev, 0xb8, 0, 0,
  133.                                         buf6, 4, TBSQBOX_READ_MSG);
  134.                         msg[0].buf[0] = buf6[2];
  135.                         msg[0].buf[1] = buf6[3];
  136.                         msleep(3);
  137.                         //info("TBSQBOX_RC_QUERY %x %x %x %x\n",buf6[0],buf6[1],buf6[2],buf6[3]);
  138.                         break;
  139.                 case (TBSQBOX_VOLTAGE_CTRL):
  140.                         buf6[0] = 3;
  141.                         buf6[1] = msg[0].buf[0];
  142.                         tbsqboxs1_op_rw(d->udev, 0x8a, 0, 0,
  143.                                         buf6, 2, TBSQBOX_WRITE_MSG);
  144.                        
  145.                         break;
  146.                 }
  147.  
  148.                 break;
  149.         }
  150.  
  151.         mutex_unlock(&d->i2c_mutex);
  152.         return num;
  153. }
  154.  
  155. static u32 tbsqboxs1_i2c_func(struct i2c_adapter *adapter)
  156. {
  157.         return I2C_FUNC_I2C;
  158. }
  159.  
  160. static struct stv0288_config earda_config = {
  161.         .demod_address = 0x68,
  162. };
  163.  
  164. static struct i2c_algorithm tbsqboxs1_i2c_algo = {
  165.         .master_xfer = tbsqboxs1_i2c_transfer,
  166.         .functionality = tbsqboxs1_i2c_func,
  167. };
  168.  
  169. static int tbsqboxs1_earda_tuner_attach(struct dvb_usb_adapter *adap)
  170. {
  171.         dvb_attach(stb6000_attach, adap->fe[0], 0x61,
  172.                 &adap->dev->i2c_adap);
  173.  
  174.         return 0;
  175. }
  176. static int tbsqboxs1_read_mac_address(struct dvb_usb_device *d, u8 mac[6])
  177. {
  178.         int i,ret;
  179.         u8 ibuf[3] = {0, 0,0};
  180.         u8 eeprom[256], eepromline[16];
  181.  
  182.         for (i = 0; i < 256; i++) {
  183.                 ibuf[0]=1;//lenth
  184.                 ibuf[1]=0xa0;//eeprom addr
  185.                 ibuf[2]=i;//register
  186.                 ret = tbsqboxs1_op_rw(d->udev, 0x90, 0, 0,
  187.                                         ibuf, 3, TBSQBOX_WRITE_MSG);
  188.                 ret = tbsqboxs1_op_rw(d->udev, 0x91, 0, 0,
  189.                                         ibuf, 1, TBSQBOX_READ_MSG);
  190.                         if (ret < 0) {
  191.                                 err("read eeprom failed.");
  192.                                 return -1;
  193.                         } else {
  194.                                 eepromline[i%16] = ibuf[0];
  195.                                 eeprom[i] = ibuf[0];
  196.                         }
  197.                        
  198.                         if ((i % 16) == 15) {
  199.                                 deb_xfer("%02x: ", i - 15);
  200.                                 debug_dump(eepromline, 16, deb_xfer);
  201.                         }
  202.         }
  203.         memcpy(mac, eeprom + 16, 6);
  204.         return 0;
  205. };
  206.  
  207. static int tbsqboxs1_set_voltage(struct dvb_frontend *fe, fe_sec_voltage_t voltage)
  208. {
  209.         static u8 command_13v[1] = {0x00};
  210.         static u8 command_18v[1] = {0x01};
  211.         struct i2c_msg msg[] = {
  212.                 {.addr = TBSQBOX_VOLTAGE_CTRL, .flags = 0,
  213.                         .buf = command_13v, .len = 1},
  214.         };
  215.        
  216.         struct dvb_usb_adapter *udev_adap =
  217.                 (struct dvb_usb_adapter *)(fe->dvb->priv);
  218.         if (voltage == SEC_VOLTAGE_18)
  219.                 msg[0].buf = command_18v;
  220.         //info("tbsqboxs1_set_voltage %d",voltage);
  221.         i2c_transfer(&udev_adap->dev->i2c_adap, msg, 1);
  222.         return 0;
  223. }
  224.  
  225. static struct dvb_usb_device_properties tbsqboxs1_properties;
  226.  
  227. static int tbsqboxs1_frontend_attach(struct dvb_usb_adapter *d)
  228. {
  229.         u8 buf[20];
  230.  
  231.         if (tbsqboxs1_properties.adapter->tuner_attach == &tbsqboxs1_earda_tuner_attach) {
  232.                 d->fe[0] = dvb_attach(stv0288_attach, &earda_config,
  233.                                         &d->dev->i2c_adap);
  234.                 if (d->fe[0] != NULL) {
  235.                         d->fe[0]->ops.set_voltage = tbsqboxs1_set_voltage;
  236.                         info("Attached stv0288!\n");
  237.  
  238.                         buf[0] = 7;
  239.                         buf[1] = 1;
  240.                         tbsqboxs1_op_rw(d->dev->udev, 0x8a, 0, 0,
  241.                                          buf, 2, TBSQBOX_WRITE_MSG);
  242.  
  243.                         return 0;
  244.                 }
  245.         }
  246.  
  247.         return -EIO;
  248. }
  249.  
  250.  
  251.  
  252. static struct rc_map_table tbsqboxs1_rc_keys[] = {
  253.         { 0xff84, KEY_POWER2},          /* power */
  254.         { 0xff94, KEY_MUTE},            /* mute */
  255.         { 0xff87, KEY_1},
  256.         { 0xff86, KEY_2},
  257.         { 0xff85, KEY_3},
  258.         { 0xff8b, KEY_4},
  259.         { 0xff8a, KEY_5},
  260.         { 0xff89, KEY_6},
  261.         { 0xff8f, KEY_7},
  262.         { 0xff8e, KEY_8},
  263.         { 0xff8d, KEY_9},
  264.         { 0xff92, KEY_0},
  265.         { 0xff96, KEY_CHANNELUP},       /* ch+ */
  266.         { 0xff91, KEY_CHANNELDOWN},     /* ch- */
  267.         { 0xff93, KEY_VOLUMEUP},        /* vol+ */
  268.         { 0xff8c, KEY_VOLUMEDOWN},      /* vol- */
  269.         { 0xff83, KEY_RECORD},          /* rec */
  270.         { 0xff98, KEY_PAUSE},           /* pause, yellow */
  271.         { 0xff99, KEY_OK},              /* ok */
  272.         { 0xff9a, KEY_CAMERA},          /* snapshot */
  273.         { 0xff81, KEY_UP},
  274.         { 0xff90, KEY_LEFT},
  275.         { 0xff82, KEY_RIGHT},
  276.         { 0xff88, KEY_DOWN},
  277.         { 0xff95, KEY_FAVORITES},       /* blue */
  278.         { 0xff97, KEY_SUBTITLE},        /* green */
  279.         { 0xff9d, KEY_ZOOM},
  280.         { 0xff9f, KEY_EXIT},
  281.         { 0xff9e, KEY_MENU},
  282.         { 0xff9c, KEY_EPG},
  283.         { 0xff80, KEY_PREVIOUS},        /* red */
  284.         { 0xff9b, KEY_MODE},
  285.         { 0xffdd, KEY_TV },
  286.         { 0xffde, KEY_PLAY },
  287.         { 0xffdc, KEY_STOP },
  288.         { 0xffdb, KEY_REWIND },
  289.         { 0xffda, KEY_FASTFORWARD },
  290.         { 0xffd9, KEY_PREVIOUS },       /* replay */
  291.         { 0xffd8, KEY_NEXT },           /* skip */
  292.         { 0xffd1, KEY_NUMERIC_STAR },
  293.         { 0xffd2, KEY_NUMERIC_POUND },
  294.         { 0xffd4, KEY_DELETE },         /* clear */
  295. };
  296.  
  297.  
  298.  
  299. static int tbsqboxs1_rc_query(struct dvb_usb_device *d, u32 *event, int *state)
  300. {
  301.         struct rc_map_table *keymap = d->props.rc.legacy.rc_map_table;
  302.         int keymap_size = d->props.rc.legacy.rc_map_size;
  303.        
  304.         struct tbsqboxs1_state *st = d->priv;
  305.         u8 key[2];
  306.         struct i2c_msg msg[] = {
  307.                 {.addr = TBSQBOX_RC_QUERY, .flags = I2C_M_RD, .buf = key,
  308.                 .len = 2},
  309.         };
  310.         int i;
  311.  
  312.         *state = REMOTE_NO_KEY_PRESSED;
  313.         if (tbsqboxs1_i2c_transfer(&d->i2c_adap, msg, 1) == 1) {
  314.                 //info("key: %x %x\n",msg[0].buf[0],msg[0].buf[1]);
  315.                 for (i = 0; i < keymap_size; i++) {
  316.                         if (rc5_data(&keymap[i]) == msg[0].buf[1]) {
  317.                                 *state = REMOTE_KEY_PRESSED;
  318.                                 *event = keymap[i].keycode;
  319.                                 st->last_key_pressed =
  320.                                         keymap[i].keycode;
  321.                                 break;
  322.                         }
  323.                 st->last_key_pressed = 0;
  324.                 }
  325.         }
  326.          
  327.         return 0;
  328. }
  329.  
  330. static struct usb_device_id tbsqboxs1_table[] = {
  331.         {USB_DEVICE(0x734c, 0x2601)},
  332.         {USB_DEVICE(0x734c, 0x2601)},
  333.         {USB_DEVICE(USB_VID_CYPRESS, USB_PID_TBSQBOX)},
  334.         {USB_DEVICE(USB_VID_CYPRESS, USB_PID_TBSQBOX_1)},
  335.         { }
  336. };
  337.  
  338. MODULE_DEVICE_TABLE(usb, tbsqboxs1_table);
  339.  
  340. static int tbsqboxs1_load_firmware(struct usb_device *dev,
  341.                         const struct firmware *frmwr)
  342. {
  343.         u8 *b, *p;
  344.         int ret = 0, i;
  345.         u8 reset;
  346.         const struct firmware *fw;
  347.         const char *filename = "dvb-usb-tbsqbox-id2601.fw";
  348.         const char *filename1 = "dvb-usb-tbsqbox-id2601.fw";
  349.         switch (dev->descriptor.idProduct) {
  350.         case 0x2601:
  351.                 ret = request_firmware(&fw, filename, &dev->dev);
  352.                 if (ret != 0) {
  353.                         err("did not find the firmware file. (%s) "
  354.                         "Please see linux/Documentation/dvb/ for more details "
  355.                         "on firmware-problems.", filename);
  356.                         return ret;
  357.                 }
  358.                 break;
  359.         default:
  360.                 fw = frmwr;
  361.                 break;
  362.         }
  363.         info("start downloading TBSQBOX firmware");
  364.         p = kmalloc(fw->size, GFP_KERNEL);
  365.         reset = 1;
  366.         /*stop the CPU*/
  367.         tbsqboxs1_op_rw(dev, 0xa0, 0x7f92, 0, &reset, 1, TBSQBOX_WRITE_MSG);
  368.         tbsqboxs1_op_rw(dev, 0xa0, 0xe600, 0, &reset, 1, TBSQBOX_WRITE_MSG);
  369.  
  370.         if (p != NULL) {
  371.                 memcpy(p, fw->data, fw->size);
  372.                 for (i = 0; i < fw->size; i += 0x40) {
  373.                         b = (u8 *) p + i;
  374.                         if (tbsqboxs1_op_rw(dev, 0xa0, i, 0, b , 0x40,
  375.                                         TBSQBOX_WRITE_MSG) != 0x40) {
  376.                                 err("error while transferring firmware");
  377.                                 ret = -EINVAL;
  378.                                 break;
  379.                         }
  380.                 }
  381.                 /* restart the CPU */
  382.                 reset = 0;
  383.                 if (ret || tbsqboxs1_op_rw(dev, 0xa0, 0x7f92, 0, &reset, 1,
  384.                                         TBSQBOX_WRITE_MSG) != 1) {
  385.                         err("could not restart the USB controller CPU.");
  386.                         ret = -EINVAL;
  387.                 }
  388.                 if (ret || tbsqboxs1_op_rw(dev, 0xa0, 0xe600, 0, &reset, 1,
  389.                                         TBSQBOX_WRITE_MSG) != 1) {
  390.                         err("could not restart the USB controller CPU.");
  391.                         ret = -EINVAL;
  392.                 }
  393.  
  394.                 msleep(100);
  395.                 kfree(p);
  396.         }
  397.         return ret;
  398. }
  399.  
  400. static struct dvb_usb_device_properties tbsqboxs1_properties = {
  401.         .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  402.         .usb_ctrl = DEVICE_SPECIFIC,
  403.         .firmware = "dvb-usb-tbsqbox-id2601.fw",
  404.         .size_of_priv = sizeof(struct tbsqboxs1_state),
  405.         .no_reconnect = 1,
  406.  
  407.         .i2c_algo = &tbsqboxs1_i2c_algo,
  408.         .rc.legacy = {
  409.                 .rc_map_table = tbsqboxs1_rc_keys,
  410.                 .rc_map_size = ARRAY_SIZE(tbsqboxs1_rc_keys),
  411.                 .rc_interval = 150,
  412.                 .rc_query = tbsqboxs1_rc_query,
  413.         },
  414.  
  415.         .generic_bulk_ctrl_endpoint = 0x81,
  416.         /* parameter for the MPEG2-data transfer */
  417.         .num_adapters = 1,
  418.         .download_firmware = tbsqboxs1_load_firmware,
  419.         .read_mac_address = tbsqboxs1_read_mac_address,
  420.                 .adapter = {
  421.                 {
  422.                         .frontend_attach = tbsqboxs1_frontend_attach,
  423.                         .streaming_ctrl = NULL,
  424.                         .tuner_attach = tbsqboxs1_earda_tuner_attach,
  425.                         .stream = {
  426.                                 .type = USB_BULK,
  427.                                 .count = 8,
  428.                                 .endpoint = 0x82,
  429.                                 .u = {
  430.                                         .bulk = {
  431.                                                 .buffersize = 4096,
  432.                                         }
  433.                                 }
  434.                         },
  435.                 }
  436.         },
  437.         .num_device_descs = 1,
  438.         .devices = {
  439.                 {"TBS QBOX DVBS USB2.0",
  440.                         {&tbsqboxs1_table[1], NULL},
  441.                         {NULL},
  442.                 }
  443.         }
  444. };
  445.  
  446. static int tbsqboxs1_probe(struct usb_interface *intf,
  447.                 const struct usb_device_id *id)
  448. {
  449.         if (0 == dvb_usb_device_init(intf, &tbsqboxs1_properties,
  450.                         THIS_MODULE, NULL, adapter_nr)) {
  451.                 return 0;
  452.         }
  453.         return -ENODEV;
  454. }
  455.  
  456. static struct usb_driver tbsqboxs1_driver = {
  457.         .name = "tbsqboxs1",
  458.         .probe = tbsqboxs1_probe,
  459.         .disconnect = dvb_usb_device_exit,
  460.         .id_table = tbsqboxs1_table,
  461. };
  462.  
  463. static int __init tbsqboxs1_module_init(void)
  464. {
  465.         int ret =  usb_register(&tbsqboxs1_driver);
  466.         if (ret)
  467.                 err("usb_register failed. Error number %d", ret);
  468.  
  469.         return ret;
  470. }
  471.  
  472. static void __exit tbsqboxs1_module_exit(void)
  473. {
  474.         usb_deregister(&tbsqboxs1_driver);
  475. }
  476.  
  477. module_init(tbsqboxs1_module_init);
  478. module_exit(tbsqboxs1_module_exit);
  479.  
  480. MODULE_AUTHOR("Bob Liu <Bob@turbosight.com>");
  481. MODULE_DESCRIPTION("Driver for TBS QBOX");
  482. MODULE_VERSION("0.1");
  483. MODULE_LICENSE("GPL");
RAW Paste Data
Top