2007-07-27 06:56:50 -03:00
/*
* vp27smpx - driver version 0.0 .1
*
* Copyright ( C ) 2007 Hans Verkuil < hverkuil @ xs4all . nl >
*
2007-08-18 11:39:28 -03:00
* Based on a tvaudio patch from Takahiro Adachi < tadachi @ tadachi - net . com >
* and Kazuhiko Kawakami < kazz - 0 @ mail . goo . ne . jp >
2007-07-27 06:56:50 -03:00
*
* 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 .
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/ioctl.h>
# include <asm/uaccess.h>
# include <linux/i2c.h>
# include <linux/i2c-id.h>
2008-07-25 05:32:50 -03:00
# include <linux/videodev2.h>
2007-07-27 06:56:50 -03:00
# include <media/v4l2-common.h>
# include <media/v4l2-chip-ident.h>
2007-12-12 07:24:27 -03:00
# include <media/v4l2-i2c-drv.h>
2007-07-27 06:56:50 -03:00
MODULE_DESCRIPTION ( " vp27smpx driver " ) ;
MODULE_AUTHOR ( " Hans Verkuil " ) ;
MODULE_LICENSE ( " GPL " ) ;
/* ----------------------------------------------------------------------- */
struct vp27smpx_state {
int radio ;
u32 audmode ;
} ;
static void vp27smpx_set_audmode ( struct i2c_client * client , u32 audmode )
{
struct vp27smpx_state * state = i2c_get_clientdata ( client ) ;
u8 data [ 3 ] = { 0x00 , 0x00 , 0x04 } ;
switch ( audmode ) {
2007-12-12 07:40:54 -03:00
case V4L2_TUNER_MODE_MONO :
case V4L2_TUNER_MODE_LANG1 :
break ;
case V4L2_TUNER_MODE_STEREO :
case V4L2_TUNER_MODE_LANG1_LANG2 :
data [ 1 ] = 0x01 ;
break ;
case V4L2_TUNER_MODE_LANG2 :
data [ 1 ] = 0x02 ;
break ;
2007-07-27 06:56:50 -03:00
}
2007-12-12 07:40:54 -03:00
if ( i2c_master_send ( client , data , sizeof ( data ) ) ! = sizeof ( data ) )
v4l_err ( client , " %s: I/O error setting audmode \n " ,
client - > name ) ;
else
2007-07-27 06:56:50 -03:00
state - > audmode = audmode ;
}
2007-12-12 07:40:54 -03:00
static int vp27smpx_command ( struct i2c_client * client , unsigned cmd , void * arg )
2007-07-27 06:56:50 -03:00
{
struct vp27smpx_state * state = i2c_get_clientdata ( client ) ;
struct v4l2_tuner * vt = arg ;
switch ( cmd ) {
case AUDC_SET_RADIO :
state - > radio = 1 ;
break ;
case VIDIOC_S_STD :
state - > radio = 0 ;
break ;
case VIDIOC_S_TUNER :
if ( ! state - > radio )
vp27smpx_set_audmode ( client , vt - > audmode ) ;
break ;
case VIDIOC_G_TUNER :
if ( state - > radio )
break ;
vt - > audmode = state - > audmode ;
vt - > capability = V4L2_TUNER_CAP_STEREO |
V4L2_TUNER_CAP_LANG1 | V4L2_TUNER_CAP_LANG2 ;
vt - > rxsubchans = V4L2_TUNER_SUB_MONO ;
break ;
case VIDIOC_G_CHIP_IDENT :
2007-12-12 07:40:54 -03:00
return v4l2_chip_ident_i2c_client ( client , arg ,
V4L2_IDENT_VP27SMPX , 0 ) ;
2007-07-27 06:56:50 -03:00
case VIDIOC_LOG_STATUS :
v4l_info ( client , " Audio Mode: %u%s \n " , state - > audmode ,
state - > radio ? " (Radio) " : " " ) ;
break ;
default :
return - EINVAL ;
}
return 0 ;
}
/* ----------------------------------------------------------------------- */
/* i2c implementation */
/*
* Generic i2c probe
* concerning the addresses : i2c wants 7 bit ( without the r / w bit ) , so ' > > 1 '
*/
2008-04-29 23:11:39 +02:00
static int vp27smpx_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
2007-07-27 06:56:50 -03:00
{
struct vp27smpx_state * state ;
/* Check if the adapter supports the needed features */
2007-09-13 11:11:44 -03:00
if ( ! i2c_check_functionality ( client - > adapter , I2C_FUNC_SMBUS_BYTE_DATA ) )
2007-09-16 10:47:15 -03:00
return - EIO ;
2007-07-27 06:56:50 -03:00
2007-12-12 07:40:54 -03:00
v4l_info ( client , " chip found @ 0x%x (%s) \n " ,
client - > addr < < 1 , client - > adapter - > name ) ;
2007-07-27 06:56:50 -03:00
state = kzalloc ( sizeof ( struct vp27smpx_state ) , GFP_KERNEL ) ;
2007-12-12 07:40:54 -03:00
if ( state = = NULL )
2007-07-27 06:56:50 -03:00
return - ENOMEM ;
state - > audmode = V4L2_TUNER_MODE_STEREO ;
i2c_set_clientdata ( client , state ) ;
/* initialize vp27smpx */
vp27smpx_set_audmode ( client , state - > audmode ) ;
return 0 ;
}
2007-09-13 11:11:44 -03:00
static int vp27smpx_remove ( struct i2c_client * client )
2007-07-27 06:56:50 -03:00
{
2007-09-13 11:11:44 -03:00
kfree ( i2c_get_clientdata ( client ) ) ;
2007-07-27 06:56:50 -03:00
return 0 ;
}
/* ----------------------------------------------------------------------- */
2008-05-18 20:49:40 +02:00
static const struct i2c_device_id vp27smpx_id [ ] = {
{ " vp27smpx " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , vp27smpx_id ) ;
2007-09-13 11:11:44 -03:00
static struct v4l2_i2c_driver_data v4l2_i2c_data = {
. name = " vp27smpx " ,
. driverid = I2C_DRIVERID_VP27SMPX ,
. command = vp27smpx_command ,
. probe = vp27smpx_probe ,
. remove = vp27smpx_remove ,
2008-05-18 20:49:40 +02:00
. id_table = vp27smpx_id ,
2007-07-27 06:56:50 -03:00
} ;