2006-05-14 13:23:56 -03:00
/* DVB USB compliant Linux driver for the
2007-08-18 17:52:35 -03:00
* - GENPIX 8 pks / qpsk / DCII USB2 .0 DVB - S module
2006-05-14 13:23:56 -03:00
*
2007-08-18 17:52:35 -03:00
* Copyright ( C ) 2006 , 2007 Alan Nisota ( alannisota @ gmail . com )
* Copyright ( C ) 2006 , 2007 Genpix Electronics ( genpix @ genpix - electronics . com )
2006-05-14 13:23:56 -03:00
*
* Thanks to GENPIX for the sample code used to implement this module .
*
* This module is based off the vp7045 and vp702x modules
*
* 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 , version 2.
*
* see Documentation / dvb / README . dvb - usb for more information
*/
# include "gp8psk.h"
struct gp8psk_fe_state {
struct dvb_frontend fe ;
struct dvb_usb_device * d ;
2007-08-18 17:52:35 -03:00
u8 lock ;
2006-05-14 13:23:56 -03:00
u16 snr ;
2007-08-18 17:52:35 -03:00
unsigned long next_status_check ;
unsigned long status_check_interval ;
2006-05-14 13:23:56 -03:00
} ;
2008-12-20 12:03:06 -03:00
static int gp8psk_tuned_to_DCII ( struct dvb_frontend * fe )
{
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
u8 status ;
gp8psk_usb_in_op ( st - > d , GET_8PSK_CONFIG , 0 , 0 , & status , 1 ) ;
return status & bmDCtuned ;
}
static int gp8psk_set_tuner_mode ( struct dvb_frontend * fe , int mode )
{
struct gp8psk_fe_state * state = fe - > demodulator_priv ;
return gp8psk_usb_out_op ( state - > d , SET_8PSK_CONFIG , mode , 0 , NULL , 0 ) ;
}
2007-08-18 17:52:35 -03:00
static int gp8psk_fe_update_status ( struct gp8psk_fe_state * st )
{
u8 buf [ 6 ] ;
if ( time_after ( jiffies , st - > next_status_check ) ) {
gp8psk_usb_in_op ( st - > d , GET_SIGNAL_LOCK , 0 , 0 , & st - > lock , 1 ) ;
gp8psk_usb_in_op ( st - > d , GET_SIGNAL_STRENGTH , 0 , 0 , buf , 6 ) ;
st - > snr = ( buf [ 1 ] ) < < 8 | buf [ 0 ] ;
st - > next_status_check = jiffies + ( st - > status_check_interval * HZ ) / 1000 ;
}
return 0 ;
}
2015-06-07 14:53:52 -03:00
static int gp8psk_fe_read_status ( struct dvb_frontend * fe ,
enum fe_status * status )
2006-05-14 13:23:56 -03:00
{
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
2007-08-18 17:52:35 -03:00
gp8psk_fe_update_status ( st ) ;
2006-05-14 13:23:56 -03:00
2007-08-18 17:52:35 -03:00
if ( st - > lock )
2006-05-14 13:23:56 -03:00
* status = FE_HAS_LOCK | FE_HAS_SYNC | FE_HAS_VITERBI | FE_HAS_SIGNAL | FE_HAS_CARRIER ;
else
* status = 0 ;
2007-08-18 17:52:35 -03:00
if ( * status & FE_HAS_LOCK )
st - > status_check_interval = 1000 ;
else
st - > status_check_interval = 100 ;
2006-05-14 13:23:56 -03: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-18 17:52:35 -03:00
gp8psk_fe_update_status ( st ) ;
/* snr is reported in dBu*256 */
* snr = st - > snr ;
2006-05-14 13:23:56 -03:00
return 0 ;
}
static int gp8psk_fe_read_signal_strength ( struct dvb_frontend * fe , u16 * strength )
{
2007-08-18 17:52:35 -03: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 13:23:56 -03:00
}
static int gp8psk_fe_get_tune_settings ( struct dvb_frontend * fe , struct dvb_frontend_tune_settings * tune )
{
2010-10-16 16:07:47 -03:00
tune - > min_delay_ms = 800 ;
2006-05-14 13:23:56 -03:00
return 0 ;
}
2011-12-26 15:49:16 -03:00
static int gp8psk_fe_set_frontend ( struct dvb_frontend * fe )
2006-05-14 13:23:56 -03:00
{
struct gp8psk_fe_state * state = fe - > demodulator_priv ;
2008-12-20 12:03:06 -03:00
struct dtv_frontend_properties * c = & fe - > dtv_property_cache ;
2006-05-14 13:23:56 -03:00
u8 cmd [ 10 ] ;
2011-12-26 15:49:16 -03:00
u32 freq = c - > frequency * 1000 ;
2008-12-20 12:03:06 -03:00
int gp_product_id = le16_to_cpu ( state - > d - > udev - > descriptor . idProduct ) ;
deb_fe ( " %s() \n " , __func__ ) ;
2006-05-14 13:23:56 -03:00
cmd [ 4 ] = freq & 0xff ;
cmd [ 5 ] = ( freq > > 8 ) & 0xff ;
cmd [ 6 ] = ( freq > > 16 ) & 0xff ;
cmd [ 7 ] = ( freq > > 24 ) & 0xff ;
2011-08-08 11:54:37 -03: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 12:03:06 -03:00
switch ( c - > delivery_system ) {
case SYS_DVBS :
2011-08-08 11:54:37 -03:00
if ( c - > modulation ! = QPSK ) {
2008-12-20 12:03:06 -03:00
deb_fe ( " %s: unsupported modulation selected (%d) \n " ,
__func__ , c - > modulation ) ;
return - EOPNOTSUPP ;
}
c - > fec_inner = FEC_AUTO ;
2006-05-14 13:23:56 -03:00
break ;
2011-08-08 11:54:37 -03:00
case SYS_DVBS2 : /* kept for backwards compatibility */
2008-12-20 12:03:06 -03:00
deb_fe ( " %s: DVB-S2 delivery system selected \n " , __func__ ) ;
break ;
2011-08-08 11:54:37 -03:00
case SYS_TURBO :
deb_fe ( " %s: Turbo-FEC delivery system selected \n " , __func__ ) ;
break ;
2008-12-20 12:03:06 -03:00
2006-05-14 13:23:56 -03:00
default :
2008-12-20 12:03:06 -03:00
deb_fe ( " %s: unsupported delivery system selected (%d) \n " ,
__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 :
if ( gp_product_id = = USB_PID_GENPIX_8PSK_REV_1_WARM )
if ( gp8psk_tuned_to_DCII ( fe ) )
gp8psk_bcm4500_reload ( state - > d ) ;
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 11:54:37 -03:00
if ( c - > delivery_system = = SYS_TURBO )
cmd [ 8 ] = ADV_MOD_TURBO_QPSK ;
else
cmd [ 8 ] = ADV_MOD_DVB_QPSK ;
2008-12-20 12: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 13:23:56 -03:00
cmd [ 9 ] = 0 ;
break ;
2008-12-20 12:03:06 -03:00
default : /* Unknown modulation */
deb_fe ( " %s: unsupported modulation selected (%d) \n " ,
__func__ , c - > modulation ) ;
return - EOPNOTSUPP ;
2006-05-14 13:23:56 -03:00
}
2008-12-20 12:03:06 -03:00
if ( gp_product_id = = USB_PID_GENPIX_8PSK_REV_1_WARM )
gp8psk_set_tuner_mode ( fe , 0 ) ;
gp8psk_usb_out_op ( state - > d , TUNE_8PSK , 0 , 0 , cmd , 10 ) ;
2006-05-14 13:23:56 -03:00
2007-08-18 17:52:35 -03:00
state - > lock = 0 ;
state - > next_status_check = jiffies ;
state - > status_check_interval = 200 ;
2006-05-14 13:23:56 -03: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 ;
2008-04-08 23:20:00 -03:00
deb_fe ( " %s \n " , __func__ ) ;
2006-05-14 13:23:56 -03:00
if ( gp8psk_usb_out_op ( st - > d , SEND_DISEQC_COMMAND , m - > msg [ 0 ] , 0 ,
m - > msg , m - > msg_len ) ) {
return - EINVAL ;
}
return 0 ;
}
2015-06-07 14:53:52 -03:00
static int gp8psk_fe_send_diseqc_burst ( struct dvb_frontend * fe ,
enum fe_sec_mini_cmd burst )
2006-05-14 13:23:56 -03:00
{
struct gp8psk_fe_state * st = fe - > demodulator_priv ;
u8 cmd ;
2008-04-08 23:20:00 -03:00
deb_fe ( " %s \n " , __func__ ) ;
2006-05-14 13:23:56 -03:00
/* These commands are certainly wrong */
cmd = ( burst = = SEC_MINI_A ) ? 0x00 : 0x01 ;
if ( gp8psk_usb_out_op ( st - > d , SEND_DISEQC_COMMAND , cmd , 0 ,
& cmd , 0 ) ) {
return - EINVAL ;
}
return 0 ;
}
2015-06-07 14:53:52 -03:00
static int gp8psk_fe_set_tone ( struct dvb_frontend * fe ,
enum fe_sec_tone_mode tone )
2006-05-14 13:23:56 -03:00
{
struct gp8psk_fe_state * state = fe - > demodulator_priv ;
if ( gp8psk_usb_out_op ( state - > d , SET_22KHZ_TONE ,
( tone = = SEC_TONE_ON ) , 0 , NULL , 0 ) ) {
return - EINVAL ;
}
return 0 ;
}
2015-06-07 14:53:52 -03:00
static int gp8psk_fe_set_voltage ( struct dvb_frontend * fe ,
enum fe_sec_voltage voltage )
2006-05-14 13:23:56 -03:00
{
struct gp8psk_fe_state * state = fe - > demodulator_priv ;
if ( gp8psk_usb_out_op ( state - > d , SET_LNB_VOLTAGE ,
voltage = = SEC_VOLTAGE_18 , 0 , NULL , 0 ) ) {
return - EINVAL ;
}
return 0 ;
}
2007-08-18 17:52:35 -03:00
static int gp8psk_fe_enable_high_lnb_voltage ( struct dvb_frontend * fe , long onoff )
{
struct gp8psk_fe_state * state = fe - > demodulator_priv ;
return gp8psk_usb_out_op ( state - > d , USE_EXTRA_VOLT , onoff , 0 , NULL , 0 ) ;
}
2006-05-14 13:23:56 -03:00
static int gp8psk_fe_send_legacy_dish_cmd ( struct dvb_frontend * fe , unsigned long sw_cmd )
{
struct gp8psk_fe_state * state = fe - > demodulator_priv ;
u8 cmd = sw_cmd & 0x7f ;
if ( gp8psk_usb_out_op ( state - > d , SET_DN_SWITCH , cmd , 0 ,
NULL , 0 ) ) {
return - EINVAL ;
}
if ( gp8psk_usb_out_op ( state - > d , SET_LNB_VOLTAGE , ! ! ( sw_cmd & 0x80 ) ,
0 , NULL , 0 ) ) {
return - EINVAL ;
}
return 0 ;
}
static void gp8psk_fe_release ( struct dvb_frontend * fe )
{
struct gp8psk_fe_state * state = fe - > demodulator_priv ;
kfree ( state ) ;
}
static struct dvb_frontend_ops gp8psk_fe_ops ;
struct dvb_frontend * gp8psk_fe_attach ( struct dvb_usb_device * d )
{
struct gp8psk_fe_state * s = kzalloc ( sizeof ( struct gp8psk_fe_state ) , GFP_KERNEL ) ;
if ( s = = NULL )
goto error ;
s - > d = d ;
memcpy ( & s - > fe . ops , & gp8psk_fe_ops , sizeof ( struct dvb_frontend_ops ) ) ;
s - > fe . demodulator_priv = s ;
goto success ;
error :
return NULL ;
success :
return & s - > fe ;
}
static struct dvb_frontend_ops gp8psk_fe_ops = {
2011-12-26 15:49:16 -03:00
. delsys = { SYS_DVBS } ,
2006-05-14 13:23:56 -03:00
. info = {
2010-10-16 16:18:15 -03:00
. name = " Genpix DVB-S " ,
2007-08-18 17:52:35 -03:00
. frequency_min = 800000 ,
. frequency_max = 2250000 ,
2006-05-14 13:23:56 -03:00
. frequency_stepsize = 100 ,
. symbol_rate_min = 1000000 ,
. symbol_rate_max = 45000000 ,
. symbol_rate_tolerance = 500 , /* ppm */
. caps = FE_CAN_INVERSION_AUTO |
2008-12-20 12: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 01:37:34 -03:00
FE_CAN_QPSK | FE_CAN_QAM_16 | FE_CAN_TURBO_FEC
2006-05-14 13:23:56 -03:00
} ,
. release = gp8psk_fe_release ,
. init = NULL ,
. sleep = NULL ,
2011-12-26 15:49:16 -03:00
. set_frontend = gp8psk_fe_set_frontend ,
2008-12-20 12:03:06 -03:00
2006-05-14 13:23:56 -03: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-18 17:52:35 -03:00
. enable_high_lnb_voltage = gp8psk_fe_enable_high_lnb_voltage
2006-05-14 13:23:56 -03:00
} ;