summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--linux/drivers/media/dvb/frontends/nxt6000.c359
1 files changed, 229 insertions, 130 deletions
diff --git a/linux/drivers/media/dvb/frontends/nxt6000.c b/linux/drivers/media/dvb/frontends/nxt6000.c
index 5822f262d..c6b46a943 100644
--- a/linux/drivers/media/dvb/frontends/nxt6000.c
+++ b/linux/drivers/media/dvb/frontends/nxt6000.c
@@ -44,7 +44,7 @@ static struct dvb_frontend_info nxt6000_info = {
.name = "NxtWave NXT6000",
.type = FE_OFDM,
- .frequency_min = 48250000,
+ .frequency_min = 0,
.frequency_max = 863250000,
.frequency_stepsize = 62500,
/*.frequency_tolerance = */ /* FIXME: 12% of SR */
@@ -57,53 +57,87 @@ static struct dvb_frontend_info nxt6000_info = {
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)
+#pragma pack(1)
+
+struct nxt6000_config {
+
+ u8 demod_addr;
+ u8 tuner_addr;
+ u8 tuner_type;
+ u8 reserved;
+
+};
+
+#pragma pack()
+
+#define FE2NXT(fe) ((struct nxt6000_config *)&(fe->data))
+#define FREQ2DIV(freq) ((freq + 36166667) / 166667)
+
+
+static int nxt6000_write(struct dvb_i2c_bus *i2c, u8 addr, u8 reg, u8 data)
{
u8 buf[] = {reg, data};
- struct i2c_msg msg = {addr: 0x18 >> 1, flags: 0, buf: buf, len: 2};
+ struct i2c_msg msg = {addr: addr >> 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);
+ printk("nxt6000: nxt6000_write error (addr: 0x%02X, reg: 0x%02X, data: 0x%02X, ret: %d)\n", addr, reg, data, ret);
return (ret != 1) ? -EFAULT : 0;
}
-static u8 nxt6000_readreg(struct dvb_i2c_bus *i2c, u8 reg)
+static u8 nxt6000_writereg(struct dvb_frontend *fe, u8 reg, u8 data)
{
- int ret;
+ struct nxt6000_config *nxt = FE2NXT(fe);
+
+ return nxt6000_write(fe->i2c, nxt->demod_addr, reg, data);
+
+}
+
+static u8 nxt6000_read(struct dvb_i2c_bus *i2c, u8 addr, u8 reg)
+{
+
+ int ret;
u8 b0[] = {reg};
u8 b1[] = {0};
- struct i2c_msg msgs[] = {{addr: 0x18 >> 1, flags: 0, buf: b0, len: 1},
- {addr: 0x18 >> 1, flags: I2C_M_RD, buf: b1, len: 1}};
+ struct i2c_msg msgs[] = {{addr: addr >> 1, flags: 0, buf: b0, len: 1},
+ {addr: addr >> 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);
+ printk("nxt6000: nxt6000_read error (addr: 0x%02X, reg: 0x%02X, ret: %d)\n", addr, reg, ret);
return b1[0];
}
-static int pll_write(struct dvb_i2c_bus *i2c, u8 addr, u8 *buf, u8 len)
+static u8 nxt6000_readreg(struct dvb_frontend *fe, u8 reg)
{
- struct i2c_msg msg = {addr: addr >> 1, flags: 0, buf: buf, len: len};
+ struct nxt6000_config *nxt = FE2NXT(fe);
+
+ return nxt6000_read(fe->i2c, nxt->demod_addr, reg);
+
+}
+
+static int pll_write(struct dvb_i2c_bus *i2c, u8 demod_addr, u8 tuner_addr, u8 *buf, u8 len)
+{
+
+ struct i2c_msg msg = {addr: tuner_addr >> 1, flags: 0, buf: buf, len: len};
int ret;
- nxt6000_writereg(i2c, ENABLE_TUNER_IIC, 0x01); /* open i2c bus switch */
+ nxt6000_write(i2c, demod_addr, 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 */
+ nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x00); /* close i2c bus switch */
if (ret != 1)
printk("nxt6000: pll_write error %d\n", ret);
@@ -112,12 +146,11 @@ static int pll_write(struct dvb_i2c_bus *i2c, u8 addr, u8 *buf, u8 len)
}
-#define FREQ2DIV(freq) ((freq + 36166667) / 166667)
-
-static int sp5659_set_tv_freq(struct dvb_i2c_bus *i2c, u32 freq)
+static int sp5659_set_tv_freq(struct dvb_frontend *fe, u32 freq)
{
u8 buf[4];
+ struct nxt6000_config *nxt = FE2NXT(fe);
buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F;
buf[1] = FREQ2DIV(freq) & 0xFF;
@@ -132,14 +165,15 @@ static int sp5659_set_tv_freq(struct dvb_i2c_bus *i2c, u32 freq)
else
return -EINVAL;
- return pll_write(i2c, 0xC2, buf, 4);
+ return pll_write(fe->i2c, nxt->demod_addr, nxt->tuner_addr, buf, 4);
}
-static int alp510_set_tv_freq(struct dvb_i2c_bus *i2c, u32 freq)
+static int alp510_set_tv_freq(struct dvb_frontend *fe, u32 freq)
{
u8 buf[4];
+ struct nxt6000_config *nxt = FE2NXT(fe);
buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F;
buf[1] = FREQ2DIV(freq) & 0xFF;
@@ -169,43 +203,82 @@ static int alp510_set_tv_freq(struct dvb_i2c_bus *i2c, u32 freq)
return -EINVAL;
#endif
- return pll_write(i2c, 0xC0, buf, 4);
+ return pll_write(fe->i2c, nxt->demod_addr, nxt->tuner_addr, buf, 4);
}
-static void nxt6000_reset(struct dvb_i2c_bus *i2c)
+static void nxt6000_reset(struct dvb_frontend *fe)
{
u8 val;
- val = nxt6000_readreg(i2c, OFDM_COR_CTL);
+ val = nxt6000_readreg(fe, OFDM_COR_CTL);
- nxt6000_writereg(i2c, OFDM_COR_CTL, val & ~COREACT);
- nxt6000_writereg(i2c, OFDM_COR_CTL, val | COREACT);
+ nxt6000_writereg(fe, OFDM_COR_CTL, val & ~COREACT);
+ nxt6000_writereg(fe, OFDM_COR_CTL, val | COREACT);
}
-static int nxt6000_set_guard_interval(struct dvb_i2c_bus *i2c, fe_guard_interval_t guard_interval)
+static int nxt6000_set_bandwidth(struct dvb_frontend *fe, fe_bandwidth_t bandwidth)
+{
+
+ u16 nominal_rate;
+ int result;
+
+ switch(bandwidth) {
+
+ case BANDWIDTH_6_MHZ:
+
+ nominal_rate = 0x55B7;
+
+ break;
+
+ case BANDWIDTH_7_MHZ:
+
+ nominal_rate = 0x6400;
+
+ break;
+
+ case BANDWIDTH_8_MHZ:
+
+ nominal_rate = 0x7249;
+
+ break;
+
+ default:
+
+ return -EINVAL;
+
+ }
+
+ if ((result = nxt6000_writereg(fe, OFDM_TRL_NOMINALRATE_1, nominal_rate & 0xFF)) < 0)
+ return result;
+
+ return nxt6000_writereg(fe, OFDM_TRL_NOMINALRATE_2, (nominal_rate >> 8) & 0xFF);
+
+}
+
+static int nxt6000_set_guard_interval(struct dvb_frontend *fe, fe_guard_interval_t guard_interval)
{
switch(guard_interval) {
case GUARD_INTERVAL_1_32:
- return nxt6000_writereg(i2c, OFDM_COR_MODEGUARD, 0x00 | (nxt6000_readreg(i2c, OFDM_COR_MODEGUARD) & ~0x03));
+ return nxt6000_writereg(fe, OFDM_COR_MODEGUARD, 0x00 | (nxt6000_readreg(fe, OFDM_COR_MODEGUARD) & ~0x03));
case GUARD_INTERVAL_1_16:
- return nxt6000_writereg(i2c, OFDM_COR_MODEGUARD, 0x01 | (nxt6000_readreg(i2c, OFDM_COR_MODEGUARD) & ~0x03));
+ return nxt6000_writereg(fe, OFDM_COR_MODEGUARD, 0x01 | (nxt6000_readreg(fe, OFDM_COR_MODEGUARD) & ~0x03));
case GUARD_INTERVAL_AUTO:
case GUARD_INTERVAL_1_8:
- return nxt6000_writereg(i2c, OFDM_COR_MODEGUARD, 0x02 | (nxt6000_readreg(i2c, OFDM_COR_MODEGUARD) & ~0x03));
+ return nxt6000_writereg(fe, OFDM_COR_MODEGUARD, 0x02 | (nxt6000_readreg(fe, OFDM_COR_MODEGUARD) & ~0x03));
case GUARD_INTERVAL_1_4:
- return nxt6000_writereg(i2c, OFDM_COR_MODEGUARD, 0x03 | (nxt6000_readreg(i2c, OFDM_COR_MODEGUARD) & ~0x03));
+ return nxt6000_writereg(fe, OFDM_COR_MODEGUARD, 0x03 | (nxt6000_readreg(fe, OFDM_COR_MODEGUARD) & ~0x03));
default:
@@ -215,18 +288,18 @@ static int nxt6000_set_guard_interval(struct dvb_i2c_bus *i2c, fe_guard_interval
}
-static int nxt6000_set_inversion(struct dvb_i2c_bus *i2c, fe_spectral_inversion_t inversion)
+static int nxt6000_set_inversion(struct dvb_frontend *fe, fe_spectral_inversion_t inversion)
{
switch(inversion) {
case INVERSION_OFF:
- return nxt6000_writereg(i2c, OFDM_ITB_CTL, 0x00);
+ return nxt6000_writereg(fe, OFDM_ITB_CTL, 0x00);
case INVERSION_ON:
- return nxt6000_writereg(i2c, OFDM_ITB_CTL, ITBINV);
+ return nxt6000_writereg(fe, OFDM_ITB_CTL, ITBINV);
default:
@@ -236,7 +309,7 @@ static int nxt6000_set_inversion(struct dvb_i2c_bus *i2c, fe_spectral_inversion_
}
-static int nxt6000_set_transmission_mode(struct dvb_i2c_bus *i2c, fe_transmit_mode_t transmission_mode)
+static int nxt6000_set_transmission_mode(struct dvb_frontend *fe, fe_transmit_mode_t transmission_mode)
{
int result;
@@ -245,18 +318,18 @@ static int nxt6000_set_transmission_mode(struct dvb_i2c_bus *i2c, fe_transmit_mo
case TRANSMISSION_MODE_2K:
- if ((result = nxt6000_writereg(i2c, EN_DMD_RACQ, 0x00 | (nxt6000_readreg(i2c, EN_DMD_RACQ) & ~0x03))) < 0)
+ if ((result = nxt6000_writereg(fe, EN_DMD_RACQ, 0x00 | (nxt6000_readreg(fe, EN_DMD_RACQ) & ~0x03))) < 0)
return result;
- return nxt6000_writereg(i2c, OFDM_COR_MODEGUARD, (0x00 << 2) | (nxt6000_readreg(i2c, OFDM_COR_MODEGUARD) & ~0x04));
+ return nxt6000_writereg(fe, OFDM_COR_MODEGUARD, (0x00 << 2) | (nxt6000_readreg(fe, OFDM_COR_MODEGUARD) & ~0x04));
case TRANSMISSION_MODE_8K:
case TRANSMISSION_MODE_AUTO:
- if ((result = nxt6000_writereg(i2c, EN_DMD_RACQ, 0x02 | (nxt6000_readreg(i2c, EN_DMD_RACQ) & ~0x03))) < 0)
+ if ((result = nxt6000_writereg(fe, EN_DMD_RACQ, 0x02 | (nxt6000_readreg(fe, EN_DMD_RACQ) & ~0x03))) < 0)
return result;
- return nxt6000_writereg(i2c, OFDM_COR_MODEGUARD, (0x01 << 2) | (nxt6000_readreg(i2c, OFDM_COR_MODEGUARD) & ~0x04));
+ return nxt6000_writereg(fe, OFDM_COR_MODEGUARD, (0x01 << 2) | (nxt6000_readreg(fe, OFDM_COR_MODEGUARD) & ~0x04));
default:
@@ -266,58 +339,56 @@ static int nxt6000_set_transmission_mode(struct dvb_i2c_bus *i2c, fe_transmit_mo
}
-static void nxt6000_setup(struct dvb_i2c_bus *i2c)
+static void nxt6000_setup(struct dvb_frontend *fe)
{
- 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));
- nxt6000_writereg(i2c, OFDM_COR_MODEGUARD, FORCEMODE8K | 2);
-// OFDM_AGC_CTL tune zap time
-// nxt6000_writereg(i2c, OFDM_AGC_CTL, );
- nxt6000_writereg(i2c, OFDM_ITB_FREQ_1, 0x06);
- nxt6000_writereg(i2c, OFDM_ITB_FREQ_2, 0x31);
- nxt6000_writereg(i2c, OFDM_CAS_CTL, (0x01 << 7) | (0x02 << 3) | 0x02);
- nxt6000_writereg(i2c, CAS_FREQ, 0xBB); // CHECK
- nxt6000_writereg(i2c, OFDM_SYR_CTL, 1 << 2);
- nxt6000_writereg(i2c, OFDM_PPM_CTL_1, PPM256);
- nxt6000_writereg(i2c, OFDM_TRL_NOMINALRATE_1, 0x49); // Hardcoded for 8K (fixme)
- nxt6000_writereg(i2c, OFDM_TRL_NOMINALRATE_2, 0x72); // Hardcoded for 8K (fixme)
- 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(fe, RS_COR_SYNC_PARAM, SYNC_PARAM);
+ nxt6000_writereg(fe, BER_CTRL, /*(1 << 2) |*/ (0x01 << 1) | 0x01);
+ nxt6000_writereg(fe, VIT_COR_CTL, VIT_COR_RESYNC);
+ nxt6000_writereg(fe, OFDM_COR_CTL, (0x01 << 5) | (nxt6000_readreg(fe, OFDM_COR_CTL) & 0x0F));
+ nxt6000_writereg(fe, OFDM_COR_MODEGUARD, FORCEMODE8K | 0x02);
+ nxt6000_writereg(fe, OFDM_AGC_CTL, AGCLAST | INITIAL_AGC_BW); // CHECKME
+ nxt6000_writereg(fe, OFDM_ITB_FREQ_1, 0x06);
+ nxt6000_writereg(fe, OFDM_ITB_FREQ_2, 0x31);
+ nxt6000_writereg(fe, OFDM_CAS_CTL, (0x01 << 7) | (0x02 << 3) | 0x04);
+ nxt6000_writereg(fe, CAS_FREQ, 0xBB); // CHECKME
+ nxt6000_writereg(fe, OFDM_SYR_CTL, 1 << 2);
+ nxt6000_writereg(fe, OFDM_PPM_CTL_1, PPM256);
+ nxt6000_writereg(fe, OFDM_TRL_NOMINALRATE_1, 0x49); // Hardcoded for 8MHz (fixme)
+ nxt6000_writereg(fe, OFDM_TRL_NOMINALRATE_2, 0x72); // Hardcoded for 8MHz (fixme)
+ nxt6000_writereg(fe, ANALOG_CONTROL_0, 1 << 5);
+ nxt6000_writereg(fe, EN_DMD_RACQ, (1 << 7) | (3 << 4) | 2);
+ nxt6000_writereg(fe, DIAG_CONFIG, TB_SET);
+// nxt6000_writereg(fe, SUB_DIAG_MODE_SEL, 0);
+ nxt6000_writereg(fe, SUB_DIAG_MODE_SEL, CLKINVERSION);
+ nxt6000_writereg(fe, TS_FORMAT, 0);
}
-static void nxt6000_dump_status(struct dvb_i2c_bus *i2c)
+static void nxt6000_dump_status(struct dvb_frontend *fe)
{
u8 val;
-// 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("RS_COR_STAT: 0x%02X\n", nxt6000_readreg(fe, RS_COR_STAT));
+// printk("VIT_SYNC_STATUS: 0x%02X\n", nxt6000_readreg(fe, VIT_SYNC_STATUS));
+// printk("OFDM_COR_STAT: 0x%02X\n", nxt6000_readreg(fe, OFDM_COR_STAT));
+// printk("OFDM_SYR_STAT: 0x%02X\n", nxt6000_readreg(fe, OFDM_SYR_STAT));
+// printk("OFDM_TPS_RCVD_1: 0x%02X\n", nxt6000_readreg(fe, OFDM_TPS_RCVD_1));
+// printk("OFDM_TPS_RCVD_2: 0x%02X\n", nxt6000_readreg(fe, OFDM_TPS_RCVD_2));
+// printk("OFDM_TPS_RCVD_3: 0x%02X\n", nxt6000_readreg(fe, OFDM_TPS_RCVD_3));
+// printk("OFDM_TPS_RCVD_4: 0x%02X\n", nxt6000_readreg(fe, OFDM_TPS_RCVD_4));
+// printk("OFDM_TPS_RESERVED_1: 0x%02X\n", nxt6000_readreg(fe, OFDM_TPS_RESERVED_1));
+// printk("OFDM_TPS_RESERVED_2: 0x%02X\n", nxt6000_readreg(fe, OFDM_TPS_RESERVED_2));
printk("NXT6000 status:");
- val = nxt6000_readreg(i2c, RS_COR_STAT);
+ val = nxt6000_readreg(fe, RS_COR_STAT);
printk(" DATA DESCR LOCK: %d,", val & 0x01);
printk(" DATA SYNC LOCK: %d,", (val >> 1) & 0x01);
- val = nxt6000_readreg(i2c, VIT_SYNC_STATUS);
+ val = nxt6000_readreg(fe, VIT_SYNC_STATUS);
printk(" VITERBI LOCK: %d,", (val >> 7) & 0x01);
@@ -357,14 +428,14 @@ static void nxt6000_dump_status(struct dvb_i2c_bus *i2c)
}
- val = nxt6000_readreg(i2c, OFDM_COR_STAT);
+ val = nxt6000_readreg(fe, OFDM_COR_STAT);
printk(" CHCTrack: %d,", (val >> 7) & 0x01);
printk(" TPSLock: %d,", (val >> 6) & 0x01);
printk(" SYRLock: %d,", (val >> 5) & 0x01);
printk(" AGCLock: %d,", (val >> 4) & 0x01);
- switch((val >> 4) & 0x07) {
+ switch(val & 0x0F) {
case 0x00:
@@ -412,7 +483,7 @@ static void nxt6000_dump_status(struct dvb_i2c_bus *i2c)
}
- val = nxt6000_readreg(i2c, OFDM_SYR_STAT);
+ val = nxt6000_readreg(fe, OFDM_SYR_STAT);
printk(" SYRLock: %d,", (val >> 4) & 0x01);
printk(" SYRMode: %s,", (val >> 2) & 0x01 ? "8K" : "2K");
@@ -445,7 +516,7 @@ static void nxt6000_dump_status(struct dvb_i2c_bus *i2c)
}
- val = nxt6000_readreg(i2c, OFDM_TPS_RCVD_3);
+ val = nxt6000_readreg(fe, OFDM_TPS_RCVD_3);
switch((val >> 4) & 0x07) {
@@ -519,7 +590,7 @@ static void nxt6000_dump_status(struct dvb_i2c_bus *i2c)
}
- val = nxt6000_readreg(i2c, OFDM_TPS_RCVD_4);
+ val = nxt6000_readreg(fe, OFDM_TPS_RCVD_4);
printk(" TPSMode: %s,", val & 0x01 ? "8K" : "2K");
@@ -552,9 +623,9 @@ static void nxt6000_dump_status(struct dvb_i2c_bus *i2c)
}
// Strange magic required to gain access to RF_AGC_STATUS
- nxt6000_readreg(i2c, RF_AGC_VAL_1);
- val = nxt6000_readreg(i2c, RF_AGC_STATUS);
- val = nxt6000_readreg(i2c, RF_AGC_STATUS);
+ nxt6000_readreg(fe, RF_AGC_VAL_1);
+ val = nxt6000_readreg(fe, RF_AGC_STATUS);
+ val = nxt6000_readreg(fe, RF_AGC_STATUS);
printk(" RF AGC LOCK: %d,", (val >> 4) & 0x01);
@@ -579,25 +650,25 @@ static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg)
u8 core_status;
- nxt6000_dump_status(fe->i2c);
+ nxt6000_dump_status(fe);
*status = 0;
- core_status = nxt6000_readreg(fe->i2c, OFDM_COR_STAT);
+ core_status = nxt6000_readreg(fe, OFDM_COR_STAT);
if (core_status & AGCLOCKED)
*status |= FE_HAS_SIGNAL;
- if (nxt6000_readreg(fe->i2c, OFDM_SYR_STAT) & GI14_SYR_LOCK)
+ if (nxt6000_readreg(fe, OFDM_SYR_STAT) & GI14_SYR_LOCK)
*status |= FE_HAS_CARRIER;
- if (nxt6000_readreg(fe->i2c, VIT_SYNC_STATUS) & VITINSYNC)
+ if (nxt6000_readreg(fe, VIT_SYNC_STATUS) & VITINSYNC)
*status |= FE_HAS_VITERBI;
- if (nxt6000_readreg(fe->i2c, RS_COR_STAT) & RSCORESTATUS)
+ if (nxt6000_readreg(fe, RS_COR_STAT) & RSCORESTATUS)
*status |= FE_HAS_SYNC;
- if (core_status & TPSLOCKED)
+ if ((core_status & TPSLOCKED) && (*status == (FE_HAS_SIGNAL | FE_HAS_CARRIER | FE_HAS_VITERBI | FE_HAS_SYNC)))
*status |= FE_HAS_LOCK;
return 0;
@@ -645,22 +716,51 @@ static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg)
case FE_INIT:
case FE_RESET:
+
+ nxt6000_reset(fe);
+ nxt6000_setup(fe);
- //nxt6000_setup(fe->i2c);
+// alp510_set_tv_freq(fe, 177500000); // OK
+// alp510_set_tv_freq(fe, 602000000); // NOK
+ alp510_set_tv_freq(fe, 650000000); // NOK
+// alp510_set_tv_freq(fe, 658000000); // NOK
+ alp510_set_tv_freq(fe, 778000000); // OK
+
+ nxt6000_set_bandwidth(fe, BANDWIDTH_8_MHZ);
+ nxt6000_set_guard_interval(fe, GUARD_INTERVAL_1_8);
+ nxt6000_set_transmission_mode(fe, TRANSMISSION_MODE_8K);
+// nxt6000_set_transmission_mode(fe, TRANSMISSION_MODE_2K);
+ nxt6000_set_inversion(fe, INVERSION_OFF);
+// nxt6000_set_inversion(fe, INVERSION_ON);
+
+ nxt6000_reset(fe);
+
+ nxt6000_dump_status(fe);
break;
-
+
case FE_SET_FRONTEND:
{
struct dvb_frontend_parameters *param = (struct dvb_frontend_parameters *)arg;
+ int result;
- alp510_set_tv_freq(fe->i2c, param->frequency);
+ printk("nxt6000: params: fre=%d inv=%d\n", param->frequency, param->inversion);
- nxt6000_set_guard_interval(fe->i2c, param->u.ofdm.guard_interval);
- nxt6000_set_transmission_mode(fe->i2c, param->u.ofdm.transmission_mode);
- nxt6000_set_inversion(fe->i2c, param->inversion);
+ if ((result = alp510_set_tv_freq(fe, param->frequency)) < 0)
+ return result;
+
+ if ((result = nxt6000_set_bandwidth(fe, param->u.ofdm.bandwidth)) < 0)
+ return result;
+ if ((result = nxt6000_set_guard_interval(fe, param->u.ofdm.guard_interval)) < 0)
+ return result;
+ if ((result = nxt6000_set_transmission_mode(fe, param->u.ofdm.transmission_mode)) < 0)
+ return result;
+ if ((result = nxt6000_set_inversion(fe, param->inversion)) < 0)
+ return result;
+
+ printk("nxt6000: param check ok\n");
- nxt6000_reset(fe->i2c);
+ nxt6000_reset(fe);
break;
}
@@ -675,58 +775,57 @@ static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg)
}
+static u8 demod_addr_tbl[] = {0x14, 0x18, 0x24, 0x28};
+
static int nxt6000_attach(struct dvb_i2c_bus *i2c)
{
- u8 core_rev;
+ u8 addr_nr;
+ u8 fe_count = 0;
+ struct nxt6000_config nxt;
+ u8 tuner_addr;
+ u8 tuner_type;
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;
-
- }
+ for (addr_nr = 0; addr_nr < sizeof(demod_addr_tbl); addr_nr++) {
- if (pll_write(i2c, 0xC0, NULL, 0) == 0) {
+ if (nxt6000_read(i2c, demod_addr_tbl[addr_nr], OFDM_MSC_REV) != NXT6000ASICDEVICE)
+ continue;
+
+ if (pll_write(i2c, demod_addr_tbl[addr_nr], 0xC0, NULL, 0) == 0) {
- printk("nxt6000: detected TI ALP510 tuner\n");
+ tuner_addr = 0xC0;
+ tuner_type = 1;
+
+ printk("nxt6000: detected TI ALP510 tuner at 0x%02X\n", tuner_addr);
- } else if (pll_write(i2c, 0xC2, NULL, 0) == 0) {
+ } else if (pll_write(i2c, demod_addr_tbl[addr_nr], 0xC2, NULL, 0) == 0) {
+
+ tuner_addr = 0xC0;
+ tuner_type = 2;
- printk("nxt6000: detected MITEL SP5659 tuner\n");
+ printk("nxt6000: detected MITEL SP5659 tuner at 0x%02X\n", tuner_addr);
- } else {
+ } else {
+
+ printk("nxt6000: unable to detect tuner\n");
- printk("nxt6000: unable to detect tuner\n");
+ continue;
+
+ }
+
+ nxt.demod_addr = demod_addr_tbl[addr_nr];
+ nxt.tuner_addr = tuner_addr;
+ nxt.tuner_type = tuner_type;
+
+ printk("nxt6000: attached at %d:%d\n", i2c->adapter->num, i2c->id);
- return -EINVAL;
+ dvb_register_frontend(nxt6000_ioctl, i2c, (void *)(*((u32 *)&nxt)), &nxt6000_info);
}
-
- printk("nxt6000: attached at %d:%d\n", i2c->adapter->num, i2c->id);
- nxt6000_reset(i2c);
- nxt6000_setup(i2c);
-
-// alp510_set_tv_freq(i2c, 177500000);
-// alp510_set_tv_freq(i2c, 602000000);
- alp510_set_tv_freq(i2c, 650000000); // NOK
-// alp510_set_tv_freq(i2c, 658000000); // OK
- alp510_set_tv_freq(i2c, 778000000); // OK
- nxt6000_set_guard_interval(i2c, GUARD_INTERVAL_1_8);
-// nxt6000_set_transmission_mode(i2c, TRANSMISSION_MODE_8K);
- nxt6000_set_transmission_mode(i2c, TRANSMISSION_MODE_2K);
- nxt6000_set_inversion(i2c, INVERSION_OFF);
-// nxt6000_set_inversion(i2c, INVERSION_ON);
-
- nxt6000_reset(i2c);
-
- nxt6000_dump_status(i2c);
-
- return dvb_register_frontend(nxt6000_ioctl, i2c, NULL, &nxt6000_info);
+ return (fe_count > 0) ? 0 : -ENODEV;
}