diff options
Diffstat (limited to 'linux/drivers/media/dvb')
-rw-r--r-- | linux/drivers/media/dvb/frontends/cx22702.c | 95 |
1 files changed, 85 insertions, 10 deletions
diff --git a/linux/drivers/media/dvb/frontends/cx22702.c b/linux/drivers/media/dvb/frontends/cx22702.c index b27722a0c..53555ad44 100644 --- a/linux/drivers/media/dvb/frontends/cx22702.c +++ b/linux/drivers/media/dvb/frontends/cx22702.c @@ -1,4 +1,4 @@ -/* +/* Conexant 22702 DVB OFDM frontend driver based on: @@ -40,12 +40,16 @@ #define I2C_EEPROM_SLAVE_ADDR 0x50 #define I2C_PLL_SLAVE_ADDR 0x61 +#define PLLTYPE_DTT7592 1 +#define PLLTYPE_DTT7595 2 +#define PLLTYPE_DTT7579 3 + static int debug = 0; #define dprintk if (debug) printk static struct dvb_frontend_info cx22702_info = { - .name = "CX22702 Demod Thompson 759x PLL", + .name = "CX22702 Demod Thompson 759x/7579 PLL", .type = FE_OFDM, .frequency_min = 177000000, .frequency_max = 858000000, @@ -60,6 +64,7 @@ struct cx22702_state { struct i2c_adapter *i2c; struct dvb_adapter *dvb; struct dvb_frontend_info cx22702_info; + char pll_type; }; /* Register values to initialise the demod */ @@ -146,7 +151,7 @@ static int pll_write (struct i2c_adapter *i2c, u8 data [4]) return (ret != 1) ? -1 : 0; } -static int pll_set_tv_freq (struct i2c_adapter *i2c, u32 freq) +static int pll_dtt759x_set_tv_freq (struct i2c_adapter *i2c, u32 freq) { int ret; @@ -216,6 +221,56 @@ static int pll_set_tv_freq (struct i2c_adapter *i2c, u32 freq) } +static int pll_dtt7579_set_tv_freq (struct i2c_adapter *i2c, u32 freq) +{ + int ret; + + u32 div = (freq + 36166667) / 166666; + + /* dividerhigh, divierlow */ + unsigned char buf [4] = { + div >> 8, + div & 0xff, + 0x00, + 0x00 + }; + + // Now compensate for the charge pump osc + if(freq <= 506000000) { + buf[2] = 0xb4; + buf[3] = 0x02; + } else if (freq <= 735000000) { + buf[2] = 0xbc; + buf[3] = 0x08; + } else if (freq <= 835000000) { + buf[2] = 0xf4; + buf[3] = 0x08; + } else if (freq <= 896000000) { + buf[2] = 0xfc; + buf[3] = 0x08; + } + + dprintk ("%s: freq == %i, div == 0x%04x\n", __FUNCTION__, (int) freq, (int) div); + + ret= pll_write (i2c, buf); + if(ret<0) { + dprintk ("%s: first pll_write failed\n",__FUNCTION__); + return ret; + } + + /* Set the AGC to search */ + buf[2]=(buf[2] & 0xdc) | 0x9c; + buf[3]=0xa0; + ret=pll_write (i2c, buf); + if(ret<0) { + dprintk ("%s: second pll_write failed\n",__FUNCTION__); + return ret; + } + + return ret; + +} + /* Reset the demod hardware and reset all of the configuration registers to a default state. */ static int cx22702_init (struct i2c_adapter *i2c) @@ -519,7 +574,16 @@ static int cx22702_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg) /* Disable aquisition */ cx22702_writereg (i2c, 0x00, 0x00); - pll_set_tv_freq (i2c, p->frequency); + switch(state->pll_type) { + case PLLTYPE_DTT7592: + case PLLTYPE_DTT7595: + pll_dtt759x_set_tv_freq (i2c, p->frequency); + break; + + case PLLTYPE_DTT7579: + pll_dtt7579_set_tv_freq (i2c, p->frequency); + break; + } cx22702_set_inversion (i2c, p->inversion); if((ret=cx22702_set_tps (i2c, &p->u.ofdm))<0) { dprintk ("%s: set_tps failed ret=%d\n",__FUNCTION__,ret); @@ -552,7 +616,7 @@ static int cx22702_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg) /* Validate the eeprom contents, make sure content look ok. Get the eeprom data. */ -static int cx22702_validate_eeprom(struct i2c_adapter *i2c, int* minfreq) +static int cx22702_validate_eeprom(struct i2c_adapter *i2c, int* minfreq, int* pll_type) { u8 b0 [] = { 0 }; u8 b1 [128]; @@ -607,10 +671,12 @@ static int cx22702_validate_eeprom(struct i2c_adapter *i2c, int* minfreq) case 0x4B: dprintk ("%s: Tuner Thompson DTT 7595\n",__FUNCTION__); *minfreq = 177000000; + *pll_type = PLLTYPE_DTT7595; break; case 0x4C: dprintk ("%s: Tuner Thompson DTT 7592\n",__FUNCTION__); *minfreq = 474000000; + *pll_type = PLLTYPE_DTT7592; break; default: printk ("%s: Unknown tuner 0x%02x not supported\n",__FUNCTION__,tuner); @@ -666,13 +732,21 @@ static int cx22702_attach_adapter(struct i2c_adapter *adapter) struct i2c_client *client; int ret; int minfreq; + int pll_type; dprintk("Trying to attach to adapter 0x%x:%s.\n", adapter->id, adapter->name); - ret=cx22702_validate_eeprom(adapter, &minfreq); - if(ret < 0) - return ret; + if (!strcmp(adapter->name, "Conexant DVB-t")) { + printk("cx22702: Detected Conexant DVB-t card - PLL Thomson DTT7579\n"); + pll_type = PLLTYPE_DTT7579; + minfreq = 474000000; // guess + } else { + // FIXME + ret=cx22702_validate_eeprom(adapter, &minfreq, &pll_type); + if(ret < 0) + return ret; + } ret=cx22702_validate_demod(adapter); if(ret < 0) @@ -689,7 +763,8 @@ static int cx22702_attach_adapter(struct i2c_adapter *adapter) state->i2c = adapter; memcpy(&state->cx22702_info, &cx22702_info, sizeof(struct dvb_frontend_info)); state->cx22702_info.frequency_min = minfreq; - + state->pll_type = pll_type; + if ( !(client = kmalloc(sizeof(struct i2c_client), GFP_KERNEL)) ) { kfree(state); return -ENOMEM; @@ -780,7 +855,7 @@ module_exit (exit_cx22702); MODULE_PARM(debug,"i"); MODULE_PARM_DESC(debug, "Enable verbose debug messages"); -MODULE_DESCRIPTION("CX22702 / Thompson DTT 759x PLL DVB Frontend driver"); +MODULE_DESCRIPTION("CX22702 / Thompson DTT 759x / Thompson DTT 7579 PLL DVB Frontend driver"); MODULE_AUTHOR("Steven Toth"); MODULE_LICENSE("GPL"); |