summaryrefslogtreecommitdiff
path: root/linux/drivers/media/dvb/frontends/nxt6000.c
diff options
context:
space:
mode:
Diffstat (limited to 'linux/drivers/media/dvb/frontends/nxt6000.c')
-rw-r--r--linux/drivers/media/dvb/frontends/nxt6000.c417
1 files changed, 417 insertions, 0 deletions
diff --git a/linux/drivers/media/dvb/frontends/nxt6000.c b/linux/drivers/media/dvb/frontends/nxt6000.c
new file mode 100644
index 000000000..984f065f7
--- /dev/null
+++ b/linux/drivers/media/dvb/frontends/nxt6000.c
@@ -0,0 +1,417 @@
+/*
+
+ NxtWave Communications - NXT6000 demodulator driver
+
+ This driver currently supports:
+
+ Alps TDME7 (Tuner: MITEL SP5659)
+ Alps TDED4 (Tuner: TI ALP510, external Nxt6000)
+
+ Copyright (C) 2002-2003 Florian Schirmer <schirmer@taytron.net>
+
+ 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 <linux/module.h>
+#include <linux/init.h>
+#include <linux/delay.h>
+#include <linux/slab.h>
+#include <linux/poll.h>
+#include <asm/io.h>
+#include <linux/i2c.h>
+
+#include "dvb_frontend.h"
+#include "nxt6000.h"
+
+MODULE_DESCRIPTION("NxtWave NXT6000 DVB demodulator driver");
+MODULE_AUTHOR("Florian Schirmer");
+MODULE_LICENSE("GPL");
+
+static struct dvb_frontend_info nxt6000_info = {
+
+ .name = "NxtWave NXT6000",
+ .type = FE_OFDM,
+ .frequency_min = 48250000,
+ .frequency_max = 863250000,
+ .frequency_stepsize = 62500,
+ /*.frequency_tolerance = */ /* FIXME: 12% of SR */
+ .symbol_rate_min = 0, /* FIXME */
+ .symbol_rate_max = 9360000, /* FIXME */
+ .symbol_rate_tolerance = 4000,
+ .notifier_delay = 0,
+ .caps = FE_CAN_FEC_1_2 | FE_CAN_FEC_2_3 | FE_CAN_FEC_3_4 |
+ FE_CAN_FEC_4_5 | FE_CAN_FEC_5_6 | FE_CAN_FEC_6_7 |
+ FE_CAN_FEC_7_8 | FE_CAN_FEC_8_9 | FE_CAN_FEC_AUTO |
+ FE_CAN_QAM_16 | FE_CAN_QAM_64 | FE_CAN_QAM_AUTO |
+ FE_CAN_TRANSMISSION_MODE_AUTO |
+// FE_CAN_BANDWIDTH_AUTO |
+ FE_CAN_GUARD_INTERVAL_AUTO |
+ FE_CAN_HIERARCHY_AUTO,
+
+};
+
+static int nxt6000_writereg(struct dvb_i2c_bus *i2c, u8 reg, u8 data)
+{
+
+ u8 buf[] = {reg, data};
+ struct i2c_msg msg = {addr: 0x14 >> 1, flags: 0, buf: buf, len: 2};
+ int ret;
+
+ if ((ret = i2c->xfer(i2c, &msg, 1)) != 1)
+ printk("nxt6000: nxt6000_write error (reg: 0x%02X <- data: 0x%02X)\n", reg, data);
+
+ return (ret != 1) ? -EFAULT : 0;
+
+}
+
+static u8 nxt6000_readreg(struct dvb_i2c_bus *i2c, u8 reg)
+{
+ int ret;
+
+ u8 b0[] = {reg};
+ u8 b1[] = {0};
+ struct i2c_msg msgs[] = {{addr: 0x14 >> 1, flags: 0, buf: b0, len: 1},
+ {addr: 0x14 >> 1, flags: I2C_M_RD, buf: b1, len: 1}};
+
+ ret = i2c->xfer(i2c, msgs, 2);
+
+ if (ret != 2)
+ printk("nxt6000: nxt6000_read error (reg: 0x%02X, ret=%d)\n", reg, ret);
+
+ return b1[0];
+
+}
+
+static int pll_write(struct dvb_i2c_bus *i2c, u8 addr, u8 *buf, u8 len)
+{
+
+ struct i2c_msg msg = {addr: addr >> 1, flags: 0, buf: buf, len: len};
+ int ret;
+
+ nxt6000_writereg(i2c, ENABLE_TUNER_IIC, 0x01); /* open i2c bus switch */
+ ret = i2c->xfer(i2c, &msg, 1);
+ nxt6000_writereg(i2c, ENABLE_TUNER_IIC, 0x00); /* close i2c bus switch */
+
+ if (ret != 1)
+ printk("nxt6000: pll_write error %d\n", ret);
+
+ return (ret != 1) ? -EFAULT : 0;
+
+}
+
+#define FREQ2DIV(freq) ((freq + 36166667) / 166667)
+
+static int sp5659_set_tv_freq(struct dvb_i2c_bus *i2c, u32 freq)
+{
+
+ u8 buf[4];
+
+ buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F;
+ buf[1] = FREQ2DIV(freq) & 0xFF;
+ buf[2] = (((FREQ2DIV(freq) >> 15) & 0x03) << 5) | 0x85;
+
+ if ((freq >= 174000000) && (freq < 230000000))
+ buf[3] = 0x82;
+ else if ((freq >= 470000000) && (freq < 782000000))
+ buf[3] = 0x85;
+ else if ((freq >= 782000000) && (freq < 863000000))
+ buf[3] = 0xC5;
+ else
+ return -EINVAL;
+
+ return pll_write(i2c, 0xC2, buf, 4);
+
+}
+
+static int alp510_set_tv_freq(struct dvb_i2c_bus *i2c, u32 freq)
+{
+
+ u8 buf[4];
+
+ buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F;
+ buf[1] = FREQ2DIV(freq) & 0xFF;
+ buf[2] = 0x85;
+
+ if ((freq >= 47000000) && (freq < 153000000))
+ buf[3] = 0x01;
+ else if ((freq >= 153000000) && (freq < 430000000))
+ buf[3] = 0x02;
+ else if ((freq >= 430000000) && (freq < 824000000))
+ buf[3] = 0x08;
+ else if ((freq >= 824000000) && (freq < 863000000))
+ buf[3] = 0x88;
+ else
+ return -EINVAL;
+
+ // Enable 8K SAW BW mode
+ buf[3] |= (1 << 2);
+
+ return pll_write(i2c, 0xC0, buf, 4);
+
+}
+
+#if 0
+static int SetInversion(struct dvb_i2c_bus *i2c, int inversion)
+{
+ u8 val;
+
+ if (inversion == stv->inv)
+ return 0;
+ stv->inv=inversion;
+ switch (inversion) {
+ case INVERSION_AUTO: // Hmm, return EINVAL or leave it like this?
+ default:
+ case INVERSION_OFF:
+ val=0x01;
+ break;
+ case INVERSION_ON:
+ val=0x00;
+ break;
+ }
+ writereg(client, 0x0c,
+ (readreg(client, 0x0c)&0xfe)|val);
+ return 0;
+}
+#endif
+
+static void nxt6000_reset(struct dvb_i2c_bus *i2c)
+{
+
+ nxt6000_writereg(i2c, RS_COR_SYNC_PARAM, SYNC_PARAM);
+ nxt6000_writereg(i2c, BER_CTRL, /*(1 << 2) |*/ (1 << 1) | 1);
+ nxt6000_writereg(i2c, VIT_COR_CTL, (1 << 7) | (1 << 1));
+// VIT_BER_TIMER_?
+ nxt6000_writereg(i2c, OFDM_COR_CTL, (1 << 5) | (nxt6000_readreg(i2c, OFDM_COR_CTL) & 0x0F));
+ // Auto mode with MODE8K & 1/8 guard as default
+ nxt6000_writereg(i2c, OFDM_COR_MODEGUARD, (1 << 2) | 2);
+// OFDM_AGC_CTL tune zap time
+ nxt6000_writereg(i2c, OFDM_ITB_FREQ_1, 0x06);
+ nxt6000_writereg(i2c, OFDM_ITB_FREQ_2, 0x31);
+// CAS_FREQ
+ nxt6000_writereg(i2c, OFDM_SYR_CTL, 1 << 2);
+ nxt6000_writereg(i2c, OFDM_PPM_CTL_1, PPM256);
+ nxt6000_writereg(i2c, OFDM_TRL_NOMINALRATE_1, 0x49);
+ nxt6000_writereg(i2c, OFDM_TRL_NOMINALRATE_2, 0x72);
+ nxt6000_writereg(i2c, ANALOG_CONTROL_0, 1 << 5);
+ nxt6000_writereg(i2c, EN_DMD_RACQ, (1 << 7) | (3 << 4) | 2);
+ nxt6000_writereg(i2c, DIAG_CONFIG, TB_SET);
+ nxt6000_writereg(i2c, SUB_DIAG_MODE_SEL, 0);
+ nxt6000_writereg(i2c, SUB_DIAG_MODE_SEL, CLKINVERSION);
+ nxt6000_writereg(i2c, TS_FORMAT, 0);
+// nxt6000_writereg(i2c, TS_FORMAT, GATED_CLOCK);
+
+}
+
+#if 1
+static void nxt6000_dump_status(struct dvb_i2c_bus *i2c)
+{
+
+ printk("RS_COR_STAT: 0x%02X\n", nxt6000_readreg(i2c, RS_COR_STAT));
+ printk("VIT_SYNC_STATUS: 0x%02X\n", nxt6000_readreg(i2c, VIT_SYNC_STATUS));
+ printk("OFDM_COR_STAT: 0x%02X\n", nxt6000_readreg(i2c, OFDM_COR_STAT));
+ printk("OFDM_SYR_STAT: 0x%02X\n", nxt6000_readreg(i2c, OFDM_SYR_STAT));
+ printk("OFDM_TPS_RCVD_1: 0x%02X\n", nxt6000_readreg(i2c, OFDM_TPS_RCVD_1));
+// printk("OFDM_TPS_RCVD_2: 0x%02X\n", nxt6000_readreg(i2c, OFDM_TPS_RCVD_2));
+// printk("OFDM_TPS_RCVD_3: 0x%02X\n", nxt6000_readreg(i2c, OFDM_TPS_RCVD_3));
+// printk("OFDM_TPS_RCVD_4: 0x%02X\n", nxt6000_readreg(i2c, OFDM_TPS_RCVD_4));
+// printk("OFDM_TPS_RESERVED_1: 0x%02X\n", nxt6000_readreg(i2c, OFDM_TPS_RESERVED_1));
+// printk("OFDM_TPS_RESERVED_2: 0x%02X\n", nxt6000_readreg(i2c, OFDM_TPS_RESERVED_2));
+// printk("RF_AGC_STATUS: 0x%02X\n", nxt6000_readreg(i2c, RF_AGC_STATUS));
+
+}
+#endif
+
+static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg)
+{
+
+ switch (cmd) {
+
+ case FE_GET_INFO:
+
+ memcpy(arg, &nxt6000_info, sizeof (struct dvb_frontend_info));
+
+ return 0;
+
+ case FE_READ_STATUS:
+ {
+ fe_status_t *status = (fe_status_t *)arg;
+#if 0
+ u8 core_status;
+ nxt6000_dump_status(fe->i2c);
+
+ *status = 0;
+
+ core_status = nxt6000_readreg(fe->i2c, OFDM_COR_STAT);
+
+ if (core_status & AGCLOCKED)
+ *status |= FE_HAS_SIGNAL;
+
+ if (nxt6000_readreg(fe->i2c, OFDM_SYR_STAT) & GI14_SYR_LOCK)
+ *status |= FE_HAS_CARRIER;
+
+ if (nxt6000_readreg(fe->i2c, VIT_SYNC_STATUS) & VITINSYNC)
+ *status |= FE_HAS_VITERBI;
+
+ if (nxt6000_readreg(fe->i2c, RS_COR_STAT) & RSCORESTATUS)
+ *status |= FE_HAS_SYNC;
+
+ if (core_status & TPSLOCKED)
+ *status |= FE_HAS_LOCK;
+#endif
+ *status = FE_HAS_LOCK;
+
+ return 0;
+
+ }
+
+ case FE_READ_BER:
+ {
+ u32 *ber = (u32 *)arg;
+
+ *ber=0;
+
+ return 0;
+
+ }
+
+ case FE_READ_SIGNAL_STRENGTH:
+ {
+// s16 *signal = (s16 *)arg;
+
+// *signal=(((signed char)readreg(client, 0x16))+128)<<8;
+
+ return 0;
+
+ }
+
+ case FE_READ_SNR:
+ {
+// s16 *snr = (s16 *)arg;
+
+// *snr=readreg(client, 0x24)<<8;
+// *snr|=readreg(client, 0x25);
+
+ break;
+ }
+
+ case FE_READ_UNCORRECTED_BLOCKS:
+ {
+ u32 *ublocks = (u32 *)arg;
+
+ *ublocks = 0;
+
+ break;
+ }
+
+ case FE_INIT:
+ case FE_RESET:
+
+ nxt6000_reset(fe->i2c);
+
+ break;
+
+ case FE_SET_FRONTEND:
+ {
+ struct dvb_frontend_parameters *param = (struct dvb_frontend_parameters *)arg;
+
+ //SetInversion(client, param->Inversion);
+ //SetFEC(client, param->u.qpsk.FEC_inner);
+ //SetSymbolrate(client, param->u.qpsk.SymbolRate);
+ alp510_set_tv_freq(fe->i2c, param->frequency);
+
+ break;
+ }
+
+ default:
+
+ return -EOPNOTSUPP;
+
+ }
+
+ return 0;
+
+}
+
+static int nxt6000_attach(struct dvb_i2c_bus *i2c)
+{
+
+ u8 core_rev;
+
+ printk("nxt6000: attach\n");
+
+ if ((core_rev = nxt6000_readreg(i2c, OFDM_MSC_REV)) != NXT6000ASICDEVICE) {
+
+ printk("nxt6000: core revision is wrong (0x%02X != 0x0B)\n", core_rev);
+
+ return -ENODEV;
+
+ }
+
+ if (pll_write(i2c, 0xC0, NULL, 0) == 0) {
+
+ printk("nxt6000: detected TI ALP510 tuner\n");
+
+ } else if (pll_write(i2c, 0xC2, NULL, 0) == 0) {
+
+ printk("nxt6000: detected MITEL SP5659 tuner\n");
+
+ } else {
+
+ printk("nxt6000: unable to detect tuner\n");
+
+ return -EINVAL;
+
+ }
+
+ printk("nxt6000: attached at %d:%d\n", i2c->adapter->num, i2c->id);
+
+ nxt6000_reset(i2c);
+ alp510_set_tv_freq(i2c, 658000000);
+ nxt6000_dump_status(i2c);
+
+ return dvb_register_frontend(nxt6000_ioctl, i2c, NULL, &nxt6000_info);
+
+}
+
+static void nxt6000_detach(struct dvb_i2c_bus *i2c)
+{
+
+ printk("nxt6000: detach\n");
+
+ dvb_unregister_frontend(nxt6000_ioctl, i2c);
+
+}
+
+static __init int nxt6000_init(void)
+{
+
+ printk("nxt6000: init\n");
+
+ return dvb_register_i2c_device(THIS_MODULE, nxt6000_attach, nxt6000_detach);
+
+}
+
+static __exit void nxt6000_exit(void)
+{
+
+ printk("nxt6000: cleanup\n");
+
+ dvb_unregister_i2c_device(nxt6000_attach);
+
+}
+
+module_init(nxt6000_init);
+module_exit(nxt6000_exit);