2005-04-16 15:20:36 -07:00
/*
tea6420 - i2c - driver for the tea6420 by SGS Thomson
Copyright ( C ) 1998 - 2003 Michael Hunold < michael @ mihu . de >
2008-07-25 10:31:23 -03:00
Copyright ( C ) 2008 Hans Verkuil < hverkuil @ xs4all . nl >
2005-04-16 15:20:36 -07:00
The tea6420 is a bus controlled audio - matrix with 5 stereo inputs ,
4 stereo outputs and gain control for each output .
It is cascadable , i . e . it can be found at the adresses 0x98
and 0x9a on the i2c - bus .
For detailed informations download the specifications directly
from SGS Thomson at http : //www.st.com
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-03-17 10:37:02 -03:00
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
# include <linux/ioctl.h>
# include <linux/i2c.h>
2008-12-18 12:33:45 -03:00
# include <media/v4l2-device.h>
2009-02-07 11:18:05 -03:00
# include <media/v4l2-chip-ident.h>
# include <media/v4l2-i2c-drv.h>
2005-04-16 15:20:36 -07:00
# include "tea6420.h"
2008-07-25 10:31:23 -03:00
MODULE_AUTHOR ( " Michael Hunold <michael@mihu.de> " ) ;
MODULE_DESCRIPTION ( " tea6420 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
static int debug ;
2005-04-16 15:20:36 -07:00
module_param ( debug , int , 0644 ) ;
2008-07-18 00:50:58 -03:00
2008-07-25 10:31:23 -03:00
MODULE_PARM_DESC ( debug , " Debug level (0-1) " ) ;
2005-04-16 15:20:36 -07:00
/* make a connection between the input 'i' and the output 'o'
2009-02-07 11:18:05 -03:00
with gain ' g ' ( note : i = 6 means ' mute ' ) */
2009-04-02 11:26:22 -03:00
static int tea6420_s_routing ( struct v4l2_subdev * sd ,
u32 i , u32 o , u32 config )
2005-04-16 15:20:36 -07:00
{
2009-02-07 11:18:05 -03:00
struct i2c_client * client = v4l2_get_subdevdata ( sd ) ;
2009-04-02 11:26:22 -03:00
int g = ( o > > 4 ) & 0xf ;
2008-07-25 10:31:23 -03:00
u8 byte ;
2005-04-16 15:20:36 -07:00
int ret ;
2009-04-02 11:26:22 -03:00
o & = 0xf ;
2009-02-07 11:18:05 -03:00
v4l2_dbg ( 1 , debug , sd , " i=%d, o=%d, g=%d \n " , i , o , g ) ;
2005-04-16 15:20:36 -07:00
2008-02-03 17:18:59 +02:00
/* check if the parameters are valid */
2005-04-16 15:20:36 -07:00
if ( i < 1 | | i > 6 | | o < 1 | | o > 4 | | g < 0 | | g > 6 | | g % 2 ! = 0 )
2009-02-07 11:18:05 -03:00
return - EINVAL ;
2005-04-16 15:20:36 -07:00
2008-07-25 10:31:23 -03:00
byte = ( ( o - 1 ) < < 5 ) ;
2005-04-16 15:20:36 -07:00
byte | = ( i - 1 ) ;
/* to understand this, have a look at the tea6420-specs (p.5) */
switch ( g ) {
case 0 :
byte | = ( 3 < < 3 ) ;
break ;
case 2 :
byte | = ( 2 < < 3 ) ;
break ;
case 4 :
byte | = ( 1 < < 3 ) ;
break ;
case 6 :
break ;
}
ret = i2c_smbus_write_byte ( client , byte ) ;
if ( ret ) {
2009-02-07 11:18:05 -03:00
v4l2_dbg ( 1 , debug , sd ,
2008-07-25 10:31:23 -03:00
" i2c_smbus_write_byte() failed, ret:%d \n " , ret ) ;
2005-04-16 15:20:36 -07:00
return - EIO ;
}
return 0 ;
}
2009-02-07 11:18:05 -03:00
static int tea6420_g_chip_ident ( struct v4l2_subdev * sd , struct v4l2_dbg_chip_ident * chip )
2005-04-16 15:20:36 -07:00
{
2009-02-07 11:18:05 -03:00
struct i2c_client * client = v4l2_get_subdevdata ( sd ) ;
2005-04-16 15:20:36 -07:00
2009-02-07 11:18:05 -03:00
return v4l2_chip_ident_i2c_client ( client , chip , V4L2_IDENT_TEA6420 , 0 ) ;
2008-07-25 10:31:23 -03:00
}
2005-04-16 15:20:36 -07:00
2008-12-18 12:33:45 -03:00
/* ----------------------------------------------------------------------- */
static const struct v4l2_subdev_core_ops tea6420_core_ops = {
2009-02-07 11:18:05 -03:00
. g_chip_ident = tea6420_g_chip_ident ,
} ;
static const struct v4l2_subdev_audio_ops tea6420_audio_ops = {
. s_routing = tea6420_s_routing ,
2008-12-18 12:33:45 -03:00
} ;
static const struct v4l2_subdev_ops tea6420_ops = {
. core = & tea6420_core_ops ,
2009-02-07 11:18:05 -03:00
. audio = & tea6420_audio_ops ,
2008-12-18 12:33:45 -03:00
} ;
2008-07-25 10:31:23 -03:00
static int tea6420_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
{
2008-12-18 12:33:45 -03:00
struct v4l2_subdev * sd ;
2008-07-25 10:31:23 -03:00
int err , i ;
2005-04-16 15:20:36 -07:00
2008-07-25 10:31:23 -03:00
/* let's see whether this adapter can support what we need */
if ( ! i2c_check_functionality ( client - > adapter , I2C_FUNC_SMBUS_WRITE_BYTE ) )
return - EIO ;
v4l_info ( client , " chip found @ 0x%x (%s) \n " ,
client - > addr < < 1 , client - > adapter - > name ) ;
2005-04-16 15:20:36 -07:00
2009-02-07 11:18:05 -03:00
sd = kmalloc ( sizeof ( struct v4l2_subdev ) , GFP_KERNEL ) ;
if ( sd = = NULL )
return - ENOMEM ;
v4l2_i2c_subdev_init ( sd , client , & tea6420_ops ) ;
2005-04-16 15:20:36 -07:00
/* set initial values: set "mute"-input to all outputs at gain 0 */
err = 0 ;
2009-04-02 11:26:22 -03:00
for ( i = 1 ; i < 5 ; i + + )
err + = tea6420_s_routing ( sd , 6 , i , 0 ) ;
2005-04-16 15:20:36 -07:00
if ( err ) {
2008-07-25 10:31:23 -03:00
v4l_dbg ( 1 , debug , client , " could not initialize tea6420 \n " ) ;
2005-04-16 15:20:36 -07:00
return - ENODEV ;
}
2008-12-18 12:33:45 -03:00
return 0 ;
}
static int tea6420_remove ( struct i2c_client * client )
{
struct v4l2_subdev * sd = i2c_get_clientdata ( client ) ;
v4l2_device_unregister_subdev ( sd ) ;
kfree ( sd ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2008-07-25 10:31:23 -03:00
static const struct i2c_device_id tea6420_id [ ] = {
{ " tea6420 " , 0 } ,
{ }
2005-04-16 15:20:36 -07:00
} ;
2008-07-25 10:31:23 -03:00
MODULE_DEVICE_TABLE ( i2c , tea6420_id ) ;
2005-04-16 15:20:36 -07:00
2008-07-25 10:31:23 -03:00
static struct v4l2_i2c_driver_data v4l2_i2c_data = {
2005-08-15 19:57:04 +02:00
. name = " tea6420 " ,
2008-07-25 10:31:23 -03:00
. probe = tea6420_probe ,
2008-12-18 12:33:45 -03:00
. remove = tea6420_remove ,
2008-07-25 10:31:23 -03:00
. id_table = tea6420_id ,
2005-04-16 15:20:36 -07:00
} ;