2019-05-20 10:19:02 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-17 02:20:36 +04:00
/*
Conexant 22702 DVB OFDM demodulator driver
based on :
2005-12-12 11:37:24 +03:00
Alps TDMB7 DVB OFDM demodulator driver
2005-04-17 02:20:36 +04:00
Copyright ( C ) 2001 - 2002 Convergence Integrated Media GmbH
Holger Waechtler < holger @ convergence . de >
2008-09-04 00:12:12 +04:00
Copyright ( C ) 2004 Steven Toth < stoth @ linuxtv . org >
2005-04-17 02:20:36 +04:00
*/
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/module.h>
# include <linux/string.h>
# include <linux/slab.h>
# include <linux/delay.h>
2017-12-28 21:03:51 +03:00
# include <media/dvb_frontend.h>
2005-04-17 02:20:36 +04:00
# include "cx22702.h"
struct cx22702_state {
2008-10-17 03:23:45 +04:00
struct i2c_adapter * i2c ;
2005-04-17 02:20:36 +04:00
/* configuration settings */
2008-10-17 03:23:45 +04:00
const struct cx22702_config * config ;
2005-04-17 02:20:36 +04:00
struct dvb_frontend frontend ;
/* previous uncorrected block counter */
u8 prevUCBlocks ;
} ;
2008-04-22 21:41:48 +04:00
static int debug ;
2008-10-17 03:23:45 +04:00
module_param ( debug , int , 0644 ) ;
MODULE_PARM_DESC ( debug , " Enable verbose debug messages " ) ;
2005-04-17 02:20:36 +04:00
# define dprintk if (debug) printk
/* Register values to initialise the demod */
2010-09-10 17:35:12 +04:00
static const u8 init_tab [ ] = {
2011-03-31 05:57:33 +04:00
0x00 , 0x00 , /* Stop acquisition */
2005-04-17 02:20:36 +04:00
0x0B , 0x06 ,
0x09 , 0x01 ,
0x0D , 0x41 ,
0x16 , 0x32 ,
0x20 , 0x0A ,
0x21 , 0x17 ,
0x24 , 0x3e ,
0x26 , 0xff ,
0x27 , 0x10 ,
0x28 , 0x00 ,
0x29 , 0x00 ,
0x2a , 0x10 ,
0x2b , 0x00 ,
0x2c , 0x10 ,
0x2d , 0x00 ,
0x48 , 0xd4 ,
0x49 , 0x56 ,
0x6b , 0x1e ,
0xc8 , 0x02 ,
0xf9 , 0x00 ,
0xfa , 0x00 ,
0xfb , 0x00 ,
0xfc , 0x00 ,
0xfd , 0x00 ,
} ;
2008-10-17 03:23:45 +04:00
static int cx22702_writereg ( struct cx22702_state * state , u8 reg , u8 data )
2005-04-17 02:20:36 +04:00
{
int ret ;
2008-10-17 03:23:45 +04:00
u8 buf [ ] = { reg , data } ;
struct i2c_msg msg = {
. addr = state - > config - > demod_address , . flags = 0 ,
. buf = buf , . len = 2 } ;
2005-04-17 02:20:36 +04:00
ret = i2c_transfer ( state - > i2c , & msg , 1 ) ;
2010-09-10 21:03:07 +04:00
if ( unlikely ( ret ! = 1 ) ) {
2008-10-17 03:23:45 +04:00
printk ( KERN_ERR
" %s: error (reg == 0x%02x, val == 0x%02x, ret == %i) \n " ,
2008-04-09 06:20:00 +04:00
__func__ , reg , data , ret ) ;
2010-09-10 21:03:07 +04:00
return - 1 ;
}
2005-04-17 02:20:36 +04:00
2010-09-10 21:03:07 +04:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
static u8 cx22702_readreg ( struct cx22702_state * state , u8 reg )
2005-04-17 02:20:36 +04:00
{
int ret ;
2010-09-10 21:03:07 +04:00
u8 data ;
2005-04-17 02:20:36 +04:00
2008-10-17 03:23:45 +04:00
struct i2c_msg msg [ ] = {
{ . addr = state - > config - > demod_address , . flags = 0 ,
2010-09-10 21:03:07 +04:00
. buf = & reg , . len = 1 } ,
2008-10-17 03:23:45 +04:00
{ . addr = state - > config - > demod_address , . flags = I2C_M_RD ,
2010-09-10 21:03:07 +04:00
. buf = & data , . len = 1 } } ;
2005-04-17 02:20:36 +04:00
ret = i2c_transfer ( state - > i2c , msg , 2 ) ;
2010-09-10 21:03:07 +04:00
if ( unlikely ( ret ! = 2 ) ) {
printk ( KERN_ERR " %s: error (reg == 0x%02x, ret == %i) \n " ,
__func__ , reg , ret ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2010-09-10 21:03:07 +04:00
return data ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
static int cx22702_set_inversion ( struct cx22702_state * state , int inversion )
2005-04-17 02:20:36 +04:00
{
u8 val ;
2010-09-10 17:33:42 +04:00
val = cx22702_readreg ( state , 0x0C ) ;
2005-04-17 02:20:36 +04:00
switch ( inversion ) {
2008-10-17 03:23:45 +04:00
case INVERSION_AUTO :
return - EOPNOTSUPP ;
case INVERSION_ON :
2010-09-10 17:33:42 +04:00
val | = 0x01 ;
break ;
2008-10-17 03:23:45 +04:00
case INVERSION_OFF :
2010-09-10 17:33:42 +04:00
val & = 0xfe ;
break ;
2008-10-17 03:23:45 +04:00
default :
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
2010-09-10 17:33:42 +04:00
return cx22702_writereg ( state , 0x0C , val ) ;
2005-04-17 02:20:36 +04:00
}
/* Retrieve the demod settings */
2008-10-17 03:23:45 +04:00
static int cx22702_get_tps ( struct cx22702_state * state ,
2011-12-22 23:40:49 +04:00
struct dtv_frontend_properties * p )
2005-04-17 02:20:36 +04:00
{
u8 val ;
/* Make sure the TPS regs are valid */
if ( ! ( cx22702_readreg ( state , 0x0A ) & 0x20 ) )
return - EAGAIN ;
2008-10-17 03:23:45 +04:00
val = cx22702_readreg ( state , 0x01 ) ;
switch ( ( val & 0x18 ) > > 3 ) {
case 0 :
2011-12-22 23:40:49 +04:00
p - > modulation = QPSK ;
2008-10-17 03:23:45 +04:00
break ;
case 1 :
2011-12-22 23:40:49 +04:00
p - > modulation = QAM_16 ;
2008-10-17 03:23:45 +04:00
break ;
case 2 :
2011-12-22 23:40:49 +04:00
p - > modulation = QAM_64 ;
2008-10-17 03:23:45 +04:00
break ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
switch ( val & 0x07 ) {
case 0 :
2011-12-22 23:40:49 +04:00
p - > hierarchy = HIERARCHY_NONE ;
2008-10-17 03:23:45 +04:00
break ;
case 1 :
2011-12-22 23:40:49 +04:00
p - > hierarchy = HIERARCHY_1 ;
2008-10-17 03:23:45 +04:00
break ;
case 2 :
2011-12-22 23:40:49 +04:00
p - > hierarchy = HIERARCHY_2 ;
2008-10-17 03:23:45 +04:00
break ;
case 3 :
2011-12-22 23:40:49 +04:00
p - > hierarchy = HIERARCHY_4 ;
2008-10-17 03:23:45 +04:00
break ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
val = cx22702_readreg ( state , 0x02 ) ;
switch ( ( val & 0x38 ) > > 3 ) {
case 0 :
p - > code_rate_HP = FEC_1_2 ;
break ;
case 1 :
p - > code_rate_HP = FEC_2_3 ;
break ;
case 2 :
p - > code_rate_HP = FEC_3_4 ;
break ;
case 3 :
p - > code_rate_HP = FEC_5_6 ;
break ;
case 4 :
p - > code_rate_HP = FEC_7_8 ;
break ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
switch ( val & 0x07 ) {
case 0 :
p - > code_rate_LP = FEC_1_2 ;
break ;
case 1 :
p - > code_rate_LP = FEC_2_3 ;
break ;
case 2 :
p - > code_rate_LP = FEC_3_4 ;
break ;
case 3 :
p - > code_rate_LP = FEC_5_6 ;
break ;
case 4 :
p - > code_rate_LP = FEC_7_8 ;
break ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
val = cx22702_readreg ( state , 0x03 ) ;
switch ( ( val & 0x0c ) > > 2 ) {
case 0 :
p - > guard_interval = GUARD_INTERVAL_1_32 ;
break ;
case 1 :
p - > guard_interval = GUARD_INTERVAL_1_16 ;
break ;
case 2 :
p - > guard_interval = GUARD_INTERVAL_1_8 ;
break ;
case 3 :
p - > guard_interval = GUARD_INTERVAL_1_4 ;
break ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
switch ( val & 0x03 ) {
case 0 :
p - > transmission_mode = TRANSMISSION_MODE_2K ;
break ;
case 1 :
p - > transmission_mode = TRANSMISSION_MODE_8K ;
break ;
2005-04-17 02:20:36 +04:00
}
return 0 ;
}
2008-10-17 03:23:45 +04:00
static int cx22702_i2c_gate_ctrl ( struct dvb_frontend * fe , int enable )
2006-01-09 20:25:12 +03:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2010-09-10 17:33:42 +04:00
u8 val ;
2008-10-17 03:23:45 +04:00
dprintk ( " %s(%d) \n " , __func__ , enable ) ;
2010-09-10 17:33:42 +04:00
val = cx22702_readreg ( state , 0x0D ) ;
2006-01-09 20:25:12 +03:00
if ( enable )
2010-09-10 17:33:42 +04:00
val & = 0xfe ;
2006-01-09 20:25:12 +03:00
else
2010-09-10 17:33:42 +04:00
val | = 0x01 ;
return cx22702_writereg ( state , 0x0D , val ) ;
2006-01-09 20:25:12 +03:00
}
2005-04-17 02:20:36 +04:00
/* Talk to the demod, set the FEC, GUARD, QAM settings etc */
2011-12-22 23:40:49 +04:00
static int cx22702_set_tps ( struct dvb_frontend * fe )
2005-04-17 02:20:36 +04:00
{
2011-12-22 23:40:49 +04:00
struct dtv_frontend_properties * p = & fe - > dtv_property_cache ;
2005-04-17 02:20:36 +04:00
u8 val ;
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2005-04-17 02:20:36 +04:00
2006-05-14 12:01:31 +04:00
if ( fe - > ops . tuner_ops . set_params ) {
2011-12-24 19:24:33 +04:00
fe - > ops . tuner_ops . set_params ( fe ) ;
2008-10-17 03:23:45 +04:00
if ( fe - > ops . i2c_gate_ctrl )
fe - > ops . i2c_gate_ctrl ( fe , 0 ) ;
2005-05-01 19:59:20 +04:00
}
2005-04-17 02:20:36 +04:00
/* set inversion */
2008-10-17 03:23:45 +04:00
cx22702_set_inversion ( state , p - > inversion ) ;
2005-04-17 02:20:36 +04:00
/* set bandwidth */
2010-09-10 17:33:42 +04:00
val = cx22702_readreg ( state , 0x0C ) & 0xcf ;
2011-12-22 23:40:49 +04:00
switch ( p - > bandwidth_hz ) {
case 6000000 :
2010-09-10 17:33:42 +04:00
val | = 0x20 ;
2005-04-17 02:20:36 +04:00
break ;
2011-12-22 23:40:49 +04:00
case 7000000 :
2010-09-10 17:33:42 +04:00
val | = 0x10 ;
2005-04-17 02:20:36 +04:00
break ;
2011-12-22 23:40:49 +04:00
case 8000000 :
2005-04-17 02:20:36 +04:00
break ;
default :
2008-10-17 03:23:45 +04:00
dprintk ( " %s: invalid bandwidth \n " , __func__ ) ;
2005-04-17 02:20:36 +04:00
return - EINVAL ;
}
2010-09-10 17:33:42 +04:00
cx22702_writereg ( state , 0x0C , val ) ;
2005-04-17 02:20:36 +04:00
2011-12-22 23:40:49 +04:00
p - > code_rate_LP = FEC_AUTO ; /* temp hack as manual not working */
2005-04-17 02:20:36 +04:00
/* use auto configuration? */
2011-12-22 23:40:49 +04:00
if ( ( p - > hierarchy = = HIERARCHY_AUTO ) | |
( p - > modulation = = QAM_AUTO ) | |
( p - > code_rate_HP = = FEC_AUTO ) | |
( p - > code_rate_LP = = FEC_AUTO ) | |
( p - > guard_interval = = GUARD_INTERVAL_AUTO ) | |
( p - > transmission_mode = = TRANSMISSION_MODE_AUTO ) ) {
2005-04-17 02:20:36 +04:00
/* TPS Source - use hardware driven values */
cx22702_writereg ( state , 0x06 , 0x10 ) ;
cx22702_writereg ( state , 0x07 , 0x9 ) ;
cx22702_writereg ( state , 0x08 , 0xC1 ) ;
2008-10-17 03:23:45 +04:00
cx22702_writereg ( state , 0x0B , cx22702_readreg ( state , 0x0B )
& 0xfc ) ;
cx22702_writereg ( state , 0x0C ,
( cx22702_readreg ( state , 0x0C ) & 0xBF ) | 0x40 ) ;
2011-03-31 05:57:33 +04:00
cx22702_writereg ( state , 0x00 , 0x01 ) ; /* Begin acquisition */
2008-10-17 03:23:45 +04:00
dprintk ( " %s: Autodetecting \n " , __func__ ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/* manually programmed values */
2011-12-22 23:40:49 +04:00
switch ( p - > modulation ) { /* mask 0x18 */
2008-10-17 03:23:45 +04:00
case QPSK :
2010-09-10 17:36:37 +04:00
val = 0x00 ;
2008-10-17 03:23:45 +04:00
break ;
case QAM_16 :
2010-09-10 17:36:37 +04:00
val = 0x08 ;
2008-10-17 03:23:45 +04:00
break ;
case QAM_64 :
2010-09-10 17:36:37 +04:00
val = 0x10 ;
2008-10-17 03:23:45 +04:00
break ;
default :
2011-12-22 23:40:49 +04:00
dprintk ( " %s: invalid modulation \n " , __func__ ) ;
2008-10-17 03:23:45 +04:00
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
2011-12-22 23:40:49 +04:00
switch ( p - > hierarchy ) { /* mask 0x07 */
2008-10-17 03:23:45 +04:00
case HIERARCHY_NONE :
break ;
case HIERARCHY_1 :
2010-09-10 17:36:37 +04:00
val | = 0x01 ;
2008-10-17 03:23:45 +04:00
break ;
case HIERARCHY_2 :
2010-09-10 17:36:37 +04:00
val | = 0x02 ;
2008-10-17 03:23:45 +04:00
break ;
case HIERARCHY_4 :
2010-09-10 17:36:37 +04:00
val | = 0x03 ;
2008-10-17 03:23:45 +04:00
break ;
default :
dprintk ( " %s: invalid hierarchy \n " , __func__ ) ;
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
cx22702_writereg ( state , 0x06 , val ) ;
2011-12-22 23:40:49 +04:00
switch ( p - > code_rate_HP ) { /* mask 0x38 */
2008-10-17 03:23:45 +04:00
case FEC_NONE :
case FEC_1_2 :
2010-09-10 17:36:37 +04:00
val = 0x00 ;
2008-10-17 03:23:45 +04:00
break ;
case FEC_2_3 :
2010-09-10 17:36:37 +04:00
val = 0x08 ;
2008-10-17 03:23:45 +04:00
break ;
case FEC_3_4 :
2010-09-10 17:36:37 +04:00
val = 0x10 ;
2008-10-17 03:23:45 +04:00
break ;
case FEC_5_6 :
2010-09-10 17:36:37 +04:00
val = 0x18 ;
2008-10-17 03:23:45 +04:00
break ;
case FEC_7_8 :
2010-09-10 17:36:37 +04:00
val = 0x20 ;
2008-10-17 03:23:45 +04:00
break ;
default :
dprintk ( " %s: invalid code_rate_HP \n " , __func__ ) ;
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
2011-12-22 23:40:49 +04:00
switch ( p - > code_rate_LP ) { /* mask 0x07 */
2008-10-17 03:23:45 +04:00
case FEC_NONE :
case FEC_1_2 :
break ;
case FEC_2_3 :
2010-09-10 17:36:37 +04:00
val | = 0x01 ;
2008-10-17 03:23:45 +04:00
break ;
case FEC_3_4 :
2010-09-10 17:36:37 +04:00
val | = 0x02 ;
2008-10-17 03:23:45 +04:00
break ;
case FEC_5_6 :
2010-09-10 17:36:37 +04:00
val | = 0x03 ;
2008-10-17 03:23:45 +04:00
break ;
case FEC_7_8 :
2010-09-10 17:36:37 +04:00
val | = 0x04 ;
2008-10-17 03:23:45 +04:00
break ;
default :
dprintk ( " %s: invalid code_rate_LP \n " , __func__ ) ;
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
cx22702_writereg ( state , 0x07 , val ) ;
2011-12-22 23:40:49 +04:00
switch ( p - > guard_interval ) { /* mask 0x0c */
2008-10-17 03:23:45 +04:00
case GUARD_INTERVAL_1_32 :
2010-09-10 17:36:37 +04:00
val = 0x00 ;
2008-10-17 03:23:45 +04:00
break ;
case GUARD_INTERVAL_1_16 :
2010-09-10 17:36:37 +04:00
val = 0x04 ;
2008-10-17 03:23:45 +04:00
break ;
case GUARD_INTERVAL_1_8 :
2010-09-10 17:36:37 +04:00
val = 0x08 ;
2008-10-17 03:23:45 +04:00
break ;
case GUARD_INTERVAL_1_4 :
2010-09-10 17:36:37 +04:00
val = 0x0c ;
2008-10-17 03:23:45 +04:00
break ;
default :
dprintk ( " %s: invalid guard_interval \n " , __func__ ) ;
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
2011-12-22 23:40:49 +04:00
switch ( p - > transmission_mode ) { /* mask 0x03 */
2008-10-17 03:23:45 +04:00
case TRANSMISSION_MODE_2K :
break ;
case TRANSMISSION_MODE_8K :
2010-09-10 17:36:37 +04:00
val | = 0x1 ;
2008-10-17 03:23:45 +04:00
break ;
default :
dprintk ( " %s: invalid transmission_mode \n " , __func__ ) ;
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
cx22702_writereg ( state , 0x08 , val ) ;
2008-10-17 03:23:45 +04:00
cx22702_writereg ( state , 0x0B ,
( cx22702_readreg ( state , 0x0B ) & 0xfc ) | 0x02 ) ;
cx22702_writereg ( state , 0x0C ,
( cx22702_readreg ( state , 0x0C ) & 0xBF ) | 0x40 ) ;
2005-04-17 02:20:36 +04:00
2011-03-31 05:57:33 +04:00
/* Begin channel acquisition */
2005-04-17 02:20:36 +04:00
cx22702_writereg ( state , 0x00 , 0x01 ) ;
return 0 ;
}
/* Reset the demod hardware and reset all of the configuration registers
to a default state . */
2008-10-17 03:23:45 +04:00
static int cx22702_init ( struct dvb_frontend * fe )
2005-04-17 02:20:36 +04:00
{
int i ;
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2005-04-17 02:20:36 +04:00
2008-10-17 03:23:45 +04:00
cx22702_writereg ( state , 0x00 , 0x02 ) ;
2005-04-17 02:20:36 +04:00
msleep ( 10 ) ;
2008-10-17 03:23:45 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( init_tab ) ; i + = 2 )
cx22702_writereg ( state , init_tab [ i ] , init_tab [ i + 1 ] ) ;
2005-04-17 02:20:36 +04:00
2008-10-17 03:23:45 +04:00
cx22702_writereg ( state , 0xf8 , ( state - > config - > output_mode < < 1 )
& 0x02 ) ;
2005-04-17 02:20:36 +04:00
2006-01-09 20:25:12 +03:00
cx22702_i2c_gate_ctrl ( fe , 0 ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2015-06-07 20:53:52 +03:00
static int cx22702_read_status ( struct dvb_frontend * fe , enum fe_status * status )
2005-04-17 02:20:36 +04:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2005-04-17 02:20:36 +04:00
u8 reg0A ;
u8 reg23 ;
* status = 0 ;
2008-10-17 03:23:45 +04:00
reg0A = cx22702_readreg ( state , 0x0A ) ;
reg23 = cx22702_readreg ( state , 0x23 ) ;
2005-04-17 02:20:36 +04:00
2008-10-17 03:23:45 +04:00
dprintk ( " %s: status demod=0x%02x agc=0x%02x \n "
, __func__ , reg0A , reg23 ) ;
2005-04-17 02:20:36 +04:00
2008-10-17 03:23:45 +04:00
if ( reg0A & 0x10 ) {
2005-04-17 02:20:36 +04:00
* status | = FE_HAS_LOCK ;
* status | = FE_HAS_VITERBI ;
* status | = FE_HAS_SYNC ;
}
2008-10-17 03:23:45 +04:00
if ( reg0A & 0x20 )
2005-04-17 02:20:36 +04:00
* status | = FE_HAS_CARRIER ;
2008-10-17 03:23:45 +04:00
if ( reg23 < 0xf0 )
2005-04-17 02:20:36 +04:00
* status | = FE_HAS_SIGNAL ;
return 0 ;
}
2008-10-17 03:23:45 +04:00
static int cx22702_read_ber ( struct dvb_frontend * fe , u32 * ber )
2005-04-17 02:20:36 +04:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2005-04-17 02:20:36 +04:00
2008-10-17 03:23:45 +04:00
if ( cx22702_readreg ( state , 0xE4 ) & 0x02 ) {
2005-04-17 02:20:36 +04:00
/* Realtime statistics */
2008-10-17 03:23:45 +04:00
* ber = ( cx22702_readreg ( state , 0xDE ) & 0x7F ) < < 7
| ( cx22702_readreg ( state , 0xDF ) & 0x7F ) ;
2005-04-17 02:20:36 +04:00
} else {
/* Averagtine statistics */
2008-10-17 03:23:45 +04:00
* ber = ( cx22702_readreg ( state , 0xDE ) & 0x7F ) < < 7
| cx22702_readreg ( state , 0xDF ) ;
2005-04-17 02:20:36 +04:00
}
return 0 ;
}
2008-10-17 03:23:45 +04:00
static int cx22702_read_signal_strength ( struct dvb_frontend * fe ,
u16 * signal_strength )
2005-04-17 02:20:36 +04:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2012-02-12 22:03:03 +04:00
u8 reg23 ;
2005-04-17 02:20:36 +04:00
2012-02-12 22:03:03 +04:00
/*
* Experience suggests that the strength signal register works as
* follows :
* - In the absence of signal , value is 0xff .
* - In the presence of a weak signal , bit 7 is set , not sure what
* the lower 7 bits mean .
* - In the presence of a strong signal , the register holds a 7 - bit
* value ( bit 7 is cleared ) , with greater values standing for
* weaker signals .
*/
reg23 = cx22702_readreg ( state , 0x23 ) ;
if ( reg23 & 0x80 ) {
* signal_strength = 0 ;
} else {
reg23 = ~ reg23 & 0x7f ;
/* Scale to 16 bit */
* signal_strength = ( reg23 < < 9 ) | ( reg23 < < 2 ) | ( reg23 > > 5 ) ;
}
2005-04-17 02:20:36 +04:00
return 0 ;
}
2008-10-17 03:23:45 +04:00
static int cx22702_read_snr ( struct dvb_frontend * fe , u16 * snr )
2005-04-17 02:20:36 +04:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2005-04-17 02:20:36 +04:00
2010-09-10 17:32:21 +04:00
u16 rs_ber ;
2008-10-17 03:23:45 +04:00
if ( cx22702_readreg ( state , 0xE4 ) & 0x02 ) {
2005-04-17 02:20:36 +04:00
/* Realtime statistics */
2008-10-17 03:23:45 +04:00
rs_ber = ( cx22702_readreg ( state , 0xDE ) & 0x7F ) < < 7
| ( cx22702_readreg ( state , 0xDF ) & 0x7F ) ;
2005-04-17 02:20:36 +04:00
} else {
/* Averagine statistics */
2008-10-17 03:23:45 +04:00
rs_ber = ( cx22702_readreg ( state , 0xDE ) & 0x7F ) < < 8
| cx22702_readreg ( state , 0xDF ) ;
2005-04-17 02:20:36 +04:00
}
* snr = ~ rs_ber ;
return 0 ;
}
2008-10-17 03:23:45 +04:00
static int cx22702_read_ucblocks ( struct dvb_frontend * fe , u32 * ucblocks )
2005-04-17 02:20:36 +04:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2005-04-17 02:20:36 +04:00
u8 _ucblocks ;
/* RS Uncorrectable Packet Count then reset */
2008-10-17 03:23:45 +04:00
_ucblocks = cx22702_readreg ( state , 0xE3 ) ;
2005-07-08 04:57:44 +04:00
if ( state - > prevUCBlocks < _ucblocks )
* ucblocks = ( _ucblocks - state - > prevUCBlocks ) ;
else
* ucblocks = state - > prevUCBlocks - _ucblocks ;
2005-04-17 02:20:36 +04:00
state - > prevUCBlocks = _ucblocks ;
return 0 ;
}
2016-02-04 17:58:30 +03:00
static int cx22702_get_frontend ( struct dvb_frontend * fe ,
struct dtv_frontend_properties * c )
2005-04-17 02:20:36 +04:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2005-04-17 02:20:36 +04:00
2008-10-17 03:23:45 +04:00
u8 reg0C = cx22702_readreg ( state , 0x0C ) ;
2005-04-17 02:20:36 +04:00
2011-12-22 23:40:49 +04:00
c - > inversion = reg0C & 0x1 ? INVERSION_ON : INVERSION_OFF ;
return cx22702_get_tps ( state , c ) ;
2005-04-17 02:20:36 +04:00
}
2008-10-17 03:23:45 +04:00
static int cx22702_get_tune_settings ( struct dvb_frontend * fe ,
struct dvb_frontend_tune_settings * tune )
2005-07-08 04:57:44 +04:00
{
tune - > min_delay_ms = 1000 ;
return 0 ;
}
2008-10-17 03:23:45 +04:00
static void cx22702_release ( struct dvb_frontend * fe )
2005-04-17 02:20:36 +04:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = fe - > demodulator_priv ;
2005-04-17 02:20:36 +04:00
kfree ( state ) ;
}
2010-09-10 17:35:12 +04:00
static const struct dvb_frontend_ops cx22702_ops ;
2005-04-17 02:20:36 +04:00
2008-10-17 03:23:45 +04:00
struct dvb_frontend * cx22702_attach ( const struct cx22702_config * config ,
struct i2c_adapter * i2c )
2005-04-17 02:20:36 +04:00
{
2008-10-17 03:23:45 +04:00
struct cx22702_state * state = NULL ;
2005-04-17 02:20:36 +04:00
/* allocate memory for the internal state */
2009-08-11 05:51:01 +04:00
state = kzalloc ( sizeof ( struct cx22702_state ) , GFP_KERNEL ) ;
2005-07-08 04:57:44 +04:00
if ( state = = NULL )
goto error ;
2005-04-17 02:20:36 +04:00
/* setup the state */
state - > config = config ;
state - > i2c = i2c ;
/* check if the demod is there */
2005-07-08 04:57:44 +04:00
if ( cx22702_readreg ( state , 0x1f ) ! = 0x3 )
goto error ;
2005-04-17 02:20:36 +04:00
/* create dvb_frontend */
2008-10-17 03:23:45 +04:00
memcpy ( & state - > frontend . ops , & cx22702_ops ,
sizeof ( struct dvb_frontend_ops ) ) ;
2005-04-17 02:20:36 +04:00
state - > frontend . demodulator_priv = state ;
return & state - > frontend ;
error :
kfree ( state ) ;
return NULL ;
}
2008-10-17 03:23:45 +04:00
EXPORT_SYMBOL ( cx22702_attach ) ;
2005-04-17 02:20:36 +04:00
2010-09-10 17:35:12 +04:00
static const struct dvb_frontend_ops cx22702_ops = {
2011-12-22 23:40:49 +04:00
. delsys = { SYS_DVBT } ,
2005-04-17 02:20:36 +04:00
. info = {
. name = " Conexant CX22702 DVB-T " ,
2018-07-06 01:59:36 +03:00
. frequency_min_hz = 177 * MHz ,
. frequency_max_hz = 858 * MHz ,
. frequency_stepsize_hz = 166666 ,
2005-04-17 02:20:36 +04:00
. caps = FE_CAN_FEC_1_2 | FE_CAN_FEC_2_3 | FE_CAN_FEC_3_4 |
FE_CAN_FEC_5_6 | FE_CAN_FEC_7_8 | FE_CAN_FEC_AUTO |
FE_CAN_QPSK | FE_CAN_QAM_16 | FE_CAN_QAM_64 | FE_CAN_QAM_AUTO |
FE_CAN_HIERARCHY_AUTO | FE_CAN_GUARD_INTERVAL_AUTO |
FE_CAN_TRANSMISSION_MODE_AUTO | FE_CAN_RECOVER
} ,
. release = cx22702_release ,
. init = cx22702_init ,
2006-04-19 00:47:09 +04:00
. i2c_gate_ctrl = cx22702_i2c_gate_ctrl ,
2005-04-17 02:20:36 +04:00
2011-12-22 23:40:49 +04:00
. set_frontend = cx22702_set_tps ,
. get_frontend = cx22702_get_frontend ,
2005-07-08 04:57:44 +04:00
. get_tune_settings = cx22702_get_tune_settings ,
2005-04-17 02:20:36 +04:00
. read_status = cx22702_read_status ,
. read_ber = cx22702_read_ber ,
. read_signal_strength = cx22702_read_signal_strength ,
. read_snr = cx22702_read_snr ,
. read_ucblocks = cx22702_read_ucblocks ,
} ;
MODULE_DESCRIPTION ( " Conexant CX22702 DVB-T Demodulator driver " ) ;
MODULE_AUTHOR ( " Steven Toth " ) ;
MODULE_LICENSE ( " GPL " ) ;