123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365 |
- /*
- * DVB USB Linux driver for the HDIC receiver
- *
- * Copyright (C) 2011 Metropolia University of Applied Sciences, Electria R&D
- *
- * Author: Antti Palosaari <crope@iki.fi>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
- */
- #include "hdic.h"
- #include "hd29l2.h"
- #include "mxl5007t.h"
- /* debug */
- static int dvb_usb_hdic_debug;
- module_param_named(debug, dvb_usb_hdic_debug, int, 0644);
- MODULE_PARM_DESC(debug, "set debugging level" DVB_USB_DEBUG_STATUS);
- DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr);
- /*
- * I2C addresses (7bit) found by probing I2C bus:
- * 0x48 ??
- * 0x51 eeprom
- * 0x60 MaxLinear MXL5007T tuner
- * 0x73 HDIC HD29L2 demod
- *
- * Xtals:
- * 24.000 MHz Cypress CY7C68013A-56 (FX2)
- * 30.400 MHz HDIC HD29L2
- * 24.000 MHz MaxLinear MXL5007T
- *
- * I/Os:
- * RDY1 / SLWR == TS_CLK (USB_SLWR = !TS_CLK&TS_VALID)
- * PA1 / INT1 == 29L1_RESET RST_N
- */
- /*
- * See Qanu DVB-T USB2.0 communication protocol specification for more
- * information used USB API.
- */
- /* I2C */
- static int hdic_i2c_xfer(struct i2c_adapter *adap, struct i2c_msg msg[],
- int num)
- {
- int ret;
- struct dvb_usb_device *d = i2c_get_adapdata(adap);
- u8 buf[64];
- /*
- * increase sleep when there is a lot of errors:
- * dvb-usb: recv bulk message failed: -110
- */
- #define HDIC_I2C_SLEEP 1
- if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
- return -EAGAIN;
- if (num == 2 && !(msg[0].flags & I2C_M_RD) &&
- (msg[1].flags & I2C_M_RD)) {
- /* I2C write + read combination (typical register read) */
- buf[0] = HDIC_CMD_I2C;
- buf[1] = (msg[0].addr << 1); /* I2C write */
- buf[2] = msg[0].len;
- buf[3] = 1; /* no I2C stop => repeated start */
- memcpy(&buf[4], msg[0].buf, msg[0].len);
- ret = dvb_usb_generic_rw(d, buf, 4+msg[0].len, buf, 1,
- HDIC_I2C_SLEEP);
- if (ret)
- goto err;
- buf[0] = HDIC_CMD_I2C;
- buf[1] = (msg[1].addr << 1) | 0x01; /* I2C read */
- buf[2] = msg[1].len;
- buf[3] = 0; /* I2C stop */
- ret = dvb_usb_generic_rw(d, buf, 4, buf, 1+msg[1].len,
- HDIC_I2C_SLEEP);
- if (ret)
- goto err;
- memcpy(msg[1].buf, &buf[1], msg[1].len);
- } else if (num == 1 && !(msg[0].flags & I2C_M_RD)) {
- /* I2C write (typical register write) */
- buf[0] = HDIC_CMD_I2C;
- buf[1] = (msg[0].addr << 1); /* I2C write */
- buf[2] = msg[0].len;
- buf[3] = 0; /* I2C stop */
- memcpy(&buf[4], msg[0].buf, msg[0].len);
- ret = dvb_usb_generic_rw(d, buf, 4+msg[0].len, buf, 1,
- HDIC_I2C_SLEEP);
- if (ret)
- goto err;
- } else {
- ret = -EOPNOTSUPP;
- goto err;
- }
- usleep_range(100, 1000);
- mutex_unlock(&d->i2c_mutex);
- return num;
- err:
- deb_info("%s: failed=%d\n", __func__, ret);
- mutex_unlock(&d->i2c_mutex);
- return ret;
- }
- static u32 hdic_i2c_func(struct i2c_adapter *adapter)
- {
- return I2C_FUNC_I2C;
- }
- static struct i2c_algorithm hdic_i2c_algo = {
- .master_xfer = hdic_i2c_xfer,
- .functionality = hdic_i2c_func,
- };
- /* Callbacks for DVB USB */
- static int hdic_power_ctrl(struct dvb_usb_device *d, int enable)
- {
- u8 sbuf[] = { HDIC_CMD_SLEEP_MODE, enable ? 0 : 1 };
- u8 rbuf[1];
- deb_info("%s: enable=%d\n", __func__, enable);
- return dvb_usb_generic_rw(d, sbuf, sizeof(sbuf), rbuf, sizeof(rbuf), 0);
- }
- static int hdic_streaming_ctrl(struct dvb_usb_adapter *adap, int enable)
- {
- u8 sbuf[] = { HDIC_CMD_CONTROL_STREAM_TRANSFER, enable };
- u8 rbuf[1];
- deb_info("%s: enable=%d\n", __func__, enable);
- return dvb_usb_generic_rw(adap->dev, sbuf, sizeof(sbuf), rbuf,
- sizeof(rbuf), 0);
- }
- /* general callback */
- static int hdic_frontend_callback(void *priv, int component, int cmd, int arg)
- {
- int ret;
- struct dvb_frontend *fe = priv;
- struct dvb_usb_adapter *adap = fe->dvb->priv;
- u8 sbuf[2] = { HDIC_CMD_DEMOD_RESET };
- u8 rbuf[1];
- deb_info("%s:\n", __func__);
- /* enable demod reset */
- sbuf[1] = 1;
- ret = dvb_usb_generic_rw(adap->dev, sbuf, sizeof(sbuf),
- rbuf, sizeof(rbuf), 0);
- if (ret)
- deb_info("%s: failed enable demod reset\n", __func__);
- usleep_range(100, 10000);
- /* disable demod reset */
- sbuf[1] = 0;
- ret = dvb_usb_generic_rw(adap->dev, sbuf, sizeof(sbuf), rbuf,
- sizeof(rbuf), 0);
- if (ret)
- deb_info("%s: failed disable demod reset\n", __func__);
- return 0;
- }
- static struct hd29l2_config hdic_hd29l2_config = {
- .i2c_addr = 0x73,
- .tuner_i2c_addr = 0x60,
- .ts_mode = HD29L2_TS_PARALLEL,
- };
- static int hdic_frontend_attach(struct dvb_usb_adapter *adap)
- {
- int ret;
- u8 sbuf[2];
- u8 rbuf[3];
- deb_info("%s:\n", __func__);
- /* wake-up device */
- sbuf[0] = HDIC_CMD_GET_FIRMWARE_VERSION;
- ret = dvb_usb_generic_rw(adap->dev, sbuf, sizeof(sbuf), rbuf,
- sizeof(rbuf), 0);
- if (ret)
- deb_info("%s: failed wake-up\n", __func__);
- /* disable demod reset */
- sbuf[0] = HDIC_CMD_DEMOD_RESET;
- sbuf[1] = 0;
- ret = dvb_usb_generic_rw(adap->dev, sbuf, sizeof(sbuf), rbuf,
- sizeof(rbuf), 0);
- if (ret)
- deb_info("%s: failed disable demod reset\n", __func__);
- /* attach demod */
- adap->fe_adap[0].fe = dvb_attach(hd29l2_attach, &hdic_hd29l2_config,
- &adap->dev->i2c_adap);
- if (adap->fe_adap[0].fe == NULL) {
- ret = -ENODEV;
- goto err;
- }
- /* setup the reset callback */
- adap->fe_adap[0].fe->callback = hdic_frontend_callback;
- return 0;
- err:
- deb_info("%s: failed=%d\n", __func__, ret);
- return ret;
- }
- static struct mxl5007t_config hdic_mxl5007t_config = {
- .xtal_freq_hz = MxL_XTAL_24_MHZ,
- .if_freq_hz = MxL_IF_36_15_MHZ,
- .invert_if = 1,
- };
- static int hdic_tuner_attach(struct dvb_usb_adapter *adap)
- {
- int ret;
- deb_info("%s:\n", __func__);
- if (dvb_attach(mxl5007t_attach, adap->fe_adap[0].fe,
- &adap->dev->i2c_adap, 0x60, &hdic_mxl5007t_config) == NULL) {
- ret = -ENODEV;
- goto err;
- }
- return 0;
- err:
- deb_info("%s: failed=%d\n", __func__, ret);
- return ret;
- }
- /* DVB USB Driver stuff */
- static struct dvb_usb_device_properties hdic_properties;
- static int hdic_probe(struct usb_interface *intf,
- const struct usb_device_id *id)
- {
- return dvb_usb_device_init(intf, &hdic_properties, THIS_MODULE, NULL,
- adapter_nr);
- }
- /*
- * 04b4:1004 HDIC development board firmware
- * 04b4:8613 CY7C68013 EZ-USB FX2 USB 2.0 Development Kit
- */
- static struct usb_device_id hdic_id[] = {
- #define HDIC_8613 0 /* CY7C68013 EZ-USB FX2 USB 2.0 Development Kit */
- #define HDIC_1004 1 /* HDIC 04b4:1004 */
- #define HDIC_LINUX 2 /* HDIC Linux custom firmware */
- [HDIC_8613] = {USB_DEVICE(USB_VID_CYPRESS, 0x8613)},
- [HDIC_1004] = {USB_DEVICE(USB_VID_CYPRESS, 0x1004)},
- [HDIC_LINUX] = {USB_DEVICE(USB_VID_CYPRESS, 0x1e04)},
- {} /* terminating entry */
- };
- MODULE_DEVICE_TABLE(usb, hdic_id);
- static struct dvb_usb_device_properties hdic_properties = {
- .caps = DVB_USB_IS_AN_I2C_ADAPTER,
- .usb_ctrl = CYPRESS_FX2,
- .firmware = "dvb-usb-hdic.fw",
- .num_adapters = 1,
- .adapter = {
- {
- .num_frontends = 1,
- .fe = {
- {
- .streaming_ctrl = hdic_streaming_ctrl,
- .frontend_attach = hdic_frontend_attach,
- .tuner_attach = hdic_tuner_attach,
- .stream = {
- .type = USB_BULK,
- .count = 5,
- .endpoint = 0x02,
- .u = {
- .bulk = {
- .buffersize =
- (4*512),
- }
- }
- },
- }
- },
- }
- },
- .power_ctrl = hdic_power_ctrl,
- .i2c_algo = &hdic_i2c_algo,
- .generic_bulk_ctrl_endpoint = 1,
- .num_device_descs = 1,
- .devices = {
- {
- .name = "HDIC reference design",
- .cold_ids = {&hdic_id[HDIC_8613],
- &hdic_id[HDIC_1004], NULL},
- .warm_ids = {&hdic_id[HDIC_LINUX], NULL},
- },
- }
- };
- static struct usb_driver hdic_driver = {
- .name = "dvb_usb_hdic",
- .probe = hdic_probe,
- .disconnect = dvb_usb_device_exit,
- .id_table = hdic_id,
- };
- /* module stuff */
- static int __init hdic_module_init(void)
- {
- int ret;
- deb_info("%s:\n", __func__);
- ret = usb_register(&hdic_driver);
- if (ret)
- err("module init failed=%d", ret);
- return ret;
- }
- static void __exit hdic_module_exit(void)
- {
- deb_info("%s:\n", __func__);
- /* deregister this driver from the USB subsystem */
- usb_deregister(&hdic_driver);
- }
- module_init(hdic_module_init);
- module_exit(hdic_module_exit);
- MODULE_AUTHOR("Antti Palosaari <crope@iki.fi>");
- MODULE_DESCRIPTION("HDIC DMB-TH reference design USB2.0 driver (custom firmware)");
- MODULE_LICENSE("GPL");
|