2006-01-09 20:25:07 +03:00
/*
2008-03-30 04:01:12 +04:00
* Conexant cx24123 / cx24109 - DVB QPSK Satellite demod / tuner driver
*
2008-09-04 00:12:12 +04:00
* Copyright ( C ) 2005 Steven Toth < stoth @ linuxtv . org >
2008-03-30 04:01:12 +04:00
*
* Support for KWorld DVB - S 100 by Vadim Catana < skystar @ moldova . cc >
*
* Support for CX24123 / CX24113 - NIM by Patrick Boettcher < pb @ linuxtv . org >
*
* 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 0213 9 , USA .
*/
2006-01-09 20:25:07 +03:00
# include <linux/slab.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
2013-04-08 04:11:53 +04:00
# include <asm/div64.h>
2006-01-09 20:25:07 +03:00
# include "dvb_frontend.h"
# include "cx24123.h"
2006-04-13 17:19:52 +04:00
# define XTAL 10111000
2006-04-14 00:26:22 +04:00
static int force_band ;
2008-10-17 03:28:32 +04:00
module_param ( force_band , int , 0644 ) ;
MODULE_PARM_DESC ( force_band , " Force a specific band select " \
" (1-9, default:off). " ) ;
2006-01-09 20:25:07 +03:00
static int debug ;
2008-10-17 03:28:32 +04:00
module_param ( debug , int , 0644 ) ;
MODULE_PARM_DESC ( debug , " Activates frontend debugging (default:0) " ) ;
2008-03-30 04:01:12 +04:00
# define info(args...) do { printk(KERN_INFO "CX24123: " args); } while (0)
# define err(args...) do { printk(KERN_ERR "CX24123: " args); } while (0)
2006-01-09 20:25:07 +03:00
# define dprintk(args...) \
do { \
2008-03-30 04:01:12 +04:00
if ( debug ) { \
printk ( KERN_DEBUG " CX24123: %s: " , __func__ ) ; \
printk ( args ) ; \
} \
2006-01-09 20:25:07 +03:00
} while ( 0 )
2008-10-17 03:28:32 +04:00
struct cx24123_state {
struct i2c_adapter * i2c ;
const struct cx24123_config * config ;
2006-01-09 20:25:07 +03:00
struct dvb_frontend frontend ;
/* Some PLL specifics for tuning */
u32 VCAarg ;
u32 VGAarg ;
u32 bandselectarg ;
u32 pllarg ;
2006-04-13 17:19:52 +04:00
u32 FILTune ;
2006-01-09 20:25:07 +03:00
2008-03-30 04:01:12 +04:00
struct i2c_adapter tuner_i2c_adapter ;
u8 demod_rev ;
2006-01-09 20:25:07 +03:00
/* The Demod/Tuner can't easily provide these, we cache them */
u32 currentfreq ;
u32 currentsymbolrate ;
} ;
2006-01-09 20:25:08 +03:00
/* Various tuner defaults need to be established for a given symbol rate Sps */
2008-10-17 03:28:32 +04:00
static struct cx24123_AGC_val {
2006-01-09 20:25:08 +03:00
u32 symbolrate_low ;
u32 symbolrate_high ;
u32 VCAprogdata ;
u32 VGAprogdata ;
2006-04-13 17:19:52 +04:00
u32 FILTune ;
2006-01-09 20:25:08 +03:00
} cx24123_AGC_vals [ ] =
{
{
. symbolrate_low = 1000000 ,
. symbolrate_high = 4999999 ,
2006-04-13 17:19:52 +04:00
/* the specs recommend other values for VGA offsets,
but tests show they are wrong */
2006-04-14 00:24:13 +04:00
. VGAprogdata = ( 1 < < 19 ) | ( 0x180 < < 9 ) | 0x1e0 ,
. VCAprogdata = ( 2 < < 19 ) | ( 0x07 < < 9 ) | 0x07 ,
. FILTune = 0x27f /* 0.41 V */
2006-01-09 20:25:08 +03:00
} ,
{
. symbolrate_low = 5000000 ,
. symbolrate_high = 14999999 ,
2006-04-14 00:24:13 +04:00
. VGAprogdata = ( 1 < < 19 ) | ( 0x180 < < 9 ) | 0x1e0 ,
. VCAprogdata = ( 2 < < 19 ) | ( 0x07 < < 9 ) | 0x1f ,
2006-04-13 17:19:52 +04:00
. FILTune = 0x317 /* 0.90 V */
2006-01-09 20:25:08 +03:00
} ,
{
. symbolrate_low = 15000000 ,
. symbolrate_high = 45000000 ,
2006-04-14 00:24:13 +04:00
. VGAprogdata = ( 1 < < 19 ) | ( 0x100 < < 9 ) | 0x180 ,
. VCAprogdata = ( 2 < < 19 ) | ( 0x07 < < 9 ) | 0x3f ,
. FILTune = 0x145 /* 2.70 V */
2006-01-09 20:25:08 +03:00
} ,
} ;
/*
* Various tuner defaults need to be established for a given frequency kHz .
* fixme : The bounds on the bands do not match the doc in real life .
* fixme : Some of them have been moved , other might need adjustment .
*/
2008-10-17 03:28:32 +04:00
static struct cx24123_bandselect_val {
2006-01-09 20:25:08 +03:00
u32 freq_low ;
u32 freq_high ;
u32 VCOdivider ;
u32 progdata ;
} cx24123_bandselect_vals [ ] =
{
2006-04-14 00:26:22 +04:00
/* band 1 */
2006-01-09 20:25:08 +03:00
{
. freq_low = 950000 ,
. freq_high = 1074999 ,
. VCOdivider = 4 ,
2006-04-14 00:26:22 +04:00
. progdata = ( 0 < < 19 ) | ( 0 < < 9 ) | 0x40 ,
2006-01-09 20:25:08 +03:00
} ,
2006-04-14 00:26:22 +04:00
/* band 2 */
2006-01-09 20:25:08 +03:00
{
. freq_low = 1075000 ,
2006-04-14 00:26:22 +04:00
. freq_high = 1177999 ,
. VCOdivider = 4 ,
. progdata = ( 0 < < 19 ) | ( 0 < < 9 ) | 0x80 ,
2006-01-09 20:25:08 +03:00
} ,
2006-04-14 00:26:22 +04:00
/* band 3 */
2006-01-09 20:25:08 +03:00
{
2006-04-14 00:26:22 +04:00
. freq_low = 1178000 ,
. freq_high = 1295999 ,
2006-01-09 20:25:08 +03:00
. VCOdivider = 2 ,
2006-04-14 00:26:22 +04:00
. progdata = ( 0 < < 19 ) | ( 1 < < 9 ) | 0x01 ,
2006-01-09 20:25:08 +03:00
} ,
2006-04-14 00:26:22 +04:00
/* band 4 */
2006-01-09 20:25:08 +03:00
{
2006-04-14 00:26:22 +04:00
. freq_low = 1296000 ,
. freq_high = 1431999 ,
2006-01-09 20:25:08 +03:00
. VCOdivider = 2 ,
2006-04-14 00:26:22 +04:00
. progdata = ( 0 < < 19 ) | ( 1 < < 9 ) | 0x02 ,
2006-01-09 20:25:08 +03:00
} ,
2006-04-14 00:26:22 +04:00
/* band 5 */
2006-01-09 20:25:08 +03:00
{
2006-04-14 00:26:22 +04:00
. freq_low = 1432000 ,
. freq_high = 1575999 ,
2006-01-09 20:25:08 +03:00
. VCOdivider = 2 ,
2006-04-14 00:26:22 +04:00
. progdata = ( 0 < < 19 ) | ( 1 < < 9 ) | 0x04 ,
2006-01-09 20:25:08 +03:00
} ,
2006-04-14 00:26:22 +04:00
/* band 6 */
2006-01-09 20:25:08 +03:00
{
2006-04-14 00:26:22 +04:00
. freq_low = 1576000 ,
2006-01-09 20:25:08 +03:00
. freq_high = 1717999 ,
. VCOdivider = 2 ,
2006-04-14 00:26:22 +04:00
. progdata = ( 0 < < 19 ) | ( 1 < < 9 ) | 0x08 ,
2006-01-09 20:25:08 +03:00
} ,
2006-04-14 00:26:22 +04:00
/* band 7 */
2006-01-09 20:25:08 +03:00
{
. freq_low = 1718000 ,
. freq_high = 1855999 ,
. VCOdivider = 2 ,
2006-04-14 00:26:22 +04:00
. progdata = ( 0 < < 19 ) | ( 1 < < 9 ) | 0x10 ,
2006-01-09 20:25:08 +03:00
} ,
2006-04-14 00:26:22 +04:00
/* band 8 */
2006-01-09 20:25:08 +03:00
{
. freq_low = 1856000 ,
. freq_high = 2035999 ,
. VCOdivider = 2 ,
2006-04-14 00:26:22 +04:00
. progdata = ( 0 < < 19 ) | ( 1 < < 9 ) | 0x20 ,
2006-01-09 20:25:08 +03:00
} ,
2006-04-14 00:26:22 +04:00
/* band 9 */
2006-01-09 20:25:08 +03:00
{
. freq_low = 2036000 ,
2006-04-14 00:26:22 +04:00
. freq_high = 2150000 ,
2006-01-09 20:25:08 +03:00
. VCOdivider = 2 ,
2006-04-14 00:26:22 +04:00
. progdata = ( 0 < < 19 ) | ( 1 < < 9 ) | 0x40 ,
2006-01-09 20:25:08 +03:00
} ,
} ;
2006-01-09 20:25:07 +03:00
static struct {
u8 reg ;
u8 data ;
} cx24123_regdata [ ] =
{
{ 0x00 , 0x03 } , /* Reset system */
{ 0x00 , 0x00 } , /* Clear reset */
2006-04-14 00:24:13 +04:00
{ 0x03 , 0x07 } , /* QPSK, DVB, Auto Acquisition (default) */
{ 0x04 , 0x10 } , /* MPEG */
{ 0x05 , 0x04 } , /* MPEG */
{ 0x06 , 0x31 } , /* MPEG (default) */
{ 0x0b , 0x00 } , /* Freq search start point (default) */
{ 0x0c , 0x00 } , /* Demodulator sample gain (default) */
2006-08-07 00:03:50 +04:00
{ 0x0d , 0x7f } , /* Force driver to shift until the maximum (+-10 MHz) */
2006-04-14 00:24:13 +04:00
{ 0x0e , 0x03 } , /* Default non-inverted, FEC 3/4 (default) */
{ 0x0f , 0xfe } , /* FEC search mask (all supported codes) */
{ 0x10 , 0x01 } , /* Default search inversion, no repeat (default) */
{ 0x16 , 0x00 } , /* Enable reading of frequency */
{ 0x17 , 0x01 } , /* Enable EsNO Ready Counter */
{ 0x1c , 0x80 } , /* Enable error counter */
{ 0x20 , 0x00 } , /* Tuner burst clock rate = 500KHz */
{ 0x21 , 0x15 } , /* Tuner burst mode, word length = 0x15 */
{ 0x28 , 0x00 } , /* Enable FILTERV with positive pol., DiSEqC 2.x off */
{ 0x29 , 0x00 } , /* DiSEqC LNB_DC off */
{ 0x2a , 0xb0 } , /* DiSEqC Parameters (default) */
{ 0x2b , 0x73 } , /* DiSEqC Tone Frequency (default) */
{ 0x2c , 0x00 } , /* DiSEqC Message (0x2c - 0x31) */
2006-01-09 20:25:07 +03:00
{ 0x2d , 0x00 } ,
{ 0x2e , 0x00 } ,
{ 0x2f , 0x00 } ,
{ 0x30 , 0x00 } ,
{ 0x31 , 0x00 } ,
2006-04-14 00:24:13 +04:00
{ 0x32 , 0x8c } , /* DiSEqC Parameters (default) */
{ 0x33 , 0x00 } , /* Interrupts off (0x33 - 0x34) */
2006-01-09 20:25:07 +03:00
{ 0x34 , 0x00 } ,
2006-04-14 00:24:13 +04:00
{ 0x35 , 0x03 } , /* DiSEqC Tone Amplitude (default) */
{ 0x36 , 0x02 } , /* DiSEqC Parameters (default) */
{ 0x37 , 0x3a } , /* DiSEqC Parameters (default) */
{ 0x3a , 0x00 } , /* Enable AGC accumulator (for signal strength) */
{ 0x44 , 0x00 } , /* Constellation (default) */
{ 0x45 , 0x00 } , /* Symbol count (default) */
{ 0x46 , 0x0d } , /* Symbol rate estimator on (default) */
2006-08-08 22:48:08 +04:00
{ 0x56 , 0xc1 } , /* Error Counter = Viterbi BER */
2006-04-14 00:24:13 +04:00
{ 0x57 , 0xff } , /* Error Counter Window (default) */
2006-08-07 00:03:50 +04:00
{ 0x5c , 0x20 } , /* Acquisition AFC Expiration window (default is 0x10) */
2006-04-14 00:24:13 +04:00
{ 0x67 , 0x83 } , /* Non-DCII symbol clock */
2006-01-09 20:25:07 +03:00
} ;
2008-03-30 04:01:12 +04:00
static int cx24123_i2c_writereg ( struct cx24123_state * state ,
u8 i2c_addr , int reg , int data )
2006-01-09 20:25:07 +03:00
{
u8 buf [ ] = { reg , data } ;
2008-03-30 04:01:12 +04:00
struct i2c_msg msg = {
. addr = i2c_addr , . flags = 0 , . buf = buf , . len = 2
} ;
2006-01-09 20:25:07 +03:00
int err ;
2008-03-30 04:01:12 +04:00
/* printk(KERN_DEBUG "wr(%02x): %02x %02x\n", i2c_addr, reg, data); */
2006-04-13 18:29:13 +04:00
2008-10-17 03:28:32 +04:00
err = i2c_transfer ( state - > i2c , & msg , 1 ) ;
if ( err ! = 1 ) {
2006-01-09 20:25:07 +03:00
printk ( " %s: writereg error(err == %i, reg == 0x%02x, "
2008-03-30 04:01:12 +04:00
" data == 0x%02x) \n " , __func__ , err , reg , data ) ;
return err ;
2006-01-09 20:25:07 +03:00
}
return 0 ;
}
2008-03-30 04:01:12 +04:00
static int cx24123_i2c_readreg ( struct cx24123_state * state , u8 i2c_addr , u8 reg )
2006-01-09 20:25:07 +03:00
{
int ret ;
2008-03-30 04:01:12 +04:00
u8 b = 0 ;
2006-01-09 20:25:07 +03:00
struct i2c_msg msg [ ] = {
2008-03-30 04:01:12 +04:00
{ . addr = i2c_addr , . flags = 0 , . buf = & reg , . len = 1 } ,
{ . addr = i2c_addr , . flags = I2C_M_RD , . buf = & b , . len = 1 }
2006-01-09 20:25:07 +03:00
} ;
ret = i2c_transfer ( state - > i2c , msg , 2 ) ;
if ( ret ! = 2 ) {
2008-03-30 04:01:12 +04:00
err ( " %s: reg=0x%x (error=%d) \n " , __func__ , reg , ret ) ;
2006-01-09 20:25:07 +03:00
return ret ;
}
2008-03-30 04:01:12 +04:00
/* printk(KERN_DEBUG "rd(%02x): %02x %02x\n", i2c_addr, reg, b); */
2006-04-13 18:29:13 +04:00
2008-03-30 04:01:12 +04:00
return b ;
2006-01-09 20:25:07 +03:00
}
2008-03-30 04:01:12 +04:00
# define cx24123_readreg(state, reg) \
cx24123_i2c_readreg ( state , state - > config - > demod_address , reg )
# define cx24123_writereg(state, reg, val) \
cx24123_i2c_writereg ( state , state - > config - > demod_address , reg , val )
2008-10-17 03:28:32 +04:00
static int cx24123_set_inversion ( struct cx24123_state * state ,
fe_spectral_inversion_t inversion )
2006-01-09 20:25:07 +03:00
{
2006-04-14 00:24:13 +04:00
u8 nom_reg = cx24123_readreg ( state , 0x0e ) ;
u8 auto_reg = cx24123_readreg ( state , 0x10 ) ;
2006-01-09 20:25:07 +03:00
switch ( inversion ) {
case INVERSION_OFF :
2008-03-30 04:01:12 +04:00
dprintk ( " inversion off \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg & ~ 0x80 ) ;
cx24123_writereg ( state , 0x10 , auto_reg | 0x80 ) ;
2006-01-09 20:25:07 +03:00
break ;
case INVERSION_ON :
2008-03-30 04:01:12 +04:00
dprintk ( " inversion on \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg | 0x80 ) ;
cx24123_writereg ( state , 0x10 , auto_reg | 0x80 ) ;
2006-01-09 20:25:07 +03:00
break ;
case INVERSION_AUTO :
2008-03-30 04:01:12 +04:00
dprintk ( " inversion auto \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x10 , auto_reg & ~ 0x80 ) ;
2006-01-09 20:25:07 +03:00
break ;
default :
return - EINVAL ;
}
return 0 ;
}
2008-10-17 03:28:32 +04:00
static int cx24123_get_inversion ( struct cx24123_state * state ,
fe_spectral_inversion_t * inversion )
2006-01-09 20:25:07 +03:00
{
u8 val ;
val = cx24123_readreg ( state , 0x1b ) > > 7 ;
2006-04-13 18:29:13 +04:00
if ( val = = 0 ) {
2008-03-30 04:01:12 +04:00
dprintk ( " read inversion off \n " ) ;
2006-01-09 20:25:08 +03:00
* inversion = INVERSION_OFF ;
2006-04-13 18:29:13 +04:00
} else {
2008-03-30 04:01:12 +04:00
dprintk ( " read inversion on \n " ) ;
2006-01-09 20:25:08 +03:00
* inversion = INVERSION_ON ;
2006-04-13 18:29:13 +04:00
}
2006-01-09 20:25:07 +03:00
return 0 ;
}
2008-10-17 03:28:32 +04:00
static int cx24123_set_fec ( struct cx24123_state * state , fe_code_rate_t fec )
2006-01-09 20:25:07 +03:00
{
2006-04-14 00:24:13 +04:00
u8 nom_reg = cx24123_readreg ( state , 0x0e ) & ~ 0x07 ;
2012-10-27 23:14:01 +04:00
if ( ( ( int ) fec < FEC_NONE ) | | ( fec > FEC_AUTO ) )
2006-01-09 20:25:08 +03:00
fec = FEC_AUTO ;
2006-01-09 20:25:07 +03:00
2006-08-08 22:48:08 +04:00
/* Set the soft decision threshold */
2008-10-17 03:28:32 +04:00
if ( fec = = FEC_1_2 )
cx24123_writereg ( state , 0x43 ,
cx24123_readreg ( state , 0x43 ) | 0x01 ) ;
2006-08-08 22:48:08 +04:00
else
2008-10-17 03:28:32 +04:00
cx24123_writereg ( state , 0x43 ,
cx24123_readreg ( state , 0x43 ) & ~ 0x01 ) ;
2006-08-08 22:48:08 +04:00
2006-01-09 20:25:07 +03:00
switch ( fec ) {
case FEC_1_2 :
2008-03-30 04:01:12 +04:00
dprintk ( " set FEC to 1/2 \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg | 0x01 ) ;
cx24123_writereg ( state , 0x0f , 0x02 ) ;
break ;
2006-01-09 20:25:07 +03:00
case FEC_2_3 :
2008-03-30 04:01:12 +04:00
dprintk ( " set FEC to 2/3 \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg | 0x02 ) ;
cx24123_writereg ( state , 0x0f , 0x04 ) ;
break ;
2006-01-09 20:25:07 +03:00
case FEC_3_4 :
2008-03-30 04:01:12 +04:00
dprintk ( " set FEC to 3/4 \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg | 0x03 ) ;
cx24123_writereg ( state , 0x0f , 0x08 ) ;
break ;
case FEC_4_5 :
2008-03-30 04:01:12 +04:00
dprintk ( " set FEC to 4/5 \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg | 0x04 ) ;
cx24123_writereg ( state , 0x0f , 0x10 ) ;
break ;
case FEC_5_6 :
2008-03-30 04:01:12 +04:00
dprintk ( " set FEC to 5/6 \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg | 0x05 ) ;
cx24123_writereg ( state , 0x0f , 0x20 ) ;
break ;
case FEC_6_7 :
2008-03-30 04:01:12 +04:00
dprintk ( " set FEC to 6/7 \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg | 0x06 ) ;
cx24123_writereg ( state , 0x0f , 0x40 ) ;
break ;
case FEC_7_8 :
2008-03-30 04:01:12 +04:00
dprintk ( " set FEC to 7/8 \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0e , nom_reg | 0x07 ) ;
cx24123_writereg ( state , 0x0f , 0x80 ) ;
break ;
2006-01-09 20:25:07 +03:00
case FEC_AUTO :
2008-03-30 04:01:12 +04:00
dprintk ( " set FEC to auto \n " ) ;
2006-04-14 00:24:13 +04:00
cx24123_writereg ( state , 0x0f , 0xfe ) ;
break ;
2006-01-09 20:25:07 +03:00
default :
return - EOPNOTSUPP ;
}
2006-04-14 00:24:13 +04:00
return 0 ;
2006-01-09 20:25:07 +03:00
}
2008-10-17 03:28:32 +04:00
static int cx24123_get_fec ( struct cx24123_state * state , fe_code_rate_t * fec )
2006-01-09 20:25:07 +03:00
{
2006-01-09 20:25:08 +03:00
int ret ;
2006-01-09 20:25:07 +03:00
2008-10-17 03:28:32 +04:00
ret = cx24123_readreg ( state , 0x1b ) ;
2006-01-09 20:25:08 +03:00
if ( ret < 0 )
return ret ;
2006-04-13 17:19:52 +04:00
ret = ret & 0x07 ;
switch ( ret ) {
2006-01-09 20:25:07 +03:00
case 1 :
2006-01-09 20:25:08 +03:00
* fec = FEC_1_2 ;
break ;
2006-04-13 17:19:52 +04:00
case 2 :
2006-01-09 20:25:08 +03:00
* fec = FEC_2_3 ;
break ;
2006-04-13 17:19:52 +04:00
case 3 :
2006-01-09 20:25:08 +03:00
* fec = FEC_3_4 ;
break ;
2006-04-13 17:19:52 +04:00
case 4 :
2006-01-09 20:25:08 +03:00
* fec = FEC_4_5 ;
break ;
2006-04-13 17:19:52 +04:00
case 5 :
2006-01-09 20:25:08 +03:00
* fec = FEC_5_6 ;
break ;
2006-04-13 17:19:52 +04:00
case 6 :
* fec = FEC_6_7 ;
break ;
2006-01-09 20:25:07 +03:00
case 7 :
2006-01-09 20:25:08 +03:00
* fec = FEC_7_8 ;
break ;
2006-01-09 20:25:07 +03:00
default :
2006-04-14 00:24:13 +04:00
/* this can happen when there's no lock */
* fec = FEC_NONE ;
2006-01-09 20:25:07 +03:00
}
2006-01-09 20:25:08 +03:00
return 0 ;
2006-01-09 20:25:07 +03:00
}
2006-04-14 00:24:13 +04:00
/* Approximation of closest integer of log2(a/b). It actually gives the
lowest integer i such that 2 ^ i > = round ( a / b ) */
static u32 cx24123_int_log2 ( u32 a , u32 b )
{
u32 exp , nearest = 0 ;
u32 div = a / b ;
2008-10-17 03:28:32 +04:00
if ( a % b > = b / 2 )
+ + div ;
if ( div < ( 1 < < 31 ) ) {
for ( exp = 1 ; div > exp ; nearest + + )
2006-04-14 00:24:13 +04:00
exp + = exp ;
}
return nearest ;
}
2008-10-17 03:28:32 +04:00
static int cx24123_set_symbolrate ( struct cx24123_state * state , u32 srate )
2006-01-09 20:25:07 +03:00
{
2013-04-08 04:11:53 +04:00
u64 tmp ;
u32 sample_rate , ratio , sample_gain ;
2006-04-13 17:19:52 +04:00
u8 pll_mult ;
/* check if symbol rate is within limits */
2006-05-14 12:01:31 +04:00
if ( ( srate > state - > frontend . ops . info . symbol_rate_max ) | |
( srate < state - > frontend . ops . info . symbol_rate_min ) )
2009-07-02 22:57:09 +04:00
return - EOPNOTSUPP ;
2006-04-13 17:19:52 +04:00
/* choose the sampling rate high enough for the required operation,
while optimizing the power consumed by the demodulator */
if ( srate < ( XTAL * 2 ) / 2 )
pll_mult = 2 ;
else if ( srate < ( XTAL * 3 ) / 2 )
pll_mult = 3 ;
else if ( srate < ( XTAL * 4 ) / 2 )
pll_mult = 4 ;
else if ( srate < ( XTAL * 5 ) / 2 )
pll_mult = 5 ;
else if ( srate < ( XTAL * 6 ) / 2 )
pll_mult = 6 ;
else if ( srate < ( XTAL * 7 ) / 2 )
pll_mult = 7 ;
else if ( srate < ( XTAL * 8 ) / 2 )
pll_mult = 8 ;
else
pll_mult = 9 ;
sample_rate = pll_mult * XTAL ;
2006-01-09 20:25:07 +03:00
2013-04-08 04:11:53 +04:00
/* SYSSymbolRate[21:0] = (srate << 23) / sample_rate */
2006-04-13 17:19:52 +04:00
2013-04-08 04:11:53 +04:00
tmp = ( ( u64 ) srate ) < < 23 ;
do_div ( tmp , sample_rate ) ;
ratio = ( u32 ) tmp ;
2006-04-13 17:19:52 +04:00
cx24123_writereg ( state , 0x01 , pll_mult * 6 ) ;
2008-10-17 03:28:32 +04:00
cx24123_writereg ( state , 0x08 , ( ratio > > 16 ) & 0x3f ) ;
cx24123_writereg ( state , 0x09 , ( ratio > > 8 ) & 0xff ) ;
cx24123_writereg ( state , 0x0a , ratio & 0xff ) ;
2006-04-13 17:19:52 +04:00
2006-04-14 00:24:13 +04:00
/* also set the demodulator sample gain */
sample_gain = cx24123_int_log2 ( sample_rate , srate ) ;
tmp = cx24123_readreg ( state , 0x0c ) & ~ 0xe0 ;
cx24123_writereg ( state , 0x0c , tmp | sample_gain < < 5 ) ;
2008-03-30 04:01:12 +04:00
dprintk ( " srate=%d, ratio=0x%08x, sample_rate=%i sample_gain=%d \n " ,
srate , ratio , sample_rate , sample_gain ) ;
2006-01-09 20:25:07 +03:00
return 0 ;
}
/*
2008-10-17 03:28:32 +04:00
* Based on the required frequency and symbolrate , the tuner AGC has
* to be configured and the correct band selected .
* Calculate those values .
2006-01-09 20:25:07 +03:00
*/
2011-12-23 00:54:00 +04:00
static int cx24123_pll_calculate ( struct dvb_frontend * fe )
2006-01-09 20:25:07 +03:00
{
2011-12-23 00:54:00 +04:00
struct dtv_frontend_properties * p = & fe - > dtv_property_cache ;
2006-01-09 20:25:07 +03:00
struct cx24123_state * state = fe - > demodulator_priv ;
2006-01-09 20:25:08 +03:00
u32 ndiv = 0 , adiv = 0 , vco_div = 0 ;
int i = 0 ;
2006-04-13 17:19:52 +04:00
int pump = 2 ;
2006-04-14 00:26:22 +04:00
int band = 0 ;
2007-02-15 03:57:42 +03:00
int num_bands = ARRAY_SIZE ( cx24123_bandselect_vals ) ;
2008-10-17 03:28:32 +04:00
struct cx24123_bandselect_val * bsv = NULL ;
struct cx24123_AGC_val * agcv = NULL ;
2006-01-09 20:25:07 +03:00
/* Defaults for low freq, low rate */
state - > VCAarg = cx24123_AGC_vals [ 0 ] . VCAprogdata ;
state - > VGAarg = cx24123_AGC_vals [ 0 ] . VGAprogdata ;
state - > bandselectarg = cx24123_bandselect_vals [ 0 ] . progdata ;
vco_div = cx24123_bandselect_vals [ 0 ] . VCOdivider ;
2008-10-17 03:28:32 +04:00
/* For the given symbol rate, determine the VCA, VGA and
* FILTUNE programming bits */
for ( i = 0 ; i < ARRAY_SIZE ( cx24123_AGC_vals ) ; i + + ) {
agcv = & cx24123_AGC_vals [ i ] ;
2011-12-23 00:54:00 +04:00
if ( ( agcv - > symbolrate_low < = p - > symbol_rate ) & &
( agcv - > symbolrate_high > = p - > symbol_rate ) ) {
2008-10-17 03:28:32 +04:00
state - > VCAarg = agcv - > VCAprogdata ;
state - > VGAarg = agcv - > VGAprogdata ;
state - > FILTune = agcv - > FILTune ;
2006-01-09 20:25:07 +03:00
}
}
2006-04-14 00:26:22 +04:00
/* determine the band to use */
2008-10-17 03:28:32 +04:00
if ( force_band < 1 | | force_band > num_bands ) {
for ( i = 0 ; i < num_bands ; i + + ) {
bsv = & cx24123_bandselect_vals [ i ] ;
if ( ( bsv - > freq_low < = p - > frequency ) & &
( bsv - > freq_high > = p - > frequency ) )
2006-04-14 00:26:22 +04:00
band = i ;
2006-01-09 20:25:07 +03:00
}
2008-10-17 03:28:32 +04:00
} else
2006-04-14 00:26:22 +04:00
band = force_band - 1 ;
state - > bandselectarg = cx24123_bandselect_vals [ band ] . progdata ;
vco_div = cx24123_bandselect_vals [ band ] . VCOdivider ;
/* determine the charge pump current */
2008-10-17 03:28:32 +04:00
if ( p - > frequency < ( cx24123_bandselect_vals [ band ] . freq_low +
cx24123_bandselect_vals [ band ] . freq_high ) / 2 )
2006-04-14 00:26:22 +04:00
pump = 0x01 ;
else
pump = 0x02 ;
2006-01-09 20:25:07 +03:00
/* Determine the N/A dividers for the requested lband freq (in kHz). */
2008-10-17 03:28:32 +04:00
/* Note: the reference divider R=10, frequency is in KHz,
* XTAL is in Hz */
ndiv = ( ( ( p - > frequency * vco_div * 10 ) /
( 2 * XTAL / 1000 ) ) / 32 ) & 0x1ff ;
adiv = ( ( ( p - > frequency * vco_div * 10 ) /
( 2 * XTAL / 1000 ) ) % 32 ) & 0x1f ;
2006-01-09 20:25:07 +03:00
2006-10-03 04:35:40 +04:00
if ( adiv = = 0 & & ndiv > 0 )
ndiv - - ;
2006-01-09 20:25:07 +03:00
2008-10-17 03:28:32 +04:00
/* control bits 11, refdiv 11, charge pump polarity 1,
* charge pump current , ndiv , adiv */
state - > pllarg = ( 3 < < 19 ) | ( 3 < < 17 ) | ( 1 < < 16 ) |
( pump < < 14 ) | ( ndiv < < 5 ) | adiv ;
2006-01-09 20:25:07 +03:00
return 0 ;
}
/*
* Tuner data is 21 bits long , must be left - aligned in data .
2008-10-17 03:28:32 +04:00
* Tuner cx24109 is written through a dedicated 3 wire interface
* on the demod chip .
2006-01-09 20:25:07 +03:00
*/
2011-12-23 00:44:43 +04:00
static int cx24123_pll_writereg ( struct dvb_frontend * fe , u32 data )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
2006-01-09 20:25:22 +03:00
unsigned long timeout ;
2006-01-09 20:25:07 +03:00
2008-03-30 04:01:12 +04:00
dprintk ( " pll writereg called, data=0x%08x \n " , data ) ;
2006-04-13 18:29:13 +04:00
2006-01-09 20:25:07 +03:00
/* align the 21 bytes into to bit23 boundary */
data = data < < 3 ;
/* Reset the demod pll word length to 0x15 bits */
cx24123_writereg ( state , 0x21 , 0x15 ) ;
/* write the msb 8 bits, wait for the send to be completed */
2006-01-09 20:25:22 +03:00
timeout = jiffies + msecs_to_jiffies ( 40 ) ;
2006-01-09 20:25:08 +03:00
cx24123_writereg ( state , 0x22 , ( data > > 16 ) & 0xff ) ;
2006-01-09 20:25:22 +03:00
while ( ( cx24123_readreg ( state , 0x20 ) & 0x40 ) = = 0 ) {
if ( time_after ( jiffies , timeout ) ) {
2008-03-30 04:01:12 +04:00
err ( " %s: demodulator is not responding, " \
" possibly hung, aborting. \n " , __func__ ) ;
2006-01-09 20:25:07 +03:00
return - EREMOTEIO ;
}
2006-01-09 20:25:22 +03:00
msleep ( 10 ) ;
2006-01-09 20:25:07 +03:00
}
/* send another 8 bytes, wait for the send to be completed */
2006-01-09 20:25:22 +03:00
timeout = jiffies + msecs_to_jiffies ( 40 ) ;
2008-10-17 03:28:32 +04:00
cx24123_writereg ( state , 0x22 , ( data > > 8 ) & 0xff ) ;
2006-01-09 20:25:22 +03:00
while ( ( cx24123_readreg ( state , 0x20 ) & 0x40 ) = = 0 ) {
if ( time_after ( jiffies , timeout ) ) {
2008-03-30 04:01:12 +04:00
err ( " %s: demodulator is not responding, " \
" possibly hung, aborting. \n " , __func__ ) ;
2006-01-09 20:25:07 +03:00
return - EREMOTEIO ;
}
2006-01-09 20:25:22 +03:00
msleep ( 10 ) ;
2006-01-09 20:25:07 +03:00
}
2008-10-17 03:28:32 +04:00
/* send the lower 5 bits of this byte, padded with 3 LBB,
* wait for the send to be completed */
2006-01-09 20:25:22 +03:00
timeout = jiffies + msecs_to_jiffies ( 40 ) ;
2008-10-17 03:28:32 +04:00
cx24123_writereg ( state , 0x22 , ( data ) & 0xff ) ;
2006-01-09 20:25:22 +03:00
while ( ( cx24123_readreg ( state , 0x20 ) & 0x80 ) ) {
if ( time_after ( jiffies , timeout ) ) {
2008-03-30 04:01:12 +04:00
err ( " %s: demodulator is not responding, " \
" possibly hung, aborting. \n " , __func__ ) ;
2006-01-09 20:25:07 +03:00
return - EREMOTEIO ;
}
2006-01-09 20:25:22 +03:00
msleep ( 10 ) ;
2006-01-09 20:25:07 +03:00
}
/* Trigger the demod to configure the tuner */
cx24123_writereg ( state , 0x20 , cx24123_readreg ( state , 0x20 ) | 2 ) ;
cx24123_writereg ( state , 0x20 , cx24123_readreg ( state , 0x20 ) & 0xfd ) ;
return 0 ;
}
2011-12-23 00:54:00 +04:00
static int cx24123_pll_tune ( struct dvb_frontend * fe )
2006-01-09 20:25:07 +03:00
{
2011-12-23 00:54:00 +04:00
struct dtv_frontend_properties * p = & fe - > dtv_property_cache ;
2006-01-09 20:25:07 +03:00
struct cx24123_state * state = fe - > demodulator_priv ;
2006-04-13 17:19:52 +04:00
u8 val ;
dprintk ( " frequency=%i \n " , p - > frequency ) ;
2006-01-09 20:25:07 +03:00
2011-12-23 00:54:00 +04:00
if ( cx24123_pll_calculate ( fe ) ! = 0 ) {
2008-03-30 04:01:12 +04:00
err ( " %s: cx24123_pll_calcutate failed \n " , __func__ ) ;
2006-01-09 20:25:07 +03:00
return - EINVAL ;
}
/* Write the new VCO/VGA */
2011-12-23 00:44:43 +04:00
cx24123_pll_writereg ( fe , state - > VCAarg ) ;
cx24123_pll_writereg ( fe , state - > VGAarg ) ;
2006-01-09 20:25:07 +03:00
/* Write the new bandselect and pll args */
2011-12-23 00:44:43 +04:00
cx24123_pll_writereg ( fe , state - > bandselectarg ) ;
cx24123_pll_writereg ( fe , state - > pllarg ) ;
2006-01-09 20:25:07 +03:00
2006-04-13 17:19:52 +04:00
/* set the FILTUNE voltage */
val = cx24123_readreg ( state , 0x28 ) & ~ 0x3 ;
cx24123_writereg ( state , 0x27 , state - > FILTune > > 2 ) ;
cx24123_writereg ( state , 0x28 , val | ( state - > FILTune & 0x3 ) ) ;
2008-03-30 04:01:12 +04:00
dprintk ( " pll tune VCA=%d, band=%d, pll=%d \n " , state - > VCAarg ,
state - > bandselectarg , state - > pllarg ) ;
2006-04-13 18:29:13 +04:00
2006-01-09 20:25:07 +03:00
return 0 ;
}
2008-03-30 04:01:12 +04:00
/*
* 0x23 :
* [ 7 : 7 ] = BTI enabled
* [ 6 : 6 ] = I2C repeater enabled
* [ 5 : 5 ] = I2C repeater start
* [ 0 : 0 ] = BTI start
*/
/* mode == 1 -> i2c-repeater, 0 -> bti */
static int cx24123_repeater_mode ( struct cx24123_state * state , u8 mode , u8 start )
{
u8 r = cx24123_readreg ( state , 0x23 ) & 0x1e ;
if ( mode )
r | = ( 1 < < 6 ) | ( start < < 5 ) ;
else
r | = ( 1 < < 7 ) | ( start ) ;
return cx24123_writereg ( state , 0x23 , r ) ;
}
2008-10-17 03:28:32 +04:00
static int cx24123_initfe ( struct dvb_frontend * fe )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
int i ;
2008-03-30 04:01:12 +04:00
dprintk ( " init frontend \n " ) ;
2006-04-13 18:29:13 +04:00
2006-01-09 20:25:07 +03:00
/* Configure the demod to a good set of defaults */
2007-02-15 03:57:42 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( cx24123_regdata ) ; i + + )
2008-10-17 03:28:32 +04:00
cx24123_writereg ( state , cx24123_regdata [ i ] . reg ,
cx24123_regdata [ i ] . data ) ;
2006-01-09 20:25:07 +03:00
2006-09-26 19:30:14 +04:00
/* Set the LNB polarity */
2008-10-17 03:28:32 +04:00
if ( state - > config - > lnb_polarity )
cx24123_writereg ( state , 0x32 ,
cx24123_readreg ( state , 0x32 ) | 0x02 ) ;
2006-09-26 19:30:14 +04:00
2008-03-30 04:01:12 +04:00
if ( state - > config - > dont_use_pll )
2008-10-17 03:28:32 +04:00
cx24123_repeater_mode ( state , 1 , 0 ) ;
2008-03-30 04:01:12 +04:00
2006-01-09 20:25:07 +03:00
return 0 ;
}
2008-10-17 03:28:32 +04:00
static int cx24123_set_voltage ( struct dvb_frontend * fe ,
fe_sec_voltage_t voltage )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
u8 val ;
2006-05-13 03:31:51 +04:00
val = cx24123_readreg ( state , 0x29 ) & ~ 0x40 ;
2006-01-09 20:25:08 +03:00
2006-05-13 03:31:51 +04:00
switch ( voltage ) {
case SEC_VOLTAGE_13 :
2008-03-30 04:01:12 +04:00
dprintk ( " setting voltage 13V \n " ) ;
2006-06-30 03:29:29 +04:00
return cx24123_writereg ( state , 0x29 , val & 0x7f ) ;
2006-05-13 03:31:51 +04:00
case SEC_VOLTAGE_18 :
2008-03-30 04:01:12 +04:00
dprintk ( " setting voltage 18V \n " ) ;
2006-06-30 03:29:29 +04:00
return cx24123_writereg ( state , 0x29 , val | 0x80 ) ;
2006-09-26 19:30:14 +04:00
case SEC_VOLTAGE_OFF :
/* already handled in cx88-dvb */
return 0 ;
2006-05-13 03:31:51 +04:00
default :
return - EINVAL ;
2013-10-09 03:29:08 +04:00
}
2006-01-09 20:25:08 +03:00
return 0 ;
2006-01-09 20:25:07 +03:00
}
2006-04-13 18:40:59 +04:00
/* wait for diseqc queue to become ready (or timeout) */
static void cx24123_wait_for_diseqc ( struct cx24123_state * state )
{
unsigned long timeout = jiffies + msecs_to_jiffies ( 200 ) ;
while ( ! ( cx24123_readreg ( state , 0x29 ) & 0x40 ) ) {
2008-10-17 03:28:32 +04:00
if ( time_after ( jiffies , timeout ) ) {
2008-03-30 04:01:12 +04:00
err ( " %s: diseqc queue not ready, " \
" command may be lost. \n " , __func__ ) ;
2006-04-13 18:40:59 +04:00
break ;
}
msleep ( 10 ) ;
}
}
2008-10-17 03:28:32 +04:00
static int cx24123_send_diseqc_msg ( struct dvb_frontend * fe ,
struct dvb_diseqc_master_cmd * cmd )
2006-01-09 20:25:07 +03:00
{
2006-04-13 17:19:52 +04:00
struct cx24123_state * state = fe - > demodulator_priv ;
2006-05-13 03:31:51 +04:00
int i , val , tone ;
2006-04-13 17:19:52 +04:00
2008-03-30 04:01:12 +04:00
dprintk ( " \n " ) ;
2006-01-09 20:25:07 +03:00
2006-05-13 03:31:51 +04:00
/* stop continuous tone if enabled */
tone = cx24123_readreg ( state , 0x29 ) ;
if ( tone & 0x10 )
cx24123_writereg ( state , 0x29 , tone & ~ 0x50 ) ;
2006-04-13 17:19:52 +04:00
2006-04-13 18:40:59 +04:00
/* wait for diseqc queue ready */
cx24123_wait_for_diseqc ( state ) ;
2006-04-13 17:19:52 +04:00
/* select tone mode */
2006-05-13 03:31:51 +04:00
cx24123_writereg ( state , 0x2a , cx24123_readreg ( state , 0x2a ) & 0xfb ) ;
2006-04-13 17:19:52 +04:00
for ( i = 0 ; i < cmd - > msg_len ; i + + )
cx24123_writereg ( state , 0x2C + i , cmd - > msg [ i ] ) ;
val = cx24123_readreg ( state , 0x29 ) ;
2008-10-17 03:28:32 +04:00
cx24123_writereg ( state , 0x29 , ( ( val & 0x90 ) | 0x40 ) |
( ( cmd - > msg_len - 3 ) & 3 ) ) ;
2006-04-13 17:19:52 +04:00
2006-04-13 18:40:59 +04:00
/* wait for diseqc message to finish sending */
cx24123_wait_for_diseqc ( state ) ;
2006-04-13 17:19:52 +04:00
2006-05-13 03:31:51 +04:00
/* restart continuous tone if enabled */
2008-10-17 03:28:32 +04:00
if ( tone & 0x10 )
2006-05-13 03:31:51 +04:00
cx24123_writereg ( state , 0x29 , tone & ~ 0x40 ) ;
2006-04-13 17:19:52 +04:00
return 0 ;
}
2008-10-17 03:28:32 +04:00
static int cx24123_diseqc_send_burst ( struct dvb_frontend * fe ,
fe_sec_mini_cmd_t burst )
2006-04-13 17:19:52 +04:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
2006-05-13 03:31:51 +04:00
int val , tone ;
2006-04-13 17:19:52 +04:00
2008-03-30 04:01:12 +04:00
dprintk ( " \n " ) ;
2006-04-13 17:19:52 +04:00
2006-05-13 03:31:51 +04:00
/* stop continuous tone if enabled */
tone = cx24123_readreg ( state , 0x29 ) ;
if ( tone & 0x10 )
cx24123_writereg ( state , 0x29 , tone & ~ 0x50 ) ;
2006-04-13 17:19:52 +04:00
2006-05-13 03:31:51 +04:00
/* wait for diseqc queue ready */
2006-04-13 18:40:59 +04:00
cx24123_wait_for_diseqc ( state ) ;
2006-04-13 17:19:52 +04:00
/* select tone mode */
2006-05-13 03:31:51 +04:00
cx24123_writereg ( state , 0x2a , cx24123_readreg ( state , 0x2a ) | 0x4 ) ;
msleep ( 30 ) ;
2006-04-13 17:19:52 +04:00
val = cx24123_readreg ( state , 0x29 ) ;
if ( burst = = SEC_MINI_A )
cx24123_writereg ( state , 0x29 , ( ( val & 0x90 ) | 0x40 | 0x00 ) ) ;
else if ( burst = = SEC_MINI_B )
cx24123_writereg ( state , 0x29 , ( ( val & 0x90 ) | 0x40 | 0x08 ) ) ;
else
return - EINVAL ;
2006-04-13 18:40:59 +04:00
cx24123_wait_for_diseqc ( state ) ;
2006-05-13 03:31:51 +04:00
cx24123_writereg ( state , 0x2a , cx24123_readreg ( state , 0x2a ) & 0xfb ) ;
2006-04-13 17:19:52 +04:00
2006-05-13 03:31:51 +04:00
/* restart continuous tone if enabled */
2008-10-17 03:28:32 +04:00
if ( tone & 0x10 )
2006-05-13 03:31:51 +04:00
cx24123_writereg ( state , 0x29 , tone & ~ 0x40 ) ;
2008-10-17 03:28:32 +04:00
2006-04-13 17:19:52 +04:00
return 0 ;
2006-01-09 20:25:07 +03:00
}
2008-10-17 03:28:32 +04:00
static int cx24123_read_status ( struct dvb_frontend * fe , fe_status_t * status )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
int sync = cx24123_readreg ( state , 0x14 ) ;
* status = 0 ;
2008-03-30 04:01:12 +04:00
if ( state - > config - > dont_use_pll ) {
u32 tun_status = 0 ;
if ( fe - > ops . tuner_ops . get_status )
fe - > ops . tuner_ops . get_status ( fe , & tun_status ) ;
if ( tun_status & TUNER_STATUS_LOCKED )
* status | = FE_HAS_SIGNAL ;
} else {
int lock = cx24123_readreg ( state , 0x20 ) ;
if ( lock & 0x01 )
* status | = FE_HAS_SIGNAL ;
}
2006-04-13 17:19:52 +04:00
if ( sync & 0x02 )
2006-08-07 00:03:50 +04:00
* status | = FE_HAS_CARRIER ; /* Phase locked */
2006-01-09 20:25:07 +03:00
if ( sync & 0x04 )
* status | = FE_HAS_VITERBI ;
2006-08-07 00:03:50 +04:00
/* Reed-Solomon Status */
2006-01-09 20:25:07 +03:00
if ( sync & 0x08 )
2006-04-13 17:19:52 +04:00
* status | = FE_HAS_SYNC ;
2006-01-09 20:25:07 +03:00
if ( sync & 0x80 )
2006-08-07 00:03:50 +04:00
* status | = FE_HAS_LOCK ; /*Full Sync */
2006-01-09 20:25:07 +03:00
return 0 ;
}
/*
2008-10-17 03:28:32 +04:00
* Configured to return the measurement of errors in blocks ,
* because no UCBLOCKS value is available , so this value doubles up
* to satisfy both measurements .
2006-01-09 20:25:07 +03:00
*/
2008-03-30 04:01:12 +04:00
static int cx24123_read_ber ( struct dvb_frontend * fe , u32 * ber )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
2006-08-08 22:48:08 +04:00
/* The true bit error rate is this value divided by
the window size ( set as 256 * 255 ) */
* ber = ( ( cx24123_readreg ( state , 0x1c ) & 0x3f ) < < 16 ) |
2006-01-09 20:25:07 +03:00
( cx24123_readreg ( state , 0x1d ) < < 8 |
2006-08-08 22:48:08 +04:00
cx24123_readreg ( state , 0x1e ) ) ;
2006-04-13 18:29:13 +04:00
2008-03-30 04:01:12 +04:00
dprintk ( " BER = %d \n " , * ber ) ;
2006-01-09 20:25:07 +03:00
return 0 ;
}
2008-03-30 04:01:12 +04:00
static int cx24123_read_signal_strength ( struct dvb_frontend * fe ,
u16 * signal_strength )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
2006-08-07 00:03:50 +04:00
2008-10-17 03:28:32 +04:00
/* larger = better */
* signal_strength = cx24123_readreg ( state , 0x3b ) < < 8 ;
2006-01-09 20:25:07 +03:00
2008-03-30 04:01:12 +04:00
dprintk ( " Signal strength = %d \n " , * signal_strength ) ;
2006-04-13 18:29:13 +04:00
2006-01-09 20:25:07 +03:00
return 0 ;
}
2008-03-30 04:01:12 +04:00
static int cx24123_read_snr ( struct dvb_frontend * fe , u16 * snr )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
2006-08-08 22:48:08 +04:00
/* Inverted raw Es/N0 count, totally bogus but better than the
BER threshold . */
* snr = 65535 - ( ( ( u16 ) cx24123_readreg ( state , 0x18 ) < < 8 ) |
( u16 ) cx24123_readreg ( state , 0x19 ) ) ;
2006-04-13 18:29:13 +04:00
2008-03-30 04:01:12 +04:00
dprintk ( " read S/N index = %d \n " , * snr ) ;
2006-04-13 18:29:13 +04:00
2006-01-09 20:25:07 +03:00
return 0 ;
}
2011-12-23 00:54:00 +04:00
static int cx24123_set_frontend ( struct dvb_frontend * fe )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
2011-12-23 00:54:00 +04:00
struct dtv_frontend_properties * p = & fe - > dtv_property_cache ;
2006-01-09 20:25:07 +03:00
2008-03-30 04:01:12 +04:00
dprintk ( " \n " ) ;
2006-04-13 18:29:13 +04:00
2006-01-09 20:25:07 +03:00
if ( state - > config - > set_ts_params )
state - > config - > set_ts_params ( fe , 0 ) ;
2008-10-17 03:28:32 +04:00
state - > currentfreq = p - > frequency ;
2011-12-23 00:54:00 +04:00
state - > currentsymbolrate = p - > symbol_rate ;
2006-01-09 20:25:07 +03:00
cx24123_set_inversion ( state , p - > inversion ) ;
2011-12-23 00:54:00 +04:00
cx24123_set_fec ( state , p - > fec_inner ) ;
cx24123_set_symbolrate ( state , p - > symbol_rate ) ;
2008-03-30 04:01:12 +04:00
if ( ! state - > config - > dont_use_pll )
2011-12-23 00:54:00 +04:00
cx24123_pll_tune ( fe ) ;
2008-03-30 04:01:12 +04:00
else if ( fe - > ops . tuner_ops . set_params )
2011-12-24 19:24:33 +04:00
fe - > ops . tuner_ops . set_params ( fe ) ;
2008-03-30 04:01:12 +04:00
else
err ( " it seems I don't have a tuner... " ) ;
2006-01-09 20:25:07 +03:00
2011-03-31 05:57:33 +04:00
/* Enable automatic acquisition and reset cycle */
2006-01-09 20:25:08 +03:00
cx24123_writereg ( state , 0x03 , ( cx24123_readreg ( state , 0x03 ) | 0x07 ) ) ;
2006-01-09 20:25:07 +03:00
cx24123_writereg ( state , 0x00 , 0x10 ) ;
cx24123_writereg ( state , 0x00 , 0 ) ;
2008-03-30 04:01:12 +04:00
if ( state - > config - > agc_callback )
state - > config - > agc_callback ( fe ) ;
2006-01-09 20:25:07 +03:00
return 0 ;
}
[media] dvb: don't require a parameter for get_frontend
Just like set_frontend, use the dvb cache properties for get_frontend.
This is more consistent, as both functions are now symetric. Also,
at the places get_frontend is called, it makes sense to update the
cache.
Most of this patch were generated by this small perl script:
while (<>) { $file .= $_; }
if ($file =~ m/\.get_frontend\s*=\s*([\d\w_]+)/) {
my $get = $1;
$file =~ s/($get)(\s*\([^\,\)]+)\,\s*struct\s+dtv_frontend_properties\s*\*\s*([_\d\w]+)\)\s*\{/\1\2)\n{\n\tstruct dtv_frontend_properties *\3 = &fe->dtv_property_cache;/g;
}
print $file;
Of course, the changes at dvb_frontend.[ch] were made by hand,
as well as the changes on a few other places, where get_frontend()
is called internally inside the driver.
On some places, get_frontend() were just a void function. Those
occurrences were removed, as the DVB core handles such cases.
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2011-12-30 18:30:21 +04:00
static int cx24123_get_frontend ( struct dvb_frontend * fe )
2006-01-09 20:25:07 +03:00
{
[media] dvb: don't require a parameter for get_frontend
Just like set_frontend, use the dvb cache properties for get_frontend.
This is more consistent, as both functions are now symetric. Also,
at the places get_frontend is called, it makes sense to update the
cache.
Most of this patch were generated by this small perl script:
while (<>) { $file .= $_; }
if ($file =~ m/\.get_frontend\s*=\s*([\d\w_]+)/) {
my $get = $1;
$file =~ s/($get)(\s*\([^\,\)]+)\,\s*struct\s+dtv_frontend_properties\s*\*\s*([_\d\w]+)\)\s*\{/\1\2)\n{\n\tstruct dtv_frontend_properties *\3 = &fe->dtv_property_cache;/g;
}
print $file;
Of course, the changes at dvb_frontend.[ch] were made by hand,
as well as the changes on a few other places, where get_frontend()
is called internally inside the driver.
On some places, get_frontend() were just a void function. Those
occurrences were removed, as the DVB core handles such cases.
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2011-12-30 18:30:21 +04:00
struct dtv_frontend_properties * p = & fe - > dtv_property_cache ;
2006-01-09 20:25:07 +03:00
struct cx24123_state * state = fe - > demodulator_priv ;
2008-03-30 04:01:12 +04:00
dprintk ( " \n " ) ;
2006-04-13 18:29:13 +04:00
2006-01-09 20:25:07 +03:00
if ( cx24123_get_inversion ( state , & p - > inversion ) ! = 0 ) {
2008-03-30 04:01:12 +04:00
err ( " %s: Failed to get inversion status \n " , __func__ ) ;
2006-01-09 20:25:07 +03:00
return - EREMOTEIO ;
}
2011-12-23 00:54:00 +04:00
if ( cx24123_get_fec ( state , & p - > fec_inner ) ! = 0 ) {
2008-03-30 04:01:12 +04:00
err ( " %s: Failed to get fec status \n " , __func__ ) ;
2006-01-09 20:25:07 +03:00
return - EREMOTEIO ;
}
p - > frequency = state - > currentfreq ;
2011-12-23 00:54:00 +04:00
p - > symbol_rate = state - > currentsymbolrate ;
2006-01-09 20:25:07 +03:00
return 0 ;
}
2008-10-17 03:28:32 +04:00
static int cx24123_set_tone ( struct dvb_frontend * fe , fe_sec_tone_mode_t tone )
2006-01-09 20:25:07 +03:00
{
struct cx24123_state * state = fe - > demodulator_priv ;
u8 val ;
2006-05-13 03:31:51 +04:00
/* wait for diseqc queue ready */
cx24123_wait_for_diseqc ( state ) ;
2006-01-09 20:25:08 +03:00
2006-05-13 03:31:51 +04:00
val = cx24123_readreg ( state , 0x29 ) & ~ 0x40 ;
2006-01-09 20:25:08 +03:00
2006-05-13 03:31:51 +04:00
switch ( tone ) {
case SEC_TONE_ON :
2008-03-30 04:01:12 +04:00
dprintk ( " setting tone on \n " ) ;
2006-05-13 03:31:51 +04:00
return cx24123_writereg ( state , 0x29 , val | 0x10 ) ;
case SEC_TONE_OFF :
2008-03-30 04:01:12 +04:00
dprintk ( " setting tone off \n " ) ;
2006-05-13 03:31:51 +04:00
return cx24123_writereg ( state , 0x29 , val & 0xef ) ;
default :
2008-03-30 04:01:12 +04:00
err ( " CASE reached default with tone=%d \n " , tone ) ;
2006-05-13 03:31:51 +04:00
return - EINVAL ;
2006-01-09 20:25:07 +03:00
}
2006-01-09 20:25:08 +03:00
return 0 ;
2006-01-09 20:25:07 +03:00
}
2008-10-17 03:28:32 +04:00
static int cx24123_tune ( struct dvb_frontend * fe ,
2011-12-27 00:48:33 +04:00
bool re_tune ,
2006-08-08 22:48:08 +04:00
unsigned int mode_flags ,
2007-07-16 16:27:20 +04:00
unsigned int * delay ,
2006-08-08 22:48:08 +04:00
fe_status_t * status )
{
int retval = 0 ;
2011-12-27 00:48:33 +04:00
if ( re_tune )
2011-12-23 00:54:00 +04:00
retval = cx24123_set_frontend ( fe ) ;
2006-08-08 22:48:08 +04:00
if ( ! ( mode_flags & FE_TUNE_MODE_ONESHOT ) )
cx24123_read_status ( fe , status ) ;
* delay = HZ / 10 ;
return retval ;
}
static int cx24123_get_algo ( struct dvb_frontend * fe )
{
2008-10-17 03:28:32 +04:00
return 1 ; /* FE_ALGO_HW */
2006-08-08 22:48:08 +04:00
}
2008-10-17 03:28:32 +04:00
static void cx24123_release ( struct dvb_frontend * fe )
2006-01-09 20:25:07 +03:00
{
2008-10-17 03:28:32 +04:00
struct cx24123_state * state = fe - > demodulator_priv ;
2008-03-30 04:01:12 +04:00
dprintk ( " \n " ) ;
i2c_del_adapter ( & state - > tuner_i2c_adapter ) ;
2006-01-09 20:25:07 +03:00
kfree ( state ) ;
}
2008-03-30 04:01:12 +04:00
static int cx24123_tuner_i2c_tuner_xfer ( struct i2c_adapter * i2c_adap ,
struct i2c_msg msg [ ] , int num )
{
struct cx24123_state * state = i2c_get_adapdata ( i2c_adap ) ;
/* this repeater closes after the first stop */
2008-10-17 03:28:32 +04:00
cx24123_repeater_mode ( state , 1 , 1 ) ;
2008-03-30 04:01:12 +04:00
return i2c_transfer ( state - > i2c , msg , num ) ;
}
static u32 cx24123_tuner_i2c_func ( struct i2c_adapter * adapter )
{
return I2C_FUNC_I2C ;
}
static struct i2c_algorithm cx24123_tuner_i2c_algo = {
. master_xfer = cx24123_tuner_i2c_tuner_xfer ,
. functionality = cx24123_tuner_i2c_func ,
} ;
struct i2c_adapter *
cx24123_get_tuner_i2c_adapter ( struct dvb_frontend * fe )
{
struct cx24123_state * state = fe - > demodulator_priv ;
return & state - > tuner_i2c_adapter ;
}
EXPORT_SYMBOL ( cx24123_get_tuner_i2c_adapter ) ;
2006-01-09 20:25:07 +03:00
static struct dvb_frontend_ops cx24123_ops ;
2008-10-17 03:28:32 +04:00
struct dvb_frontend * cx24123_attach ( const struct cx24123_config * config ,
struct i2c_adapter * i2c )
2006-01-09 20:25:07 +03:00
{
2009-02-23 18:26:38 +03:00
/* allocate memory for the internal state */
2008-03-30 04:01:12 +04:00
struct cx24123_state * state =
kzalloc ( sizeof ( struct cx24123_state ) , GFP_KERNEL ) ;
2006-01-09 20:25:07 +03:00
2008-03-30 04:01:12 +04:00
dprintk ( " \n " ) ;
2006-01-09 20:25:07 +03:00
if ( state = = NULL ) {
2009-02-23 18:26:38 +03:00
err ( " Unable to kzalloc \n " ) ;
2006-01-09 20:25:07 +03:00
goto error ;
}
/* setup the state */
state - > config = config ;
state - > i2c = i2c ;
/* check if the demod is there */
2008-03-30 04:01:12 +04:00
state - > demod_rev = cx24123_readreg ( state , 0x00 ) ;
switch ( state - > demod_rev ) {
2008-10-17 03:28:32 +04:00
case 0xe1 :
info ( " detected CX24123C \n " ) ;
break ;
case 0xd1 :
info ( " detected CX24123 \n " ) ;
break ;
2008-03-30 04:01:12 +04:00
default :
err ( " wrong demod revision: %x \n " , state - > demod_rev ) ;
2006-01-09 20:25:07 +03:00
goto error ;
}
/* create dvb_frontend */
2008-10-17 03:28:32 +04:00
memcpy ( & state - > frontend . ops , & cx24123_ops ,
sizeof ( struct dvb_frontend_ops ) ) ;
2006-01-09 20:25:07 +03:00
state - > frontend . demodulator_priv = state ;
2008-03-30 04:01:12 +04:00
2008-10-17 03:28:32 +04:00
/* create tuner i2c adapter */
if ( config - > dont_use_pll )
cx24123_repeater_mode ( state , 1 , 0 ) ;
2008-03-30 04:01:12 +04:00
2008-09-04 00:12:23 +04:00
strlcpy ( state - > tuner_i2c_adapter . name , " CX24123 tuner I2C bus " ,
sizeof ( state - > tuner_i2c_adapter . name ) ) ;
2008-03-30 04:01:12 +04:00
state - > tuner_i2c_adapter . algo = & cx24123_tuner_i2c_algo ;
state - > tuner_i2c_adapter . algo_data = NULL ;
2014-09-20 16:36:26 +04:00
state - > tuner_i2c_adapter . dev . parent = i2c - > dev . parent ;
2008-03-30 04:01:12 +04:00
i2c_set_adapdata ( & state - > tuner_i2c_adapter , state ) ;
if ( i2c_add_adapter ( & state - > tuner_i2c_adapter ) < 0 ) {
2008-10-17 03:28:32 +04:00
err ( " tuner i2c bus could not be initialized \n " ) ;
2008-03-30 04:01:12 +04:00
goto error ;
}
2006-01-09 20:25:07 +03:00
return & state - > frontend ;
error :
kfree ( state ) ;
return NULL ;
}
2008-10-17 03:28:32 +04:00
EXPORT_SYMBOL ( cx24123_attach ) ;
2006-01-09 20:25:07 +03:00
static struct dvb_frontend_ops cx24123_ops = {
2011-12-23 00:54:00 +04:00
. delsys = { SYS_DVBS } ,
2006-01-09 20:25:07 +03:00
. info = {
. name = " Conexant CX24123/CX24109 " ,
. frequency_min = 950000 ,
. frequency_max = 2150000 ,
. frequency_stepsize = 1011 , /* kHz for QPSK frontends */
2006-04-14 00:24:13 +04:00
. frequency_tolerance = 5000 ,
2006-01-09 20:25:07 +03:00
. symbol_rate_min = 1000000 ,
. symbol_rate_max = 45000000 ,
. caps = FE_CAN_INVERSION_AUTO |
FE_CAN_FEC_1_2 | FE_CAN_FEC_2_3 | FE_CAN_FEC_3_4 |
2006-04-14 00:24:13 +04:00
FE_CAN_FEC_4_5 | FE_CAN_FEC_5_6 | FE_CAN_FEC_6_7 |
FE_CAN_FEC_7_8 | FE_CAN_FEC_AUTO |
2006-01-09 20:25:07 +03:00
FE_CAN_QPSK | FE_CAN_RECOVER
} ,
. release = cx24123_release ,
. init = cx24123_initfe ,
2011-12-23 00:54:00 +04:00
. set_frontend = cx24123_set_frontend ,
. get_frontend = cx24123_get_frontend ,
2006-01-09 20:25:07 +03:00
. read_status = cx24123_read_status ,
. read_ber = cx24123_read_ber ,
. read_signal_strength = cx24123_read_signal_strength ,
. read_snr = cx24123_read_snr ,
. diseqc_send_master_cmd = cx24123_send_diseqc_msg ,
2006-04-13 17:19:52 +04:00
. diseqc_send_burst = cx24123_diseqc_send_burst ,
2006-01-09 20:25:07 +03:00
. set_tone = cx24123_set_tone ,
. set_voltage = cx24123_set_voltage ,
2006-08-08 22:48:08 +04:00
. tune = cx24123_tune ,
. get_frontend_algo = cx24123_get_algo ,
2006-01-09 20:25:07 +03:00
} ;
2008-03-30 04:01:12 +04:00
MODULE_DESCRIPTION ( " DVB Frontend module for Conexant " \
" CX24123/CX24109/CX24113 hardware " ) ;
2006-01-09 20:25:07 +03:00
MODULE_AUTHOR ( " Steven Toth " ) ;
MODULE_LICENSE ( " GPL " ) ;