2010-01-06 17:04:18 -05:00
/*
2010-01-08 06:01:24 -05:00
* I2C driver for Marvell 88 PM860x
2010-01-06 17:04:18 -05:00
*
* Copyright ( C ) 2009 Marvell International Ltd .
2014-05-08 12:55:35 +01:00
*
* Author : Haojian Zhuang < haojian . zhuang @ marvell . com >
2010-01-06 17:04:18 -05:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/i2c.h>
2011-11-11 15:38:27 +08:00
# include <linux/regmap.h>
2010-01-08 06:01:24 -05:00
# include <linux/mfd/88pm860x.h>
2010-01-06 17:04:18 -05:00
2010-01-08 06:01:24 -05:00
int pm860x_reg_read ( struct i2c_client * i2c , int reg )
2010-01-06 17:04:18 -05:00
{
2010-01-08 06:01:24 -05:00
struct pm860x_chip * chip = i2c_get_clientdata ( i2c ) ;
2011-11-11 15:38:27 +08:00
struct regmap * map = ( i2c = = chip - > client ) ? chip - > regmap
: chip - > regmap_companion ;
unsigned int data ;
2010-01-06 17:04:18 -05:00
int ret ;
2011-11-11 15:38:27 +08:00
ret = regmap_read ( map , reg , & data ) ;
2010-01-06 17:04:18 -05:00
if ( ret < 0 )
return ret ;
else
return ( int ) data ;
}
2010-01-08 06:01:24 -05:00
EXPORT_SYMBOL ( pm860x_reg_read ) ;
2010-01-06 17:04:18 -05:00
2010-01-08 06:01:24 -05:00
int pm860x_reg_write ( struct i2c_client * i2c , int reg ,
2010-01-06 17:04:18 -05:00
unsigned char data )
{
2010-01-08 06:01:24 -05:00
struct pm860x_chip * chip = i2c_get_clientdata ( i2c ) ;
2011-11-11 15:38:27 +08:00
struct regmap * map = ( i2c = = chip - > client ) ? chip - > regmap
: chip - > regmap_companion ;
2010-01-06 17:04:18 -05:00
int ret ;
2011-11-11 15:38:27 +08:00
ret = regmap_write ( map , reg , data ) ;
2010-01-06 17:04:18 -05:00
return ret ;
}
2010-01-08 06:01:24 -05:00
EXPORT_SYMBOL ( pm860x_reg_write ) ;
2010-01-06 17:04:18 -05:00
2010-01-08 06:01:24 -05:00
int pm860x_bulk_read ( struct i2c_client * i2c , int reg ,
2010-01-06 17:04:18 -05:00
int count , unsigned char * buf )
{
2010-01-08 06:01:24 -05:00
struct pm860x_chip * chip = i2c_get_clientdata ( i2c ) ;
2011-11-11 15:38:27 +08:00
struct regmap * map = ( i2c = = chip - > client ) ? chip - > regmap
: chip - > regmap_companion ;
2010-01-06 17:04:18 -05:00
int ret ;
2011-11-11 15:38:27 +08:00
ret = regmap_raw_read ( map , reg , buf , count ) ;
2010-01-06 17:04:18 -05:00
return ret ;
}
2010-01-08 06:01:24 -05:00
EXPORT_SYMBOL ( pm860x_bulk_read ) ;
2010-01-06 17:04:18 -05:00
2010-01-08 06:01:24 -05:00
int pm860x_bulk_write ( struct i2c_client * i2c , int reg ,
2010-01-06 17:04:18 -05:00
int count , unsigned char * buf )
{
2010-01-08 06:01:24 -05:00
struct pm860x_chip * chip = i2c_get_clientdata ( i2c ) ;
2011-11-11 15:38:27 +08:00
struct regmap * map = ( i2c = = chip - > client ) ? chip - > regmap
: chip - > regmap_companion ;
2010-01-06 17:04:18 -05:00
int ret ;
2011-11-11 15:38:27 +08:00
ret = regmap_raw_write ( map , reg , buf , count ) ;
2010-01-06 17:04:18 -05:00
return ret ;
}
2010-01-08 06:01:24 -05:00
EXPORT_SYMBOL ( pm860x_bulk_write ) ;
2010-01-06 17:04:18 -05:00
2010-01-08 06:01:24 -05:00
int pm860x_set_bits ( struct i2c_client * i2c , int reg ,
2010-01-06 17:04:18 -05:00
unsigned char mask , unsigned char data )
{
2010-01-08 06:01:24 -05:00
struct pm860x_chip * chip = i2c_get_clientdata ( i2c ) ;
2011-11-11 15:38:27 +08:00
struct regmap * map = ( i2c = = chip - > client ) ? chip - > regmap
: chip - > regmap_companion ;
2010-01-06 17:04:18 -05:00
int ret ;
2011-11-11 15:38:27 +08:00
ret = regmap_update_bits ( map , reg , mask , data ) ;
2010-01-06 17:04:18 -05:00
return ret ;
}
2010-01-08 06:01:24 -05:00
EXPORT_SYMBOL ( pm860x_set_bits ) ;
2010-01-06 17:04:18 -05:00
2011-11-11 15:38:26 +08:00
static int read_device ( struct i2c_client * i2c , int reg ,
int bytes , void * dest )
{
unsigned char msgbuf0 [ I2C_SMBUS_BLOCK_MAX + 3 ] ;
unsigned char msgbuf1 [ I2C_SMBUS_BLOCK_MAX + 2 ] ;
struct i2c_adapter * adap = i2c - > adapter ;
2012-09-17 21:14:58 +05:30
struct i2c_msg msg [ 2 ] = {
{
. addr = i2c - > addr ,
. flags = 0 ,
. len = 1 ,
. buf = msgbuf0
} ,
{ . addr = i2c - > addr ,
. flags = I2C_M_RD ,
. len = 0 ,
. buf = msgbuf1
} ,
2011-11-11 15:38:26 +08:00
} ;
int num = 1 , ret = 0 ;
if ( dest = = NULL )
return - EINVAL ;
msgbuf0 [ 0 ] = ( unsigned char ) reg ; /* command */
msg [ 1 ] . len = bytes ;
/* if data needs to read back, num should be 2 */
if ( bytes > 0 )
num = 2 ;
ret = adap - > algo - > master_xfer ( adap , msg , num ) ;
memcpy ( dest , msgbuf1 , bytes ) ;
if ( ret < 0 )
return ret ;
return 0 ;
}
static int write_device ( struct i2c_client * i2c , int reg ,
int bytes , void * src )
{
2014-08-28 10:51:26 +01:00
unsigned char buf [ 2 ] ;
2011-11-11 15:38:26 +08:00
struct i2c_adapter * adap = i2c - > adapter ;
struct i2c_msg msg ;
int ret ;
buf [ 0 ] = ( unsigned char ) reg ;
memcpy ( & buf [ 1 ] , src , bytes ) ;
msg . addr = i2c - > addr ;
msg . flags = 0 ;
msg . len = bytes + 1 ;
msg . buf = buf ;
ret = adap - > algo - > master_xfer ( adap , & msg , 1 ) ;
if ( ret < 0 )
return ret ;
return 0 ;
}
2011-03-07 23:43:16 +08:00
int pm860x_page_reg_write ( struct i2c_client * i2c , int reg ,
unsigned char data )
{
unsigned char zero ;
int ret ;
2018-06-20 07:18:02 +02:00
i2c_lock_bus ( i2c - > adapter , I2C_LOCK_SEGMENT ) ;
2011-11-11 15:38:26 +08:00
read_device ( i2c , 0xFA , 0 , & zero ) ;
read_device ( i2c , 0xFB , 0 , & zero ) ;
read_device ( i2c , 0xFF , 0 , & zero ) ;
ret = write_device ( i2c , reg , 1 , & data ) ;
read_device ( i2c , 0xFE , 0 , & zero ) ;
read_device ( i2c , 0xFC , 0 , & zero ) ;
2018-06-20 07:18:02 +02:00
i2c_unlock_bus ( i2c - > adapter , I2C_LOCK_SEGMENT ) ;
2011-03-07 23:43:16 +08:00
return ret ;
}
EXPORT_SYMBOL ( pm860x_page_reg_write ) ;
int pm860x_page_bulk_read ( struct i2c_client * i2c , int reg ,
int count , unsigned char * buf )
{
unsigned char zero = 0 ;
int ret ;
2018-06-20 07:18:02 +02:00
i2c_lock_bus ( i2c - > adapter , I2C_LOCK_SEGMENT ) ;
2011-11-11 15:38:26 +08:00
read_device ( i2c , 0xfa , 0 , & zero ) ;
read_device ( i2c , 0xfb , 0 , & zero ) ;
read_device ( i2c , 0xff , 0 , & zero ) ;
ret = read_device ( i2c , reg , count , buf ) ;
read_device ( i2c , 0xFE , 0 , & zero ) ;
read_device ( i2c , 0xFC , 0 , & zero ) ;
2018-06-20 07:18:02 +02:00
i2c_unlock_bus ( i2c - > adapter , I2C_LOCK_SEGMENT ) ;
2011-03-07 23:43:16 +08:00
return ret ;
}
EXPORT_SYMBOL ( pm860x_page_bulk_read ) ;