2006-03-28 18:23:48 -03:00
/*
* wm8739
*
* Copyright ( C ) 2005 T . Adachi < tadachi @ tadachi - net . com >
*
* Copyright ( C ) 2005 Hans Verkuil < hverkuil @ xs4all . nl >
* - Cleanup
*
* 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>
2008-11-29 13:02:26 -03:00
# include <media/v4l2-device.h>
2007-04-27 12:31:26 -03:00
# include <media/v4l2-chip-ident.h>
2007-12-12 07:24:27 -03:00
# include <media/v4l2-i2c-drv.h>
2006-03-28 18:23:48 -03:00
MODULE_DESCRIPTION ( " wm8739 driver " ) ;
MODULE_AUTHOR ( " T. Adachi, Hans Verkuil " ) ;
MODULE_LICENSE ( " GPL " ) ;
2007-11-01 07:45:54 -03:00
static int debug ;
2006-03-28 18:23:48 -03:00
module_param ( debug , int , 0644 ) ;
MODULE_PARM_DESC ( debug , " Debug level (0-1) " ) ;
/* ------------------------------------------------------------------------ */
enum {
R0 = 0 , R1 ,
R5 = 5 , R6 , R7 , R8 , R9 , R15 = 15 ,
TOT_REGS
} ;
struct wm8739_state {
2008-11-29 13:02:26 -03:00
struct v4l2_subdev sd ;
2006-03-28 18:23:48 -03:00
u32 clock_freq ;
u8 muted ;
u16 volume ;
u16 balance ;
u8 vol_l ; /* +12dB to -34.5dB 1.5dB step (5bit) def:0dB */
u8 vol_r ; /* +12dB to -34.5dB 1.5dB step (5bit) def:0dB */
} ;
2008-11-29 13:02:26 -03:00
static inline struct wm8739_state * to_state ( struct v4l2_subdev * sd )
{
return container_of ( sd , struct wm8739_state , sd ) ;
}
2006-03-28 18:23:48 -03:00
/* ------------------------------------------------------------------------ */
2008-11-29 13:02:26 -03:00
static int wm8739_write ( struct v4l2_subdev * sd , int reg , u16 val )
2006-03-28 18:23:48 -03:00
{
2008-11-29 13:02:26 -03:00
struct i2c_client * client = v4l2_get_subdevdata ( sd ) ;
2006-03-28 18:23:48 -03:00
int i ;
if ( reg < 0 | | reg > = TOT_REGS ) {
2008-11-29 13:02:26 -03:00
v4l2_err ( sd , " Invalid register R%d \n " , reg ) ;
2006-03-28 18:23:48 -03:00
return - 1 ;
}
2008-11-29 13:02:26 -03:00
v4l2_dbg ( 1 , debug , sd , " write: %02x %02x \n " , reg , val ) ;
2006-03-28 18:23:48 -03:00
2007-11-01 07:45:54 -03:00
for ( i = 0 ; i < 3 ; i + + )
if ( i2c_smbus_write_byte_data ( client ,
( reg < < 1 ) | ( val > > 8 ) , val & 0xff ) = = 0 )
2006-03-28 18:23:48 -03:00
return 0 ;
2008-11-29 13:02:26 -03:00
v4l2_err ( sd , " I2C: cannot write %03x to register R%d \n " , val , reg ) ;
2006-03-28 18:23:48 -03:00
return - 1 ;
}
/* write regs to set audio volume etc */
2008-11-29 13:02:26 -03:00
static void wm8739_set_audio ( struct v4l2_subdev * sd )
2006-03-28 18:23:48 -03:00
{
2008-11-29 13:02:26 -03:00
struct wm8739_state * state = to_state ( sd ) ;
2006-03-28 18:23:48 -03:00
u16 mute = state - > muted ? 0x80 : 0 ;
/* Volume setting: bits 0-4, 0x1f = 12 dB, 0x00 = -34.5 dB
* Default setting : 0x17 = 0 dB
*/
2008-11-29 13:02:26 -03:00
wm8739_write ( sd , R0 , ( state - > vol_l & 0x1f ) | mute ) ;
wm8739_write ( sd , R1 , ( state - > vol_r & 0x1f ) | mute ) ;
2006-03-28 18:23:48 -03:00
}
2008-11-29 13:02:26 -03:00
static int wm8739_g_ctrl ( struct v4l2_subdev * sd , struct v4l2_control * ctrl )
2006-03-28 18:23:48 -03:00
{
2008-11-29 13:02:26 -03:00
struct wm8739_state * state = to_state ( sd ) ;
2006-03-28 18:23:48 -03:00
switch ( ctrl - > id ) {
case V4L2_CID_AUDIO_MUTE :
ctrl - > value = state - > muted ;
break ;
case V4L2_CID_AUDIO_VOLUME :
ctrl - > value = state - > volume ;
break ;
case V4L2_CID_AUDIO_BALANCE :
ctrl - > value = state - > balance ;
break ;
default :
return - EINVAL ;
}
return 0 ;
}
2008-11-29 13:02:26 -03:00
static int wm8739_s_ctrl ( struct v4l2_subdev * sd , struct v4l2_control * ctrl )
2006-03-28 18:23:48 -03:00
{
2008-11-29 13:02:26 -03:00
struct wm8739_state * state = to_state ( sd ) ;
2006-03-28 18:23:48 -03:00
unsigned int work_l , work_r ;
switch ( ctrl - > id ) {
case V4L2_CID_AUDIO_MUTE :
state - > muted = ctrl - > value ;
break ;
case V4L2_CID_AUDIO_VOLUME :
state - > volume = ctrl - > value ;
break ;
case V4L2_CID_AUDIO_BALANCE :
state - > balance = ctrl - > value ;
break ;
default :
return - EINVAL ;
}
/* normalize ( 65535 to 0 -> 31 to 0 (12dB to -34.5dB) ) */
work_l = ( min ( 65536 - state - > balance , 32768 ) * state - > volume ) / 32768 ;
work_r = ( min ( state - > balance , ( u16 ) 32768 ) * state - > volume ) / 32768 ;
state - > vol_l = ( long ) work_l * 31 / 65535 ;
state - > vol_r = ( long ) work_r * 31 / 65535 ;
/* set audio volume etc. */
2008-11-29 13:02:26 -03:00
wm8739_set_audio ( sd ) ;
2006-03-28 18:23:48 -03:00
return 0 ;
}
/* ------------------------------------------------------------------------ */
static struct v4l2_queryctrl wm8739_qctrl [ ] = {
{
. id = V4L2_CID_AUDIO_VOLUME ,
. name = " Volume " ,
. minimum = 0 ,
. maximum = 65535 ,
. step = 65535 / 100 ,
. default_value = 58880 ,
. flags = 0 ,
. type = V4L2_CTRL_TYPE_INTEGER ,
2007-11-01 07:45:54 -03:00
} , {
2006-03-28 18:23:48 -03:00
. id = V4L2_CID_AUDIO_MUTE ,
. name = " Mute " ,
. minimum = 0 ,
. maximum = 1 ,
. step = 1 ,
. default_value = 1 ,
. flags = 0 ,
. type = V4L2_CTRL_TYPE_BOOLEAN ,
2007-11-01 07:45:54 -03:00
} , {
2006-03-28 18:23:48 -03:00
. id = V4L2_CID_AUDIO_BALANCE ,
. name = " Balance " ,
. minimum = 0 ,
. maximum = 65535 ,
. step = 65535 / 100 ,
. default_value = 32768 ,
. flags = 0 ,
. type = V4L2_CTRL_TYPE_INTEGER ,
}
} ;
/* ------------------------------------------------------------------------ */
2008-11-29 13:02:26 -03:00
static int wm8739_s_clock_freq ( struct v4l2_subdev * sd , u32 audiofreq )
2006-03-28 18:23:48 -03:00
{
2008-11-29 13:02:26 -03:00
struct wm8739_state * state = to_state ( sd ) ;
state - > clock_freq = audiofreq ;
/* de-activate */
wm8739_write ( sd , R9 , 0x000 ) ;
switch ( audiofreq ) {
case 44100 :
/* 256fps, fs=44.1k */
wm8739_write ( sd , R8 , 0x020 ) ;
break ;
case 48000 :
/* 256fps, fs=48k */
wm8739_write ( sd , R8 , 0x000 ) ;
break ;
case 32000 :
/* 256fps, fs=32k */
wm8739_write ( sd , R8 , 0x018 ) ;
break ;
default :
2006-03-28 18:23:48 -03:00
break ;
}
2008-11-29 13:02:26 -03:00
/* activate */
wm8739_write ( sd , R9 , 0x001 ) ;
return 0 ;
}
2006-03-28 18:23:48 -03:00
2008-11-29 13:02:26 -03:00
static int wm8739_queryctrl ( struct v4l2_subdev * sd , struct v4l2_queryctrl * qc )
{
int i ;
2006-03-28 18:23:48 -03:00
2008-11-29 13:02:26 -03:00
for ( i = 0 ; i < ARRAY_SIZE ( wm8739_qctrl ) ; i + + )
if ( qc - > id & & qc - > id = = wm8739_qctrl [ i ] . id ) {
memcpy ( qc , & wm8739_qctrl [ i ] , sizeof ( * qc ) ) ;
return 0 ;
}
return - EINVAL ;
}
2006-03-28 18:23:48 -03:00
2008-12-30 07:14:19 -03:00
static int wm8739_g_chip_ident ( struct v4l2_subdev * sd , struct v4l2_dbg_chip_ident * chip )
2008-11-29 13:02:26 -03:00
{
struct i2c_client * client = v4l2_get_subdevdata ( sd ) ;
2007-04-27 12:31:26 -03:00
2008-11-29 13:02:26 -03:00
return v4l2_chip_ident_i2c_client ( client , chip , V4L2_IDENT_WM8739 , 0 ) ;
}
2006-03-28 18:23:48 -03:00
2008-11-29 13:02:26 -03:00
static int wm8739_log_status ( struct v4l2_subdev * sd )
{
struct wm8739_state * state = to_state ( sd ) ;
2006-03-28 18:23:48 -03:00
2008-11-29 13:02:26 -03:00
v4l2_info ( sd , " Frequency: %u Hz \n " , state - > clock_freq ) ;
v4l2_info ( sd , " Volume L: %02x%s \n " , state - > vol_l & 0x1f ,
state - > muted ? " (muted) " : " " ) ;
v4l2_info ( sd , " Volume R: %02x%s \n " , state - > vol_r & 0x1f ,
state - > muted ? " (muted) " : " " ) ;
2006-03-28 18:23:48 -03:00
return 0 ;
}
2008-11-29 13:02:26 -03:00
/* ----------------------------------------------------------------------- */
static const struct v4l2_subdev_core_ops wm8739_core_ops = {
. log_status = wm8739_log_status ,
. g_chip_ident = wm8739_g_chip_ident ,
. queryctrl = wm8739_queryctrl ,
. g_ctrl = wm8739_g_ctrl ,
. s_ctrl = wm8739_s_ctrl ,
} ;
static const struct v4l2_subdev_audio_ops wm8739_audio_ops = {
. s_clock_freq = wm8739_s_clock_freq ,
} ;
static const struct v4l2_subdev_ops wm8739_ops = {
. core = & wm8739_core_ops ,
. audio = & wm8739_audio_ops ,
} ;
2006-03-28 18:23:48 -03:00
/* ------------------------------------------------------------------------ */
/* i2c implementation */
2008-04-29 23:11:39 +02:00
static int wm8739_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
2006-03-28 18:23:48 -03:00
{
struct wm8739_state * state ;
2008-11-29 13:02:26 -03:00
struct v4l2_subdev * sd ;
2006-03-28 18:23:48 -03:00
2007-09-16 10:47:15 -03:00
/* Check if the adapter supports the needed features */
if ( ! i2c_check_functionality ( client - > adapter , I2C_FUNC_SMBUS_BYTE_DATA ) )
return - EIO ;
2007-11-01 07:45:54 -03:00
v4l_info ( client , " chip found @ 0x%x (%s) \n " ,
client - > addr < < 1 , client - > adapter - > name ) ;
2006-03-28 18:23:48 -03:00
state = kmalloc ( sizeof ( struct wm8739_state ) , GFP_KERNEL ) ;
2008-08-24 06:29:30 -03:00
if ( state = = NULL )
2006-03-28 18:23:48 -03:00
return - ENOMEM ;
2008-11-29 13:02:26 -03:00
sd = & state - > sd ;
v4l2_i2c_subdev_init ( sd , client , & wm8739_ops ) ;
2006-03-28 18:23:48 -03:00
state - > vol_l = 0x17 ; /* 0dB */
state - > vol_r = 0x17 ; /* 0dB */
state - > muted = 0 ;
state - > balance = 32768 ;
/* normalize (12dB(31) to -34.5dB(0) [0dB(23)] -> 65535 to 0) */
state - > volume = ( ( long ) state - > vol_l + 1 ) * 65535 / 31 ;
state - > clock_freq = 48000 ;
2007-11-01 07:45:54 -03:00
/* Initialize wm8739 */
/* reset */
2008-11-29 13:02:26 -03:00
wm8739_write ( sd , R15 , 0x00 ) ;
2007-11-01 07:45:54 -03:00
/* filter setting, high path, offet clear */
2008-11-29 13:02:26 -03:00
wm8739_write ( sd , R5 , 0x000 ) ;
2007-11-01 07:45:54 -03:00
/* ADC, OSC, Power Off mode Disable */
2008-11-29 13:02:26 -03:00
wm8739_write ( sd , R6 , 0x000 ) ;
2007-11-01 07:45:54 -03:00
/* Digital Audio interface format:
Enable Master mode , 24 bit , MSB first / left justified */
2008-11-29 13:02:26 -03:00
wm8739_write ( sd , R7 , 0x049 ) ;
2007-11-01 07:45:54 -03:00
/* sampling control: normal, 256fs, 48KHz sampling rate */
2008-11-29 13:02:26 -03:00
wm8739_write ( sd , R8 , 0x000 ) ;
2007-11-01 07:45:54 -03:00
/* activate */
2008-11-29 13:02:26 -03:00
wm8739_write ( sd , R9 , 0x001 ) ;
2007-11-01 07:45:54 -03:00
/* set volume/mute */
2008-11-29 13:02:26 -03:00
wm8739_set_audio ( sd ) ;
2006-03-28 18:23:48 -03:00
return 0 ;
}
2007-09-13 11:10:07 -03:00
static int wm8739_remove ( struct i2c_client * client )
2006-03-28 18:23:48 -03:00
{
2008-11-29 13:02:26 -03:00
struct v4l2_subdev * sd = i2c_get_clientdata ( client ) ;
v4l2_device_unregister_subdev ( sd ) ;
kfree ( to_state ( sd ) ) ;
2006-03-28 18:23:48 -03:00
return 0 ;
}
2008-05-18 20:49:40 +02:00
static const struct i2c_device_id wm8739_id [ ] = {
{ " wm8739 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , wm8739_id ) ;
2007-09-13 11:10:07 -03:00
static struct v4l2_i2c_driver_data v4l2_i2c_data = {
. name = " wm8739 " ,
. probe = wm8739_probe ,
. remove = wm8739_remove ,
2008-05-18 20:49:40 +02:00
. id_table = wm8739_id ,
2006-03-28 18:23:48 -03:00
} ;