summaryrefslogtreecommitdiff
path: root/linux/drivers/media/dvb
diff options
context:
space:
mode:
Diffstat (limited to 'linux/drivers/media/dvb')
-rw-r--r--linux/drivers/media/dvb/frontends/ves1820.c415
1 files changed, 212 insertions, 203 deletions
diff --git a/linux/drivers/media/dvb/frontends/ves1820.c b/linux/drivers/media/dvb/frontends/ves1820.c
index 7f759af1f..d5f3883c0 100644
--- a/linux/drivers/media/dvb/frontends/ves1820.c
+++ b/linux/drivers/media/dvb/frontends/ves1820.c
@@ -27,8 +27,11 @@
#include "dvb_frontend.h"
-static int debug = 0;
-#define dprintk if (debug) printk
+#if 0
+#define dprintk(x...) printk(x)
+#else
+#define dprintk(x...)
+#endif
/**
@@ -36,43 +39,58 @@ static int debug = 0;
* extra memory but use frontend->data as bitfield
*/
-#define SET_PWM(frontend,pwm) do { \
- (int) frontend->data &= ~0xff; \
- (int) frontend->data |= pwm; \
+#define SET_PWM(data,pwm) do { \
+ (int) data &= ~0xff; \
+ (int) data |= pwm; \
+} while (0)
+
+#define SET_REG0(data,reg0) do { \
+ (int) data &= ~(0xff << 8); \
+ (int) data |= reg0 << 8; \
} while (0)
-#define SET_REG0(frontend,reg0) do { \
- (int) frontend->data &= ~(0xff << 8); \
- (int) frontend->data |= reg0 << 8; \
+#define SET_TUNER(data,type) do { \
+ (int) data &= ~(0xff << 16); \
+ (int) data |= type << 16; \
} while (0)
-#define SET_TUNER(frontend,type) do { \
- (int) frontend->data &= ~(0xff << 16); \
- (int) frontend->data |= type << 16; \
+#define SET_DEMOD_ADDR(data,type) do { \
+ (int) data &= ~(0xff << 24); \
+ (int) data |= type << 24; \
} while (0)
-#define GET_PWM(frontend) ((u8) ((int) frontend->data & 0xff))
-#define GET_REG0(frontend) ((u8) (((int) frontend->data >> 8) & 0xff))
-#define GET_TUNER(frontend) ((u8) (((int) frontend->data >> 16) & 0xff))
+#define GET_PWM(data) ((u8) ((int) data & 0xff))
+#define GET_REG0(data) ((u8) (((int) data >> 8) & 0xff))
+#define GET_TUNER(data) ((u8) (((int) data >> 16) & 0xff))
+#define GET_DEMOD_ADDR(data) ((u8) (((int) data >> 24) & 0xff))
+static inline
+void ddelay (int ms)
+{
+ current->state=TASK_INTERRUPTIBLE;
+ schedule_timeout((HZ*ms)/1000);
+}
+
static
struct dvb_frontend_info ves1820_info = {
- name: "VES1820/Grundig tuner as used on the Siemens DVB-C card",
- type: FE_QAM,
- frequency_stepsize: 62500,
- frequency_min: 51000000,
- frequency_max: 858000000,
+ .name = "VES1820 based DVB-C frontend",
+ .type = FE_QAM,
+ .frequency_stepsize = 62500,
+ .frequency_min = 51000000,
+ .frequency_max = 858000000,
+ .symbol_rate_min = (57840000/2)/64, /* SACLK/64 == (XIN/2)/64 */
+ .symbol_rate_max = (57840000/2)/4, /* SACLK/4 */
#if 0
frequency_tolerance: ???,
- symbol_rate_min: ???,
- symbol_rate_max: ???,
symbol_rate_tolerance: ???, /* ppm */ /* == 8% (spec p. 5) */
notifier_delay: ?,
#endif
- caps: FE_CAN_QAM_16 | FE_CAN_QAM_32 | FE_CAN_QAM_64 |
- FE_CAN_QAM_128 | FE_CAN_QAM_256
+ .caps = FE_CAN_QAM_16 | FE_CAN_QAM_32 | FE_CAN_QAM_64 |
+ FE_CAN_QAM_128 | FE_CAN_QAM_256 |
+ FE_CAN_FEC_AUTO | FE_CAN_INVERSION_AUTO |
+ FE_CAN_CLEAN_SETUP
};
@@ -80,22 +98,24 @@ struct dvb_frontend_info ves1820_info = {
static
u8 ves1820_inittab [] =
{
- 0x69, 0x6A, 0x9B, 0x0A, 0x52, 0x46, 0x26, 0x1A,
- 0x43, 0x6A, 0xAA, 0xAA, 0x1E, 0x85, 0x43, 0x28,
- 0xE0, 0x00, 0xA1, 0x00, 0x00, 0x00, 0x00, 0x00,
- 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00,
- 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
- 0x00, 0x00, 0x00, 0x00, 0x40
+ 0x69, 0x6A, 0x9B, 0x0A, 0x52, 0x46, 0x26, 0x1A,
+ 0x43, 0x6A, 0xAA, 0xAA, 0x1E, 0x85, 0x43, 0x28,
+ 0xE0, 0x00, 0xA1, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00,
+ 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x40
};
static
-int ves1820_writereg (struct dvb_i2c_bus *i2c, u8 reg, u8 data)
+int ves1820_writereg (struct dvb_frontend *fe, u8 reg, u8 data)
{
- int ret;
+ u8 addr = GET_DEMOD_ADDR(fe->data);
u8 buf[] = { 0x00, reg, data };
- struct i2c_msg msg = { addr: 0x09, flags: 0, buf: buf, len: 3 };
+ struct i2c_msg msg = { addr: addr, flags: 0, buf: buf, len: 3 };
+ struct dvb_i2c_bus *i2c = fe->i2c;
+ int ret;
ret = i2c->xfer (i2c, &msg, 1);
@@ -110,14 +130,15 @@ int ves1820_writereg (struct dvb_i2c_bus *i2c, u8 reg, u8 data)
static
-u8 ves1820_readreg (struct dvb_i2c_bus *i2c, u8 reg)
+u8 ves1820_readreg (struct dvb_frontend *fe, u8 reg)
{
- int ret;
u8 b0 [] = { 0x00, reg };
u8 b1 [] = { 0 };
- struct i2c_msg msg [] = { { addr: 0x09, flags: 0, buf: b0, len: 2 },
- { addr: 0x09, flags: I2C_M_RD, buf: b1, len: 1 } };
-
+ u8 addr = GET_DEMOD_ADDR(fe->data);
+ struct i2c_msg msg [] = { { addr: addr, flags: 0, buf: b0, len: 2 },
+ { addr: addr, flags: I2C_M_RD, buf: b1, len: 1 } };
+ struct dvb_i2c_bus *i2c = fe->i2c;
+ int ret;
ret = i2c->xfer (i2c, msg, 2);
@@ -148,14 +169,17 @@ int tuner_write (struct dvb_i2c_bus *i2c, u8 addr, u8 data [4])
* reference clock comparision frequency of 62.5 kHz.
*/
static
-int tuner_set_tv_freq (struct dvb_frontend *frontend, u32 freq)
+int tuner_set_tv_freq (struct dvb_frontend *fe, u32 freq)
{
u32 div;
static u8 addr [] = { 0x61, 0x62 };
static u8 byte3 [] = { 0x8e, 0x85 };
- int tuner_type = GET_TUNER(frontend);
+ int tuner_type = GET_TUNER(fe->data);
u8 buf [4];
+ if (tuner_type == 0xff) /* PLL not reachable over i2c ... */
+ return 0;
+
div = (freq + 36250000 + 31250) / 62500;
buf[0] = (div >> 8) & 0x7f;
buf[1] = div & 0xff;
@@ -170,115 +194,56 @@ int tuner_set_tv_freq (struct dvb_frontend *frontend, u32 freq)
freq < 454000000 ? 0x92 : 0x34);
}
- return tuner_write (frontend->i2c, addr[tuner_type], buf);
+ return tuner_write (fe->i2c, addr[tuner_type], buf);
}
static
-int probe_tuner (struct dvb_frontend *frontend)
+int ves1820_setup_reg0 (struct dvb_frontend *fe, u8 reg0)
{
- struct dvb_i2c_bus *i2c = frontend->i2c;
- struct i2c_msg msg = { addr: 0x61, flags: 0, buf: NULL, len: 0 };
-
- if (i2c->xfer(i2c, &msg, 1) == 1) {
- SET_TUNER(frontend,0);
- printk ("%s: setup for tuner spXXXX\n", __FILE__);
- } else {
- SET_TUNER(frontend,1);
- printk ("%s: setup for tuner sp5659c\n", __FILE__);
+ reg0 |= GET_REG0(fe->data) & 0x62;
+
+ ves1820_writereg (fe, 0x00, reg0 & 0xfe);
+ ves1820_writereg (fe, 0x00, reg0 | 0x01);
+
+ /**
+ * check lock and toggle inversion bit if required...
+ */
+ if (!ves1820_readreg (fe, 0x11) & 0x08) {
+ ddelay(1);
+ if (!ves1820_readreg (fe, 0x11) & 0x08) {
+ reg0 ^= 0x20;
+ ves1820_writereg (fe, 0x00, reg0 & 0xfe);
+ ves1820_writereg (fe, 0x00, reg0);
+ }
}
+ SET_REG0(fe->data, reg0);
+
return 0;
}
static
-int ves1820_init (struct dvb_frontend *frontend)
+int ves1820_init (struct dvb_frontend *fe)
{
- struct dvb_i2c_bus *i2c = frontend->i2c;
- u8 b0 [] = { 0xff };
- u8 pwm;
int i;
- struct i2c_msg msg [] = { { addr: 0x50, flags: 0, buf: b0, len: 1 },
- { addr: 0x50, flags: I2C_M_RD, buf: &pwm, len: 1 } };
dprintk("VES1820: init chip\n");
- i2c->xfer (i2c, msg, 2);
-
- dprintk("VES1820: pwm=%02x\n", pwm);
-
- if (pwm == 0xff)
- pwm=0x48;
-
- ves1820_writereg (i2c, 0, 0);
+ ves1820_writereg (fe, 0, 0);
for (i=0; i<53; i++)
- ves1820_writereg (i2c, i, ves1820_inittab[i]);
+ ves1820_writereg (fe, i, ves1820_inittab[i]);
- ves1820_writereg (i2c, 0x34, pwm);
-
- (int) frontend->data = 0;
- SET_PWM(frontend,pwm);
-
- probe_tuner (frontend);
+ ves1820_writereg (fe, 0x34, GET_PWM(fe->data));
return 0;
}
static
-int ves1820_setup_reg0 (struct dvb_frontend *frontend,
- u8 real_qam, fe_spectral_inversion_t inversion)
-{
- struct dvb_i2c_bus *i2c = frontend->i2c;
- u8 reg0 = (ves1820_inittab[0] & 0xe3) | (real_qam << 2);
-
- switch (inversion) {
- case INVERSION_OFF: /* XXX FIXME: reversed?? p. 25 */
- reg0 |= 0x20;
- break;
-
- case INVERSION_ON:
- reg0 &= 0xdf;
- break;
-
- default:
- return -EINVAL;
- }
-
- SET_REG0(frontend, reg0);
-
- ves1820_writereg (i2c, 0x00, reg0 & 0xfe);
- ves1820_writereg (i2c, 0x00, reg0);
-
- return 0;
-}
-
-
-static
-int ves1820_reset (struct dvb_frontend *frontend)
-{
- struct dvb_i2c_bus *i2c = frontend->i2c;
- u8 reg0 = GET_REG0(frontend);
-
- ves1820_writereg (i2c, 0x00, reg0 & 0xfe);
- ves1820_writereg (i2c, 0x00, reg0);
-
- return 0;
-}
-
-
-static
-void ves1820_reset_uncorrected_block_counter (struct dvb_i2c_bus *i2c)
-{
- ves1820_writereg (i2c, 0x10, ves1820_inittab[0x10] & 0xdf);
- ves1820_writereg (i2c, 0x10, ves1820_inittab[0x10]);
-}
-
-
-static
-int ves1820_set_symbolrate (struct dvb_i2c_bus *i2c, u32 symbolrate)
+int ves1820_set_symbolrate (struct dvb_frontend *fe, u32 symbolrate)
{
s32 BDR;
s32 BDRI;
@@ -287,7 +252,7 @@ int ves1820_set_symbolrate (struct dvb_i2c_bus *i2c, u32 symbolrate)
u32 tmp, ratio;
#define XIN 57840000UL
-#define FIN (57840000UL>>4)
+#define FIN (XIN >> 4)
if (symbolrate > XIN/2)
symbolrate = XIN/2;
@@ -324,81 +289,50 @@ int ves1820_set_symbolrate (struct dvb_i2c_bus *i2c, u32 symbolrate)
NDEC = (NDEC << 6) | ves1820_inittab[0x03];
- ves1820_writereg (i2c, 0x03, NDEC);
- ves1820_writereg (i2c, 0x0a, BDR&0xff);
- ves1820_writereg (i2c, 0x0b, (BDR>> 8)&0xff);
- ves1820_writereg (i2c, 0x0c, (BDR>>16)&0x3f);
+ ves1820_writereg (fe, 0x03, NDEC);
+ ves1820_writereg (fe, 0x0a, BDR&0xff);
+ ves1820_writereg (fe, 0x0b, (BDR>> 8)&0xff);
+ ves1820_writereg (fe, 0x0c, (BDR>>16)&0x3f);
- ves1820_writereg (i2c, 0x0d, BDRI);
- ves1820_writereg (i2c, 0x0e, SFIL);
+ ves1820_writereg (fe, 0x0d, BDRI);
+ ves1820_writereg (fe, 0x0e, SFIL);
return 0;
}
static
-void ves1820_reset_pwm (struct dvb_frontend *frontend)
-{
- u8 pwm = GET_PWM(frontend);
-
- ves1820_writereg (frontend->i2c, 0x34, pwm);
-}
-
-
-typedef struct {
- fe_modulation_t QAM_Mode;
- int NoOfSym;
- u8 Reg1;
- u8 Reg5;
- u8 Reg8;
- u8 Reg9;
-} QAM_SETTING;
-
-
-QAM_SETTING QAM_Values[] = {
- { QAM_16, 16, 140, 164, 162, 145 },
- { QAM_32, 32, 140, 120, 116, 150 },
- { QAM_64, 64, 106, 70, 67, 106 },
- { QAM_128, 128, 120, 54, 52, 126 },
- { QAM_256, 256, 92, 38, 35, 107 }
-};
-
-
-static
-int ves1820_set_parameters (struct dvb_frontend *frontend,
+int ves1820_set_parameters (struct dvb_frontend *fe,
struct dvb_frontend_parameters *p)
{
- struct dvb_i2c_bus* i2c = frontend->i2c;
- int real_qam;
-
- switch (p->u.qam.modulation) {
- case QAM_16 : real_qam = 0; break;
- case QAM_32 : real_qam = 1; break;
- case QAM_64 : real_qam = 2; break;
- case QAM_128: real_qam = 3; break;
- case QAM_256: real_qam = 4; break;
- default:
- return -EINVAL;
- }
+ static const u8 reg0x00 [] = { 0x00, 0x04, 0x08, 0x0c, 0x10 };
+ static const u8 reg0x01 [] = { 140, 140, 106, 120, 92 };
+ static const u8 reg0x05 [] = { 135, 100, 70, 54, 38 };
+ static const u8 reg0x08 [] = { 162, 116, 67, 52, 35 };
+ static const u8 reg0x09 [] = { 145, 150, 106, 126, 107 };
+ int real_qam = p->u.qam.modulation - QAM_16;
+
+ if (real_qam < 0 || real_qam > 4)
+ return -EINVAL;
- tuner_set_tv_freq (frontend, p->frequency);
- ves1820_set_symbolrate (i2c, p->u.qam.symbol_rate);
- ves1820_reset_pwm (frontend);
+ tuner_set_tv_freq (fe, p->frequency);
+ ves1820_set_symbolrate (fe, p->u.qam.symbol_rate);
+ ves1820_writereg (fe, 0x34, GET_PWM(fe->data));
- ves1820_writereg (i2c, 0x01, QAM_Values[real_qam].Reg1);
- ves1820_writereg (i2c, 0x05, QAM_Values[real_qam].Reg5);
- ves1820_writereg (i2c, 0x08, QAM_Values[real_qam].Reg8);
- ves1820_writereg (i2c, 0x09, QAM_Values[real_qam].Reg9);
+ ves1820_writereg (fe, 0x01, reg0x01[real_qam]);
+ ves1820_writereg (fe, 0x05, reg0x05[real_qam]);
+ ves1820_writereg (fe, 0x08, reg0x08[real_qam]);
+ ves1820_writereg (fe, 0x09, reg0x09[real_qam]);
+
+ ves1820_setup_reg0 (fe, reg0x00[real_qam]);
- ves1820_setup_reg0 (frontend, real_qam, p->inversion);
-
return 0;
}
static
-int ves1820_ioctl (struct dvb_frontend *frontend, unsigned int cmd, void *arg)
+int ves1820_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg)
{
switch (cmd) {
case FE_GET_INFO:
@@ -412,7 +346,7 @@ int ves1820_ioctl (struct dvb_frontend *frontend, unsigned int cmd, void *arg)
*status = 0;
- sync = ves1820_readreg (frontend->i2c, 0x11);
+ sync = ves1820_readreg (fe, 0x11);
if (sync & 2)
*status |= FE_HAS_SIGNAL;
@@ -434,35 +368,37 @@ int ves1820_ioctl (struct dvb_frontend *frontend, unsigned int cmd, void *arg)
case FE_READ_BER:
{
- u32 ber = ves1820_readreg(frontend->i2c, 0x14) |
- (ves1820_readreg(frontend->i2c, 0x15) << 8) |
- ((ves1820_readreg(frontend->i2c, 0x16) & 0x0f) << 16);
+ u32 ber = ves1820_readreg(fe, 0x14) |
+ (ves1820_readreg(fe, 0x15) << 8) |
+ ((ves1820_readreg(fe, 0x16) & 0x0f) << 16);
*((u32*) arg) = 10 * ber;
break;
}
case FE_READ_SIGNAL_STRENGTH:
{
- u8 gain = ves1820_readreg(frontend->i2c, 0x17);
+ u8 gain = ves1820_readreg(fe, 0x17);
*((u16*) arg) = (gain << 8) | gain;
break;
}
case FE_READ_SNR:
{
- u8 quality = ~ves1820_readreg(frontend->i2c, 0x18);
+ u8 quality = ~ves1820_readreg(fe, 0x18);
*((u16*) arg) = (quality << 8) | quality;
break;
}
case FE_READ_UNCORRECTED_BLOCKS:
- *((u32*) arg) = ves1820_readreg (frontend->i2c, 0x13) & 0x7f;
+ *((u32*) arg) = ves1820_readreg (fe, 0x13) & 0x7f;
if (*((u32*) arg) == 0x7f)
*((u32*) arg) = 0xffffffff;
- ves1820_reset_uncorrected_block_counter (frontend->i2c);
+ /* reset uncorrected block counter */
+ ves1820_writereg (fe, 0x10, ves1820_inittab[0x10] & 0xdf);
+ ves1820_writereg (fe, 0x10, ves1820_inittab[0x10]);
break;
case FE_SET_FRONTEND:
- return ves1820_set_parameters (frontend, arg);
+ return ves1820_set_parameters (fe, arg);
case FE_GET_FRONTEND:
/* XXX FIXME: implement! */
@@ -475,16 +411,12 @@ int ves1820_ioctl (struct dvb_frontend *frontend, unsigned int cmd, void *arg)
break;
case FE_SLEEP:
- ves1820_writereg (frontend->i2c, 0x1b, 0x02); /* pdown ADC */
- ves1820_writereg (frontend->i2c, 0x00, 0x80); /* standby */
+ ves1820_writereg (fe, 0x1b, 0x02); /* pdown ADC */
+ ves1820_writereg (fe, 0x00, 0x80); /* standby */
break;
case FE_INIT:
- return ves1820_init (frontend);
-
- case FE_RESET:
- ves1820_reset (frontend);
- break;
+ return ves1820_init (fe);
default:
return -EINVAL;
@@ -495,12 +427,89 @@ int ves1820_ioctl (struct dvb_frontend *frontend, unsigned int cmd, void *arg)
static
+int probe_tuner (struct dvb_i2c_bus *i2c)
+{
+ static const
+ struct i2c_msg msg1 = { addr: 0x61, flags: 0, buf: NULL, len: 0 };
+ static const
+ struct i2c_msg msg2 = { addr: 0x62, flags: 0, buf: NULL, len: 0 };
+ int type;
+
+ if (i2c->xfer(i2c, &msg1, 1) == 1) {
+ type = 0;
+ printk ("%s: setup for tuner spXXXX\n", __FILE__);
+ } else if (i2c->xfer(i2c, &msg2, 1) == 1) {
+ type = 1;
+ printk ("%s: setup for tuner sp5659c\n", __FILE__);
+ } else {
+ type = -1;
+ printk ("%s: unknown PLL, "
+ "please report to <linuxdvb@linuxtv.org>!!\n",
+ __FILE__);
+ }
+
+ return type;
+}
+
+
+static
+u8 read_pwm (struct dvb_i2c_bus *i2c)
+{
+ u8 b = 0xff;
+ u8 pwm;
+ struct i2c_msg msg [] = { { addr: 0x50, flags: 0, buf: &b, len: 1 },
+ { addr: 0x50, flags: I2C_M_RD, buf: &pwm, len: 1 } };
+
+ i2c->xfer (i2c, msg, 2);
+
+ dprintk("VES1820: pwm=%02x\n", pwm);
+
+ if (pwm == 0xff)
+ pwm = 0x48;
+
+ return pwm;
+}
+
+
+static
+int probe_demod_addr (struct dvb_i2c_bus *i2c)
+{
+ u8 b [] = { 0x00, 0x1a };
+ u8 id;
+ struct i2c_msg msg [] = { { addr: 0x08, flags: 0, buf: b, len: 2 },
+ { addr: 0x08, flags: I2C_M_RD, buf: &id, len: 1 } };
+
+ if (i2c->xfer(i2c, msg, 2) == 2 && (id & 0xf0) == 0x70)
+ return msg[0].addr;
+
+ msg[0].addr = msg[1].addr = 0x09;
+
+ if (i2c->xfer(i2c, msg, 2) == 2 && (id & 0xf0) == 0x70)
+ return msg[0].addr;
+
+ return -1;
+}
+
+
+static
int ves1820_attach (struct dvb_i2c_bus *i2c)
{
- if ((ves1820_readreg (i2c, 0x1a) & 0xf0) != 0x70)
- return -ENODEV;
-
- dvb_register_frontend (ves1820_ioctl, i2c, NULL, &ves1820_info);
+ void *data = NULL;
+ int demod_addr;
+ int tuner_type;
+
+ if ((demod_addr = probe_demod_addr(i2c)) < 0)
+ return -ENODEV;
+
+ if ((tuner_type = probe_tuner(i2c)) < 0)
+ return -ENODEV;
+
+ SET_PWM(data, read_pwm(i2c));
+ SET_REG0(data, ves1820_inittab[0]);
+ SET_TUNER(data, tuner_type);
+ SET_DEMOD_ADDR(data, demod_addr);
+
+ dvb_register_frontend (ves1820_ioctl, i2c, data, &ves1820_info);
return 0;
}
@@ -531,8 +540,8 @@ void __exit exit_ves1820 (void)
module_init(init_ves1820);
module_exit(exit_ves1820);
-MODULE_DESCRIPTION("");
-MODULE_AUTHOR("Ralph Metzler");
+MODULE_DESCRIPTION("VES1820 DVB-C frontend driver");
+MODULE_AUTHOR("Ralph Metzler, Holger Waechtler");
MODULE_LICENSE("GPL");
MODULE_PARM(debug,"i");