2005-06-28 20:45:21 -07:00
/*
* For Philips TEA5767 FM Chip used on some TV Cards like Prolink Pixelview
* I2C address is allways 0xC0 .
*
2005-07-12 13:58:55 -07:00
* $ Id : tea5767 . c , v 1.18 2005 / 07 / 07 03 : 02 : 55 mchehab Exp $
2005-06-28 20:45:21 -07:00
*
* Copyright ( c ) 2005 Mauro Carvalho Chehab ( mchehab @ brturbo . com . br )
* This code is placed under the terms of the GNU General Public License
*
* tea5767 autodetection thanks to Torsten Seeboth and Atsushi Nakagawa
* from their contributions on DScaler .
*/
# include <linux/i2c.h>
2005-07-12 13:58:55 -07:00
# include <linux/videodev.h>
# include <linux/delay.h>
# include <media/tuner.h>
2005-06-28 20:45:21 -07:00
# include <media/tuner.h>
# define PREFIX "TEA5767 "
/*****************************************************************************/
/******************************
* Write mode register values *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/* First register */
2005-07-12 13:58:55 -07:00
# define TEA5767_MUTE 0x80 /* Mutes output */
# define TEA5767_SEARCH 0x40 /* Activates station search */
2005-06-28 20:45:21 -07:00
/* Bits 0-5 for divider MSB */
/* Second register */
/* Bits 0-7 for divider LSB */
/* Third register */
/* Station search from botton to up */
# define TEA5767_SEARCH_UP 0x80
/* Searches with ADC output = 10 */
# define TEA5767_SRCH_HIGH_LVL 0x60
/* Searches with ADC output = 10 */
# define TEA5767_SRCH_MID_LVL 0x40
/* Searches with ADC output = 5 */
# define TEA5767_SRCH_LOW_LVL 0x20
/* if on, div=4*(Frf+Fif)/Fref otherwise, div=4*(Frf-Fif)/Freq) */
# define TEA5767_HIGH_LO_INJECT 0x10
/* Disable stereo */
# define TEA5767_MONO 0x08
/* Disable right channel and turns to mono */
# define TEA5767_MUTE_RIGHT 0x04
/* Disable left channel and turns to mono */
# define TEA5767_MUTE_LEFT 0x02
# define TEA5767_PORT1_HIGH 0x01
/* Forth register */
# define TEA5767_PORT2_HIGH 0x80
/* Chips stops working. Only I2C bus remains on */
# define TEA5767_STDBY 0x40
/* Japan freq (76-108 MHz. If disabled, 87.5-108 MHz */
# define TEA5767_JAPAN_BAND 0x20
/* Unselected means 32.768 KHz freq as reference. Otherwise Xtal at 13 MHz */
# define TEA5767_XTAL_32768 0x10
/* Cuts weak signals */
# define TEA5767_SOFT_MUTE 0x08
/* Activates high cut control */
# define TEA5767_HIGH_CUT_CTRL 0x04
/* Activates stereo noise control */
# define TEA5767_ST_NOISE_CTL 0x02
/* If activate PORT 1 indicates SEARCH or else it is used as PORT1 */
# define TEA5767_SRCH_IND 0x01
/* Fiveth register */
/* By activating, it will use Xtal at 13 MHz as reference for divider */
# define TEA5767_PLLREF_ENABLE 0x80
/* By activating, deemphasis=50, or else, deemphasis of 50us */
# define TEA5767_DEEMPH_75 0X40
/*****************************
* Read mode register values *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/* First register */
# define TEA5767_READY_FLAG_MASK 0x80
# define TEA5767_BAND_LIMIT_MASK 0X40
/* Bits 0-5 for divider MSB after search or preset */
/* Second register */
/* Bits 0-7 for divider LSB after search or preset */
/* Third register */
# define TEA5767_STEREO_MASK 0x80
# define TEA5767_IF_CNTR_MASK 0x7f
/* Four register */
# define TEA5767_ADC_LEVEL_MASK 0xf0
/* should be 0 */
# define TEA5767_CHIP_ID_MASK 0x0f
/* Fiveth register */
/* Reserved for future extensions */
# define TEA5767_RESERVED_MASK 0xff
2005-07-12 13:58:55 -07:00
enum tea5767_xtal_freq {
TEA5767_LOW_LO_32768 = 0 ,
TEA5767_HIGH_LO_32768 = 1 ,
TEA5767_LOW_LO_13MHz = 2 ,
TEA5767_HIGH_LO_13MHz = 3 ,
} ;
2005-06-28 20:45:21 -07:00
/*****************************************************************************/
static void set_tv_freq ( struct i2c_client * c , unsigned int freq )
{
struct tuner * t = i2c_get_clientdata ( c ) ;
tuner_warn ( " This tuner doesn't support TV freq. \n " ) ;
}
static void tea5767_status_dump ( unsigned char * buffer )
{
unsigned int div , frq ;
if ( TEA5767_READY_FLAG_MASK & buffer [ 0 ] )
printk ( PREFIX " Ready Flag ON \n " ) ;
else
printk ( PREFIX " Ready Flag OFF \n " ) ;
if ( TEA5767_BAND_LIMIT_MASK & buffer [ 0 ] )
printk ( PREFIX " Tuner at band limit \n " ) ;
else
printk ( PREFIX " Tuner not at band limit \n " ) ;
2005-07-12 13:58:55 -07:00
div = ( ( buffer [ 0 ] & 0x3f ) < < 8 ) | buffer [ 1 ] ;
2005-06-28 20:45:21 -07:00
switch ( TEA5767_HIGH_LO_32768 ) {
case TEA5767_HIGH_LO_13MHz :
2005-07-12 13:58:55 -07:00
frq = 1000 * ( div * 50 - 700 - 225 ) / 4 ; /* Freq in KHz */
2005-06-28 20:45:21 -07:00
break ;
case TEA5767_LOW_LO_13MHz :
2005-07-12 13:58:55 -07:00
frq = 1000 * ( div * 50 + 700 + 225 ) / 4 ; /* Freq in KHz */
2005-06-28 20:45:21 -07:00
break ;
case TEA5767_LOW_LO_32768 :
2005-07-12 13:58:55 -07:00
frq = 1000 * ( div * 32768 / 1000 + 700 + 225 ) / 4 ; /* Freq in KHz */
2005-06-28 20:45:21 -07:00
break ;
case TEA5767_HIGH_LO_32768 :
default :
2005-07-12 13:58:55 -07:00
frq = 1000 * ( div * 32768 / 1000 - 700 - 225 ) / 4 ; /* Freq in KHz */
2005-06-28 20:45:21 -07:00
break ;
}
2005-07-12 13:58:55 -07:00
buffer [ 0 ] = ( div > > 8 ) & 0x3f ;
buffer [ 1 ] = div & 0xff ;
2005-06-28 20:45:21 -07:00
printk ( PREFIX " Frequency %d.%03d KHz (divider = 0x%04x) \n " ,
2005-07-12 13:58:55 -07:00
frq / 1000 , frq % 1000 , div ) ;
2005-06-28 20:45:21 -07:00
if ( TEA5767_STEREO_MASK & buffer [ 2 ] )
printk ( PREFIX " Stereo \n " ) ;
else
printk ( PREFIX " Mono \n " ) ;
2005-07-12 13:58:55 -07:00
printk ( PREFIX " IF Counter = %d \n " , buffer [ 2 ] & TEA5767_IF_CNTR_MASK ) ;
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
printk ( PREFIX " ADC Level = %d \n " ,
( buffer [ 3 ] & TEA5767_ADC_LEVEL_MASK ) > > 4 ) ;
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
printk ( PREFIX " Chip ID = %d \n " , ( buffer [ 3 ] & TEA5767_CHIP_ID_MASK ) ) ;
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
printk ( PREFIX " Reserved = 0x%02x \n " ,
( buffer [ 4 ] & TEA5767_RESERVED_MASK ) ) ;
2005-06-28 20:45:21 -07:00
}
/* Freq should be specifyed at 62.5 Hz */
static void set_radio_freq ( struct i2c_client * c , unsigned int frq )
{
struct tuner * t = i2c_get_clientdata ( c ) ;
2005-07-12 13:58:55 -07:00
unsigned char buffer [ 5 ] ;
2005-06-28 20:45:21 -07:00
unsigned div ;
int rc ;
2005-07-12 13:58:55 -07:00
tuner_dbg ( PREFIX " radio freq counter %d \n " , frq ) ;
2005-06-28 20:45:21 -07:00
/* Rounds freq to next decimal value - for 62.5 KHz step */
/* frq = 20*(frq/16)+radio_frq[frq%16]; */
buffer [ 2 ] = TEA5767_PORT1_HIGH ;
2005-07-12 13:58:55 -07:00
buffer [ 3 ] = TEA5767_PORT2_HIGH | TEA5767_HIGH_CUT_CTRL |
TEA5767_ST_NOISE_CTL | TEA5767_JAPAN_BAND ;
buffer [ 4 ] = 0 ;
if ( t - > mode = = T_STANDBY ) {
tuner_dbg ( " TEA5767 set to standby mode \n " ) ;
buffer [ 3 ] | = TEA5767_STDBY ;
}
2005-06-28 20:45:21 -07:00
if ( t - > audmode = = V4L2_TUNER_MODE_MONO ) {
tuner_dbg ( " TEA5767 set to mono \n " ) ;
buffer [ 2 ] | = TEA5767_MONO ;
2005-07-12 13:58:55 -07:00
} else {
tuner_dbg ( " TEA5767 set to stereo \n " ) ;
}
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
/* Should be replaced */
switch ( TEA5767_HIGH_LO_32768 ) {
2005-06-28 20:45:21 -07:00
case TEA5767_HIGH_LO_13MHz :
2005-07-12 13:58:55 -07:00
tuner_dbg ( " TEA5767 radio HIGH LO inject xtal @ 13 MHz \n " ) ;
2005-06-28 20:45:21 -07:00
buffer [ 2 ] | = TEA5767_HIGH_LO_INJECT ;
buffer [ 4 ] | = TEA5767_PLLREF_ENABLE ;
2005-07-12 13:58:55 -07:00
div = ( frq * 4 / 16 + 700 + 225 + 25 ) / 50 ;
2005-06-28 20:45:21 -07:00
break ;
case TEA5767_LOW_LO_13MHz :
2005-07-12 13:58:55 -07:00
tuner_dbg ( " TEA5767 radio LOW LO inject xtal @ 13 MHz \n " ) ;
2005-06-28 20:45:21 -07:00
buffer [ 4 ] | = TEA5767_PLLREF_ENABLE ;
2005-07-12 13:58:55 -07:00
div = ( frq * 4 / 16 - 700 - 225 + 25 ) / 50 ;
2005-06-28 20:45:21 -07:00
break ;
case TEA5767_LOW_LO_32768 :
2005-07-12 13:58:55 -07:00
tuner_dbg ( " TEA5767 radio LOW LO inject xtal @ 32,768 MHz \n " ) ;
2005-06-28 20:45:21 -07:00
buffer [ 3 ] | = TEA5767_XTAL_32768 ;
/* const 700=4000*175 Khz - to adjust freq to right value */
2005-07-12 13:58:55 -07:00
div = ( 1000 * ( frq * 4 / 16 - 700 - 225 ) + 16384 ) > > 15 ;
2005-06-28 20:45:21 -07:00
break ;
case TEA5767_HIGH_LO_32768 :
default :
2005-07-12 13:58:55 -07:00
tuner_dbg ( " TEA5767 radio HIGH LO inject xtal @ 32,768 MHz \n " ) ;
2005-06-28 20:45:21 -07:00
buffer [ 2 ] | = TEA5767_HIGH_LO_INJECT ;
buffer [ 3 ] | = TEA5767_XTAL_32768 ;
2005-07-12 13:58:55 -07:00
div = ( 1000 * ( frq * 4 / 16 + 700 + 225 ) + 16384 ) > > 15 ;
2005-06-28 20:45:21 -07:00
break ;
}
2005-07-12 13:58:55 -07:00
buffer [ 0 ] = ( div > > 8 ) & 0x3f ;
buffer [ 1 ] = div & 0xff ;
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
if ( tuner_debug )
2005-06-28 20:45:21 -07:00
tea5767_status_dump ( buffer ) ;
2005-07-12 13:58:55 -07:00
if ( 5 ! = ( rc = i2c_master_send ( c , buffer , 5 ) ) )
tuner_warn ( " i2c i/o error: rc == %d (should be 5) \n " , rc ) ;
2005-06-28 20:45:21 -07:00
}
static int tea5767_signal ( struct i2c_client * c )
{
unsigned char buffer [ 5 ] ;
int rc ;
struct tuner * t = i2c_get_clientdata ( c ) ;
2005-07-12 13:58:55 -07:00
memset ( buffer , 0 , sizeof ( buffer ) ) ;
if ( 5 ! = ( rc = i2c_master_recv ( c , buffer , 5 ) ) )
tuner_warn ( " i2c i/o error: rc == %d (should be 5) \n " , rc ) ;
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
return ( ( buffer [ 3 ] & TEA5767_ADC_LEVEL_MASK ) < < ( 13 - 4 ) ) ;
2005-06-28 20:45:21 -07:00
}
static int tea5767_stereo ( struct i2c_client * c )
{
unsigned char buffer [ 5 ] ;
int rc ;
struct tuner * t = i2c_get_clientdata ( c ) ;
2005-07-12 13:58:55 -07:00
memset ( buffer , 0 , sizeof ( buffer ) ) ;
if ( 5 ! = ( rc = i2c_master_recv ( c , buffer , 5 ) ) )
tuner_warn ( " i2c i/o error: rc == %d (should be 5) \n " , rc ) ;
2005-06-28 20:45:21 -07:00
rc = buffer [ 2 ] & TEA5767_STEREO_MASK ;
2005-07-12 13:58:55 -07:00
tuner_dbg ( " TEA5767 radio ST GET = %02x \n " , rc ) ;
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
return ( ( buffer [ 2 ] & TEA5767_STEREO_MASK ) ? V4L2_TUNER_SUB_STEREO : 0 ) ;
2005-06-28 20:45:21 -07:00
}
2005-07-12 13:58:55 -07:00
int tea5767_autodetection ( struct i2c_client * c )
2005-06-28 20:45:21 -07:00
{
2005-07-12 13:58:55 -07:00
unsigned char buffer [ 5 ] = { 0xff , 0xff , 0xff , 0xff , 0xff } ;
2005-06-28 20:45:21 -07:00
int rc ;
struct tuner * t = i2c_get_clientdata ( c ) ;
2005-07-12 13:58:55 -07:00
if ( 5 ! = ( rc = i2c_master_recv ( c , buffer , 5 ) ) ) {
tuner_warn ( " it is not a TEA5767. Received %i chars. \n " , rc ) ;
2005-06-28 20:45:21 -07:00
return EINVAL ;
}
/* If all bytes are the same then it's a TV tuner and not a tea5767 chip. */
2005-07-12 13:58:55 -07:00
if ( buffer [ 0 ] = = buffer [ 1 ] & & buffer [ 0 ] = = buffer [ 2 ] & &
buffer [ 0 ] = = buffer [ 3 ] & & buffer [ 0 ] = = buffer [ 4 ] ) {
tuner_warn ( " All bytes are equal. It is not a TEA5767 \n " ) ;
2005-06-28 20:45:21 -07:00
return EINVAL ;
}
/* Status bytes:
* Byte 4 : bit 3 : 1 : CI ( Chip Identification ) = = 0
2005-07-12 13:58:55 -07:00
* bit 0 : internally set to 0
2005-06-28 20:45:21 -07:00
* Byte 5 : bit 7 : 0 : = = 0
*/
if ( ! ( ( buffer [ 3 ] & 0x0f ) = = 0x00 ) & & ( buffer [ 4 ] = = 0x00 ) ) {
2005-07-12 13:58:55 -07:00
tuner_warn ( " Chip ID is not zero. It is not a TEA5767 \n " ) ;
2005-06-28 20:45:21 -07:00
return EINVAL ;
}
2005-07-12 13:58:55 -07:00
tuner_warn ( " TEA5767 detected. \n " ) ;
2005-06-28 20:45:21 -07:00
return 0 ;
}
int tea5767_tuner_init ( struct i2c_client * c )
{
struct tuner * t = i2c_get_clientdata ( c ) ;
2005-07-12 13:58:55 -07:00
if ( tea5767_autodetection ( c ) = = EINVAL )
return EINVAL ;
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
tuner_info ( " type set to %d (%s) \n " , t - > type , " Philips TEA5767HN FM Radio " ) ;
strlcpy ( c - > name , " tea5767 " , sizeof ( c - > name ) ) ;
2005-06-28 20:45:21 -07:00
2005-07-12 13:58:55 -07:00
t - > tv_freq = set_tv_freq ;
2005-06-28 20:45:21 -07:00
t - > radio_freq = set_radio_freq ;
t - > has_signal = tea5767_signal ;
2005-07-12 13:58:55 -07:00
t - > is_stereo = tea5767_stereo ;
2005-06-28 20:45:21 -07:00
return ( 0 ) ;
}