2019-05-31 11:09:32 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2016-11-14 16:14:37 +03:00
/*
* Frontend driver for the GENPIX 8 pks / qpsk / DCII USB2 .0 DVB - S module
2006-05-14 20:23:56 +04:00
*
2007-08-19 00:52:35 +04:00
* Copyright ( C ) 2006 , 2007 Alan Nisota ( alannisota @ gmail . com )
* Copyright ( C ) 2006 , 2007 Genpix Electronics ( genpix @ genpix - electronics . com )
2006-05-14 20:23:56 +04:00
*
* Thanks to GENPIX for the sample code used to implement this module .
*
* This module is based off the vp7045 and vp702x modules
*/
2016-11-12 17:46:28 +03:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
# include "gp8psk-fe.h"
2017-12-28 21:03:51 +03:00
# include <media/dvb_frontend.h>
2016-11-12 17:46:28 +03:00
static int debug ;
module_param ( debug , int , 0644 ) ;
MODULE_PARM_DESC ( debug , " Turn on/off debugging (default:off). " ) ;
# define dprintk(fmt, arg...) do { \
if ( debug ) \
printk ( KERN_DEBUG pr_fmt ( " %s: " fmt ) , \
__func__ , # # arg ) ; \
} while ( 0 )
2006-05-14 20:23:56 +04:00
struct gp8psk_fe_state {
struct dvb_frontend fe ;
2016-11-12 17:46:28 +03:00
void * priv ;
const struct gp8psk_fe_ops * ops ;
bool is_rev1 ;
2007-08-19 00:52:35 +04:00
u8 lock ;
2006-05-14 20:23:56 +04:00
u16 snr ;
2007-08-19 00:52:35 +04:00
unsigned long next_status_check ;
unsigned long status_check_interval ;
2006-05-14 20:23:56 +04:00
} ;
2008-12-20 18:03:06 +03:00
static int gp8psk_tuned_to_DCII ( struct dvb_frontend * fe )
{
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
u8 status ;
2016-11-12 17:46:28 +03:00
st - > ops - > in ( st - > priv , GET_8PSK_CONFIG , 0 , 0 , & status , 1 ) ;
2008-12-20 18:03:06 +03:00
return status & bmDCtuned ;
}
static int gp8psk_set_tuner_mode ( struct dvb_frontend * fe , int mode )
{
2016-11-12 17:46:28 +03:00
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
return st - > ops - > out ( st - > priv , SET_8PSK_CONFIG , mode , 0 , NULL , 0 ) ;
2008-12-20 18:03:06 +03:00
}
2007-08-19 00:52:35 +04:00
static int gp8psk_fe_update_status ( struct gp8psk_fe_state * st )
{
u8 buf [ 6 ] ;
if ( time_after ( jiffies , st - > next_status_check ) ) {
2016-11-12 17:46:28 +03:00
st - > ops - > in ( st - > priv , GET_SIGNAL_LOCK , 0 , 0 , & st - > lock , 1 ) ;
st - > ops - > in ( st - > priv , GET_SIGNAL_STRENGTH , 0 , 0 , buf , 6 ) ;
2007-08-19 00:52:35 +04:00
st - > snr = ( buf [ 1 ] ) < < 8 | buf [ 0 ] ;
st - > next_status_check = jiffies + ( st - > status_check_interval * HZ ) / 1000 ;
}
return 0 ;
}
2015-06-07 20:53:52 +03:00
static int gp8psk_fe_read_status ( struct dvb_frontend * fe ,
enum fe_status * status )
2006-05-14 20:23:56 +04:00
{
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
2007-08-19 00:52:35 +04:00
gp8psk_fe_update_status ( st ) ;
2006-05-14 20:23:56 +04:00
2007-08-19 00:52:35 +04:00
if ( st - > lock )
2006-05-14 20:23:56 +04:00
* status = FE_HAS_LOCK | FE_HAS_SYNC | FE_HAS_VITERBI | FE_HAS_SIGNAL | FE_HAS_CARRIER ;
else
* status = 0 ;
2007-08-19 00:52:35 +04:00
if ( * status & FE_HAS_LOCK )
st - > status_check_interval = 1000 ;
else
st - > status_check_interval = 100 ;
2006-05-14 20:23:56 +04:00
return 0 ;
}
/* not supported by this Frontend */
static int gp8psk_fe_read_ber ( struct dvb_frontend * fe , u32 * ber )
{
( void ) fe ;
* ber = 0 ;
return 0 ;
}
/* not supported by this Frontend */
static int gp8psk_fe_read_unc_blocks ( struct dvb_frontend * fe , u32 * unc )
{
( void ) fe ;
* unc = 0 ;
return 0 ;
}
static int gp8psk_fe_read_snr ( struct dvb_frontend * fe , u16 * snr )
{
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
2007-08-19 00:52:35 +04:00
gp8psk_fe_update_status ( st ) ;
/* snr is reported in dBu*256 */
* snr = st - > snr ;
2006-05-14 20:23:56 +04:00
return 0 ;
}
static int gp8psk_fe_read_signal_strength ( struct dvb_frontend * fe , u16 * strength )
{
2007-08-19 00:52:35 +04:00
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
gp8psk_fe_update_status ( st ) ;
/* snr is reported in dBu*256 */
/* snr / 38.4 ~= 100% strength */
/* snr * 17 returns 100% strength as 65535 */
if ( st - > snr > 0xf00 )
* strength = 0xffff ;
else
* strength = ( st - > snr < < 4 ) + st - > snr ; /* snr*17 */
return 0 ;
2006-05-14 20:23:56 +04:00
}
static int gp8psk_fe_get_tune_settings ( struct dvb_frontend * fe , struct dvb_frontend_tune_settings * tune )
{
2010-10-16 23:07:47 +04:00
tune - > min_delay_ms = 800 ;
2006-05-14 20:23:56 +04:00
return 0 ;
}
2011-12-26 22:49:16 +04:00
static int gp8psk_fe_set_frontend ( struct dvb_frontend * fe )
2006-05-14 20:23:56 +04:00
{
2016-11-12 17:46:28 +03:00
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
2008-12-20 18:03:06 +03:00
struct dtv_frontend_properties * c = & fe - > dtv_property_cache ;
2006-05-14 20:23:56 +04:00
u8 cmd [ 10 ] ;
2011-12-26 22:49:16 +04:00
u32 freq = c - > frequency * 1000 ;
2008-12-20 18:03:06 +03:00
2016-11-12 17:46:28 +03:00
dprintk ( " %s() \n " , __func__ ) ;
2006-05-14 20:23:56 +04:00
cmd [ 4 ] = freq & 0xff ;
cmd [ 5 ] = ( freq > > 8 ) & 0xff ;
cmd [ 6 ] = ( freq > > 16 ) & 0xff ;
cmd [ 7 ] = ( freq > > 24 ) & 0xff ;
2011-08-08 18:54:37 +04:00
/* backwards compatibility: DVB-S + 8-PSK were used for Turbo-FEC */
if ( c - > delivery_system = = SYS_DVBS & & c - > modulation = = PSK_8 )
c - > delivery_system = SYS_TURBO ;
2008-12-20 18:03:06 +03:00
switch ( c - > delivery_system ) {
case SYS_DVBS :
2011-08-08 18:54:37 +04:00
if ( c - > modulation ! = QPSK ) {
2016-11-12 17:46:28 +03:00
dprintk ( " %s: unsupported modulation selected (%d) \n " ,
2008-12-20 18:03:06 +03:00
__func__ , c - > modulation ) ;
return - EOPNOTSUPP ;
}
c - > fec_inner = FEC_AUTO ;
2006-05-14 20:23:56 +04:00
break ;
2011-08-08 18:54:37 +04:00
case SYS_DVBS2 : /* kept for backwards compatibility */
2016-11-12 17:46:28 +03:00
dprintk ( " %s: DVB-S2 delivery system selected \n " , __func__ ) ;
2008-12-20 18:03:06 +03:00
break ;
2011-08-08 18:54:37 +04:00
case SYS_TURBO :
2016-11-12 17:46:28 +03:00
dprintk ( " %s: Turbo-FEC delivery system selected \n " , __func__ ) ;
2011-08-08 18:54:37 +04:00
break ;
2008-12-20 18:03:06 +03:00
2006-05-14 20:23:56 +04:00
default :
2016-11-12 17:46:28 +03:00
dprintk ( " %s: unsupported delivery system selected (%d) \n " ,
2008-12-20 18:03:06 +03:00
__func__ , c - > delivery_system ) ;
return - EOPNOTSUPP ;
}
cmd [ 0 ] = c - > symbol_rate & 0xff ;
cmd [ 1 ] = ( c - > symbol_rate > > 8 ) & 0xff ;
cmd [ 2 ] = ( c - > symbol_rate > > 16 ) & 0xff ;
cmd [ 3 ] = ( c - > symbol_rate > > 24 ) & 0xff ;
switch ( c - > modulation ) {
case QPSK :
2016-11-12 17:46:28 +03:00
if ( st - > is_rev1 )
2008-12-20 18:03:06 +03:00
if ( gp8psk_tuned_to_DCII ( fe ) )
2016-11-12 17:46:28 +03:00
st - > ops - > reload ( st - > priv ) ;
2008-12-20 18:03:06 +03:00
switch ( c - > fec_inner ) {
case FEC_1_2 :
cmd [ 9 ] = 0 ; break ;
case FEC_2_3 :
cmd [ 9 ] = 1 ; break ;
case FEC_3_4 :
cmd [ 9 ] = 2 ; break ;
case FEC_5_6 :
cmd [ 9 ] = 3 ; break ;
case FEC_7_8 :
cmd [ 9 ] = 4 ; break ;
case FEC_AUTO :
cmd [ 9 ] = 5 ; break ;
default :
cmd [ 9 ] = 5 ; break ;
}
2011-08-08 18:54:37 +04:00
if ( c - > delivery_system = = SYS_TURBO )
cmd [ 8 ] = ADV_MOD_TURBO_QPSK ;
else
cmd [ 8 ] = ADV_MOD_DVB_QPSK ;
2008-12-20 18:03:06 +03:00
break ;
case PSK_8 : /* PSK_8 is for compatibility with DN */
cmd [ 8 ] = ADV_MOD_TURBO_8PSK ;
switch ( c - > fec_inner ) {
case FEC_2_3 :
cmd [ 9 ] = 0 ; break ;
case FEC_3_4 :
cmd [ 9 ] = 1 ; break ;
case FEC_3_5 :
cmd [ 9 ] = 2 ; break ;
case FEC_5_6 :
cmd [ 9 ] = 3 ; break ;
case FEC_8_9 :
cmd [ 9 ] = 4 ; break ;
default :
cmd [ 9 ] = 0 ; break ;
}
break ;
case QAM_16 : /* QAM_16 is for compatibility with DN */
cmd [ 8 ] = ADV_MOD_TURBO_16QAM ;
2006-05-14 20:23:56 +04:00
cmd [ 9 ] = 0 ;
break ;
2008-12-20 18:03:06 +03:00
default : /* Unknown modulation */
2016-11-12 17:46:28 +03:00
dprintk ( " %s: unsupported modulation selected (%d) \n " ,
2008-12-20 18:03:06 +03:00
__func__ , c - > modulation ) ;
return - EOPNOTSUPP ;
2006-05-14 20:23:56 +04:00
}
2016-11-12 17:46:28 +03:00
if ( st - > is_rev1 )
2008-12-20 18:03:06 +03:00
gp8psk_set_tuner_mode ( fe , 0 ) ;
2016-11-12 17:46:28 +03:00
st - > ops - > out ( st - > priv , TUNE_8PSK , 0 , 0 , cmd , 10 ) ;
2006-05-14 20:23:56 +04:00
2016-11-12 17:46:28 +03:00
st - > lock = 0 ;
st - > next_status_check = jiffies ;
st - > status_check_interval = 200 ;
2006-05-14 20:23:56 +04:00
return 0 ;
}
static int gp8psk_fe_send_diseqc_msg ( struct dvb_frontend * fe ,
struct dvb_diseqc_master_cmd * m )
{
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
2016-11-12 17:46:28 +03:00
dprintk ( " %s \n " , __func__ ) ;
2006-05-14 20:23:56 +04:00
2016-11-12 17:46:28 +03:00
if ( st - > ops - > out ( st - > priv , SEND_DISEQC_COMMAND , m - > msg [ 0 ] , 0 ,
2006-05-14 20:23:56 +04:00
m - > msg , m - > msg_len ) ) {
return - EINVAL ;
}
return 0 ;
}
2015-06-07 20:53:52 +03:00
static int gp8psk_fe_send_diseqc_burst ( struct dvb_frontend * fe ,
enum fe_sec_mini_cmd burst )
2006-05-14 20:23:56 +04:00
{
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
u8 cmd ;
2016-11-12 17:46:28 +03:00
dprintk ( " %s \n " , __func__ ) ;
2006-05-14 20:23:56 +04:00
/* These commands are certainly wrong */
cmd = ( burst = = SEC_MINI_A ) ? 0x00 : 0x01 ;
2016-11-12 17:46:28 +03:00
if ( st - > ops - > out ( st - > priv , SEND_DISEQC_COMMAND , cmd , 0 ,
2006-05-14 20:23:56 +04:00
& cmd , 0 ) ) {
return - EINVAL ;
}
return 0 ;
}
2015-06-07 20:53:52 +03:00
static int gp8psk_fe_set_tone ( struct dvb_frontend * fe ,
enum fe_sec_tone_mode tone )
2006-05-14 20:23:56 +04:00
{
2016-11-12 17:46:28 +03:00
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
2006-05-14 20:23:56 +04:00
2016-11-12 17:46:28 +03:00
if ( st - > ops - > out ( st - > priv , SET_22KHZ_TONE ,
( tone = = SEC_TONE_ON ) , 0 , NULL , 0 ) ) {
2006-05-14 20:23:56 +04:00
return - EINVAL ;
}
return 0 ;
}
2015-06-07 20:53:52 +03:00
static int gp8psk_fe_set_voltage ( struct dvb_frontend * fe ,
enum fe_sec_voltage voltage )
2006-05-14 20:23:56 +04:00
{
2016-11-12 17:46:28 +03:00
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
2006-05-14 20:23:56 +04:00
2016-11-12 17:46:28 +03:00
if ( st - > ops - > out ( st - > priv , SET_LNB_VOLTAGE ,
2006-05-14 20:23:56 +04:00
voltage = = SEC_VOLTAGE_18 , 0 , NULL , 0 ) ) {
return - EINVAL ;
}
return 0 ;
}
2007-08-19 00:52:35 +04:00
static int gp8psk_fe_enable_high_lnb_voltage ( struct dvb_frontend * fe , long onoff )
{
2016-11-12 17:46:28 +03:00
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
return st - > ops - > out ( st - > priv , USE_EXTRA_VOLT , onoff , 0 , NULL , 0 ) ;
2007-08-19 00:52:35 +04:00
}
2006-05-14 20:23:56 +04:00
static int gp8psk_fe_send_legacy_dish_cmd ( struct dvb_frontend * fe , unsigned long sw_cmd )
{
2016-11-12 17:46:28 +03:00
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
2006-05-14 20:23:56 +04:00
u8 cmd = sw_cmd & 0x7f ;
2016-11-12 17:46:28 +03:00
if ( st - > ops - > out ( st - > priv , SET_DN_SWITCH , cmd , 0 , NULL , 0 ) )
2006-05-14 20:23:56 +04:00
return - EINVAL ;
2016-11-12 17:46:28 +03:00
if ( st - > ops - > out ( st - > priv , SET_LNB_VOLTAGE , ! ! ( sw_cmd & 0x80 ) ,
0 , NULL , 0 ) )
2006-05-14 20:23:56 +04:00
return - EINVAL ;
return 0 ;
}
static void gp8psk_fe_release ( struct dvb_frontend * fe )
{
2016-11-12 17:46:28 +03:00
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
kfree ( st ) ;
2006-05-14 20:23:56 +04:00
}
2016-08-10 00:32:21 +03:00
static const struct dvb_frontend_ops gp8psk_fe_ops ;
2006-05-14 20:23:56 +04:00
2016-11-12 17:46:28 +03:00
struct dvb_frontend * gp8psk_fe_attach ( const struct gp8psk_fe_ops * ops ,
void * priv , bool is_rev1 )
2006-05-14 20:23:56 +04:00
{
2016-11-12 17:46:28 +03:00
struct gp8psk_fe_state * st ;
2006-05-14 20:23:56 +04:00
2016-11-12 17:46:28 +03:00
if ( ! ops | | ! ops - > in | | ! ops - > out | | ! ops - > reload ) {
pr_err ( " Error! gp8psk-fe ops not defined. \n " ) ;
return NULL ;
}
st = kzalloc ( sizeof ( struct gp8psk_fe_state ) , GFP_KERNEL ) ;
if ( ! st )
return NULL ;
memcpy ( & st - > fe . ops , & gp8psk_fe_ops , sizeof ( struct dvb_frontend_ops ) ) ;
st - > fe . demodulator_priv = st ;
st - > ops = ops ;
st - > priv = priv ;
st - > is_rev1 = is_rev1 ;
pr_info ( " Frontend %sattached \n " , is_rev1 ? " revision 1 " : " " ) ;
return & st - > fe ;
}
EXPORT_SYMBOL_GPL ( gp8psk_fe_attach ) ;
2006-05-14 20:23:56 +04:00
2016-08-10 00:32:21 +03:00
static const struct dvb_frontend_ops gp8psk_fe_ops = {
2011-12-26 22:49:16 +04:00
. delsys = { SYS_DVBS } ,
2006-05-14 20:23:56 +04:00
. info = {
2010-10-16 23:18:15 +04:00
. name = " Genpix DVB-S " ,
2018-07-06 01:59:36 +03:00
. frequency_min_hz = 800 * MHz ,
. frequency_max_hz = 2250 * MHz ,
. frequency_stepsize_hz = 100 * kHz ,
2006-05-14 20:23:56 +04:00
. symbol_rate_min = 1000000 ,
. symbol_rate_max = 45000000 ,
. symbol_rate_tolerance = 500 , /* ppm */
. caps = FE_CAN_INVERSION_AUTO |
2008-12-20 18:03:06 +03:00
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_QAM_16 is for compatibility
* ( Myth incorrectly detects Turbo - QPSK as plain QAM - 16 )
*/
2010-07-01 08:37:34 +04:00
FE_CAN_QPSK | FE_CAN_QAM_16 | FE_CAN_TURBO_FEC
2006-05-14 20:23:56 +04:00
} ,
. release = gp8psk_fe_release ,
. init = NULL ,
. sleep = NULL ,
2011-12-26 22:49:16 +04:00
. set_frontend = gp8psk_fe_set_frontend ,
2008-12-20 18:03:06 +03:00
2006-05-14 20:23:56 +04:00
. get_tune_settings = gp8psk_fe_get_tune_settings ,
. read_status = gp8psk_fe_read_status ,
. read_ber = gp8psk_fe_read_ber ,
. read_signal_strength = gp8psk_fe_read_signal_strength ,
. read_snr = gp8psk_fe_read_snr ,
. read_ucblocks = gp8psk_fe_read_unc_blocks ,
. diseqc_send_master_cmd = gp8psk_fe_send_diseqc_msg ,
. diseqc_send_burst = gp8psk_fe_send_diseqc_burst ,
. set_tone = gp8psk_fe_set_tone ,
. set_voltage = gp8psk_fe_set_voltage ,
. dishnetwork_send_legacy_command = gp8psk_fe_send_legacy_dish_cmd ,
2007-08-19 00:52:35 +04:00
. enable_high_lnb_voltage = gp8psk_fe_enable_high_lnb_voltage
2006-05-14 20:23:56 +04:00
} ;
2016-11-14 16:14:37 +03:00
MODULE_AUTHOR ( " Alan Nisota <alannisota@gamil.com> " ) ;
MODULE_DESCRIPTION ( " Frontend Driver for Genpix DVB-S " ) ;
MODULE_VERSION ( " 1.1 " ) ;
MODULE_LICENSE ( " GPL " ) ;