2019-05-27 08:55:06 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2008-04-18 21:34:00 -03:00
/*
* Driver for the Auvitek AU0828 USB bridge
*
2008-09-03 17:12:12 -03:00
* Copyright ( c ) 2008 Steven Toth < stoth @ linuxtv . org >
2008-04-18 21:34:00 -03:00
*/
2014-08-09 21:47:17 -03:00
# include "au0828.h"
2008-04-18 21:34:00 -03:00
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/delay.h>
2008-04-18 22:12:52 -03:00
# include <linux/io.h>
2008-04-18 21:34:00 -03:00
2012-08-06 22:47:04 -03:00
# include "media/tuner.h"
2008-04-18 21:34:00 -03:00
# include <media/v4l2-common.h>
2008-04-25 19:06:03 -03:00
static int i2c_scan ;
2008-04-18 21:34:00 -03:00
module_param ( i2c_scan , int , 0444 ) ;
MODULE_PARM_DESC ( i2c_scan , " scan i2c bus at insmod time " ) ;
2012-08-06 22:47:06 -03:00
# define I2C_WAIT_DELAY 25
# define I2C_WAIT_RETRY 1000
2008-04-18 21:34:00 -03:00
static inline int i2c_slave_did_write_ack ( struct i2c_adapter * i2c_adap )
{
struct au0828_dev * dev = i2c_adap - > algo_data ;
2009-03-31 23:58:49 -03:00
return au0828_read ( dev , AU0828_I2C_STATUS_201 ) &
AU0828_I2C_STATUS_NO_WRITE_ACK ? 0 : 1 ;
2008-04-18 21:34:00 -03:00
}
static inline int i2c_slave_did_read_ack ( struct i2c_adapter * i2c_adap )
{
struct au0828_dev * dev = i2c_adap - > algo_data ;
2009-03-31 23:58:49 -03:00
return au0828_read ( dev , AU0828_I2C_STATUS_201 ) &
AU0828_I2C_STATUS_NO_READ_ACK ? 0 : 1 ;
2008-04-18 21:34:00 -03:00
}
static int i2c_wait_read_ack ( struct i2c_adapter * i2c_adap )
{
int count ;
for ( count = 0 ; count < I2C_WAIT_RETRY ; count + + ) {
if ( ! i2c_slave_did_read_ack ( i2c_adap ) )
break ;
udelay ( I2C_WAIT_DELAY ) ;
}
if ( I2C_WAIT_RETRY = = count )
return 0 ;
return 1 ;
}
static inline int i2c_is_read_busy ( struct i2c_adapter * i2c_adap )
{
struct au0828_dev * dev = i2c_adap - > algo_data ;
2009-03-31 23:58:49 -03:00
return au0828_read ( dev , AU0828_I2C_STATUS_201 ) &
AU0828_I2C_STATUS_READ_DONE ? 0 : 1 ;
2008-04-18 21:34:00 -03:00
}
static int i2c_wait_read_done ( struct i2c_adapter * i2c_adap )
{
int count ;
for ( count = 0 ; count < I2C_WAIT_RETRY ; count + + ) {
if ( ! i2c_is_read_busy ( i2c_adap ) )
break ;
udelay ( I2C_WAIT_DELAY ) ;
}
if ( I2C_WAIT_RETRY = = count )
return 0 ;
return 1 ;
}
static inline int i2c_is_write_done ( struct i2c_adapter * i2c_adap )
{
struct au0828_dev * dev = i2c_adap - > algo_data ;
2009-03-31 23:58:49 -03:00
return au0828_read ( dev , AU0828_I2C_STATUS_201 ) &
AU0828_I2C_STATUS_WRITE_DONE ? 1 : 0 ;
2008-04-18 21:34:00 -03:00
}
static int i2c_wait_write_done ( struct i2c_adapter * i2c_adap )
{
int count ;
for ( count = 0 ; count < I2C_WAIT_RETRY ; count + + ) {
if ( i2c_is_write_done ( i2c_adap ) )
break ;
udelay ( I2C_WAIT_DELAY ) ;
}
if ( I2C_WAIT_RETRY = = count )
return 0 ;
return 1 ;
}
static inline int i2c_is_busy ( struct i2c_adapter * i2c_adap )
{
struct au0828_dev * dev = i2c_adap - > algo_data ;
2009-03-31 23:58:49 -03:00
return au0828_read ( dev , AU0828_I2C_STATUS_201 ) &
AU0828_I2C_STATUS_BUSY ? 1 : 0 ;
2008-04-18 21:34:00 -03:00
}
static int i2c_wait_done ( struct i2c_adapter * i2c_adap )
{
int count ;
for ( count = 0 ; count < I2C_WAIT_RETRY ; count + + ) {
if ( ! i2c_is_busy ( i2c_adap ) )
break ;
udelay ( I2C_WAIT_DELAY ) ;
}
if ( I2C_WAIT_RETRY = = count )
return 0 ;
return 1 ;
}
/* FIXME: Implement join handling correctly */
static int i2c_sendbytes ( struct i2c_adapter * i2c_adap ,
const struct i2c_msg * msg , int joined_rlen )
{
int i , strobe = 0 ;
struct au0828_dev * dev = i2c_adap - > algo_data ;
2014-07-25 14:02:14 -03:00
u8 i2c_speed = dev - > board . i2c_clk_divider ;
2008-04-18 21:34:00 -03:00
2008-04-18 21:42:30 -03:00
dprintk ( 4 , " %s() \n " , __func__ ) ;
2008-04-18 21:34:00 -03:00
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_MULTIBYTE_MODE_2FF , 0x01 ) ;
2009-03-11 03:00:41 -03:00
2012-08-06 22:47:13 -03:00
if ( ( ( dev - > board . tuner_type = = TUNER_XC5000 ) | |
( dev - > board . tuner_type = = TUNER_XC5000C ) ) & &
2014-07-25 14:02:14 -03:00
( dev - > board . tuner_addr = = msg - > addr ) ) {
/*
* Due to I2C clock stretch , we need to use a lower speed
* on xc5000 for commands . However , firmware transfer can
* speed up to 400 KHz .
*/
if ( msg - > len = = 64 )
i2c_speed = AU0828_I2C_CLK_250KHZ ;
else
i2c_speed = AU0828_I2C_CLK_20KHZ ;
2012-08-06 22:47:04 -03:00
}
2014-07-25 14:02:14 -03:00
/* Set the I2C clock */
au0828_write ( dev , AU0828_I2C_CLK_DIVIDER_202 , i2c_speed ) ;
2008-04-18 21:34:00 -03:00
/* Hardware needs 8 bit addresses */
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_DEST_ADDR_203 , msg - > addr < < 1 ) ;
2008-04-18 21:34:00 -03:00
2008-04-18 21:39:11 -03:00
dprintk ( 4 , " SEND: %02x \n " , msg - > addr ) ;
2008-04-18 21:34:00 -03:00
2009-03-11 03:00:56 -03:00
/* Deal with i2c_scan */
if ( msg - > len = = 0 ) {
/* The analog tuner detection code makes use of the SMBUS_QUICK
message ( which involves a zero length i2c write ) . To avoid
checking the status register when we didn ' t strobe out any
actual bytes to the bus , just do a read check . This is
consistent with how I saw i2c device checking done in the
USB trace of the Windows driver */
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_TRIGGER_200 ,
AU0828_I2C_TRIGGER_READ ) ;
2009-03-11 03:00:56 -03:00
if ( ! i2c_wait_done ( i2c_adap ) )
return - EIO ;
if ( i2c_wait_read_ack ( i2c_adap ) )
return - EIO ;
return 0 ;
}
2008-04-17 21:41:28 -03:00
for ( i = 0 ; i < msg - > len ; ) {
2008-04-18 21:34:00 -03:00
2008-04-18 21:39:11 -03:00
dprintk ( 4 , " %02x \n " , msg - > buf [ i ] ) ;
2008-04-18 21:34:00 -03:00
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_WRITE_FIFO_205 , msg - > buf [ i ] ) ;
2008-04-18 21:34:00 -03:00
strobe + + ;
i + + ;
if ( ( strobe > = 4 ) | | ( i > = msg - > len ) ) {
/* Strobe the byte into the bus */
if ( i < msg - > len )
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_TRIGGER_200 ,
AU0828_I2C_TRIGGER_WRITE |
AU0828_I2C_TRIGGER_HOLD ) ;
2008-04-18 21:34:00 -03:00
else
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_TRIGGER_200 ,
AU0828_I2C_TRIGGER_WRITE ) ;
2008-04-18 21:34:00 -03:00
/* Reset strobe trigger */
strobe = 0 ;
if ( ! i2c_wait_write_done ( i2c_adap ) )
return - EIO ;
}
}
if ( ! i2c_wait_done ( i2c_adap ) )
return - EIO ;
2008-04-18 21:39:11 -03:00
dprintk ( 4 , " \n " ) ;
2008-04-18 21:34:00 -03:00
return msg - > len ;
}
/* FIXME: Implement join handling correctly */
static int i2c_readbytes ( struct i2c_adapter * i2c_adap ,
const struct i2c_msg * msg , int joined )
{
struct au0828_dev * dev = i2c_adap - > algo_data ;
2014-07-25 14:02:14 -03:00
u8 i2c_speed = dev - > board . i2c_clk_divider ;
2008-04-18 21:34:00 -03:00
int i ;
2008-04-18 21:42:30 -03:00
dprintk ( 4 , " %s() \n " , __func__ ) ;
2008-04-18 21:34:00 -03:00
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_MULTIBYTE_MODE_2FF , 0x01 ) ;
2009-03-11 03:00:41 -03:00
2014-07-25 14:02:14 -03:00
/*
* Due to xc5000c clock stretch , we cannot use full speed at
* readings from xc5000 , as otherwise they ' ll fail .
*/
if ( ( ( dev - > board . tuner_type = = TUNER_XC5000 ) | |
( dev - > board . tuner_type = = TUNER_XC5000C ) ) & &
( dev - > board . tuner_addr = = msg - > addr ) )
i2c_speed = AU0828_I2C_CLK_20KHZ ;
2009-04-01 00:11:31 -03:00
/* Set the I2C clock */
2014-07-25 14:02:14 -03:00
au0828_write ( dev , AU0828_I2C_CLK_DIVIDER_202 , i2c_speed ) ;
2008-04-18 21:34:00 -03:00
/* Hardware needs 8 bit addresses */
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_DEST_ADDR_203 , msg - > addr < < 1 ) ;
2008-04-18 21:34:00 -03:00
2008-04-18 21:39:11 -03:00
dprintk ( 4 , " RECV: \n " ) ;
2008-04-18 21:34:00 -03:00
/* Deal with i2c_scan */
if ( msg - > len = = 0 ) {
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_TRIGGER_200 ,
AU0828_I2C_TRIGGER_READ ) ;
2008-04-18 21:34:00 -03:00
if ( i2c_wait_read_ack ( i2c_adap ) )
return - EIO ;
return 0 ;
}
2008-04-17 21:41:28 -03:00
for ( i = 0 ; i < msg - > len ; ) {
2008-04-18 21:34:00 -03:00
i + + ;
if ( i < msg - > len )
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_TRIGGER_200 ,
AU0828_I2C_TRIGGER_READ |
AU0828_I2C_TRIGGER_HOLD ) ;
2008-04-18 21:34:00 -03:00
else
2009-03-31 23:58:49 -03:00
au0828_write ( dev , AU0828_I2C_TRIGGER_200 ,
AU0828_I2C_TRIGGER_READ ) ;
2008-04-18 21:34:00 -03:00
if ( ! i2c_wait_read_done ( i2c_adap ) )
return - EIO ;
2009-03-31 23:58:49 -03:00
msg - > buf [ i - 1 ] = au0828_read ( dev , AU0828_I2C_READ_FIFO_209 ) &
0xff ;
2008-04-18 21:34:00 -03:00
2008-04-18 21:39:11 -03:00
dprintk ( 4 , " %02x \n " , msg - > buf [ i - 1 ] ) ;
2008-04-18 21:34:00 -03:00
}
if ( ! i2c_wait_done ( i2c_adap ) )
return - EIO ;
2008-04-18 21:39:11 -03:00
dprintk ( 4 , " \n " ) ;
2008-04-18 21:34:00 -03:00
return msg - > len ;
}
static int i2c_xfer ( struct i2c_adapter * i2c_adap ,
struct i2c_msg * msgs , int num )
{
int i , retval = 0 ;
2008-04-18 21:42:30 -03:00
dprintk ( 4 , " %s(num = %d) \n " , __func__ , num ) ;
2008-04-18 21:34:00 -03:00
2008-04-17 21:41:28 -03:00
for ( i = 0 ; i < num ; i + + ) {
2008-04-18 21:39:11 -03:00
dprintk ( 4 , " %s(num = %d) addr = 0x%02x len = 0x%x \n " ,
2008-04-18 21:42:30 -03:00
__func__ , num , msgs [ i ] . addr , msgs [ i ] . len ) ;
2008-04-18 21:34:00 -03:00
if ( msgs [ i ] . flags & I2C_M_RD ) {
/* read */
retval = i2c_readbytes ( i2c_adap , & msgs [ i ] , 0 ) ;
} else if ( i + 1 < num & & ( msgs [ i + 1 ] . flags & I2C_M_RD ) & &
msgs [ i ] . addr = = msgs [ i + 1 ] . addr ) {
/* write then read from same address */
retval = i2c_sendbytes ( i2c_adap , & msgs [ i ] ,
msgs [ i + 1 ] . len ) ;
if ( retval < 0 )
goto err ;
i + + ;
retval = i2c_readbytes ( i2c_adap , & msgs [ i ] , 1 ) ;
} else {
/* write */
retval = i2c_sendbytes ( i2c_adap , & msgs [ i ] , 0 ) ;
}
if ( retval < 0 )
goto err ;
}
return num ;
err :
return retval ;
}
static u32 au0828_functionality ( struct i2c_adapter * adap )
{
return I2C_FUNC_SMBUS_EMUL | I2C_FUNC_I2C ;
}
2017-08-18 12:06:58 -04:00
static const struct i2c_algorithm au0828_i2c_algo_template = {
2008-04-18 21:34:00 -03:00
. master_xfer = i2c_xfer ,
. functionality = au0828_functionality ,
} ;
/* ----------------------------------------------------------------------- */
2017-08-19 06:34:15 -04:00
static const struct i2c_adapter au0828_i2c_adap_template = {
2014-08-09 21:47:17 -03:00
. name = KBUILD_MODNAME ,
2008-04-18 21:34:00 -03:00
. owner = THIS_MODULE ,
. algo = & au0828_i2c_algo_template ,
} ;
2017-08-28 03:21:35 -04:00
static const struct i2c_client au0828_i2c_client_template = {
2008-04-18 21:34:00 -03:00
. name = " au0828 internal " ,
} ;
static char * i2c_devs [ 128 ] = {
2008-04-18 22:12:52 -03:00
[ 0x8e > > 1 ] = " au8522 " ,
[ 0xa0 > > 1 ] = " eeprom " ,
[ 0xc2 > > 1 ] = " tuner/xc5000 " ,
2008-04-18 21:34:00 -03:00
} ;
static void do_i2c_scan ( char * name , struct i2c_client * c )
{
unsigned char buf ;
int i , rc ;
for ( i = 0 ; i < 128 ; i + + ) {
c - > addr = i ;
rc = i2c_master_recv ( c , & buf , 0 ) ;
if ( rc < 0 )
continue ;
2014-08-09 21:47:17 -03:00
pr_info ( " %s: i2c scan: found device @ 0x%x [%s] \n " ,
2008-04-18 21:34:00 -03:00
name , i < < 1 , i2c_devs [ i ] ? i2c_devs [ i ] : " ??? " ) ;
}
}
2011-11-09 13:27:53 -03:00
/* init + register i2c adapter */
2008-04-18 21:34:00 -03:00
int au0828_i2c_register ( struct au0828_dev * dev )
{
2008-04-18 21:42:30 -03:00
dprintk ( 1 , " %s() \n " , __func__ ) ;
2008-04-18 21:34:00 -03:00
2012-10-23 15:57:14 -03:00
dev - > i2c_adap = au0828_i2c_adap_template ;
dev - > i2c_algo = au0828_i2c_algo_template ;
dev - > i2c_client = au0828_i2c_client_template ;
2008-04-18 21:34:00 -03:00
dev - > i2c_adap . dev . parent = & dev - > usbdev - > dev ;
2018-09-10 08:19:14 -04:00
strscpy ( dev - > i2c_adap . name , KBUILD_MODNAME ,
2008-04-18 21:34:00 -03:00
sizeof ( dev - > i2c_adap . name ) ) ;
2009-03-11 03:01:04 -03:00
dev - > i2c_adap . algo = & dev - > i2c_algo ;
2008-04-18 21:34:00 -03:00
dev - > i2c_adap . algo_data = dev ;
2012-12-04 11:30:00 -03:00
# ifdef CONFIG_VIDEO_AU0828_V4L2
2009-03-11 03:01:04 -03:00
i2c_set_adapdata ( & dev - > i2c_adap , & dev - > v4l2_dev ) ;
2012-12-04 11:30:00 -03:00
# else
i2c_set_adapdata ( & dev - > i2c_adap , dev ) ;
# endif
2008-04-18 21:34:00 -03:00
i2c_add_adapter ( & dev - > i2c_adap ) ;
dev - > i2c_client . adapter = & dev - > i2c_adap ;
if ( 0 = = dev - > i2c_rc ) {
2014-08-09 21:47:17 -03:00
pr_info ( " i2c bus registered \n " ) ;
2008-04-18 21:34:00 -03:00
if ( i2c_scan )
2014-08-09 21:47:17 -03:00
do_i2c_scan ( KBUILD_MODNAME , & dev - > i2c_client ) ;
2008-04-18 21:34:00 -03:00
} else
2014-08-09 21:47:17 -03:00
pr_info ( " i2c bus register FAILED \n " ) ;
2008-04-18 21:39:11 -03:00
2008-04-18 21:34:00 -03:00
return dev - > i2c_rc ;
}
int au0828_i2c_unregister ( struct au0828_dev * dev )
{
i2c_del_adapter ( & dev - > i2c_adap ) ;
return 0 ;
}