2013-10-28 22:12:35 +04:00
/*
* Register map access API - SPMI support
*
* Copyright ( c ) 2012 - 2013 , The Linux Foundation . All rights reserved .
*
* Based on regmap - i2c . c :
* Copyright 2011 Wolfson Microelectronics plc
* Author : Mark Brown < broonie @ opensource . wolfsonmicro . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation .
*
* 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 .
*
*/
# include <linux/regmap.h>
# include <linux/spmi.h>
# include <linux/module.h>
# include <linux/init.h>
2014-02-12 23:44:27 +04:00
static int regmap_spmi_base_read ( void * context ,
const void * reg , size_t reg_size ,
void * val , size_t val_size )
2013-10-28 22:12:35 +04:00
{
2014-02-12 23:44:27 +04:00
u8 addr = * ( u8 * ) reg ;
int err = 0 ;
BUG_ON ( reg_size ! = 1 ) ;
while ( val_size - - & & ! err )
err = spmi_register_read ( context , addr + + , val + + ) ;
return err ;
}
static int regmap_spmi_base_gather_write ( void * context ,
const void * reg , size_t reg_size ,
const void * val , size_t val_size )
{
const u8 * data = val ;
u8 addr = * ( u8 * ) reg ;
int err = 0 ;
BUG_ON ( reg_size ! = 1 ) ;
/*
* SPMI defines a more bandwidth - efficient ' Register 0 Write ' sequence ,
* use it when possible .
*/
if ( addr = = 0 & & val_size ) {
err = spmi_register_zero_write ( context , * data ) ;
if ( err )
goto err_out ;
data + + ;
addr + + ;
val_size - - ;
}
while ( val_size ) {
err = spmi_register_write ( context , addr , * data ) ;
if ( err )
goto err_out ;
data + + ;
addr + + ;
val_size - - ;
}
err_out :
return err ;
}
static int regmap_spmi_base_write ( void * context , const void * data ,
size_t count )
{
BUG_ON ( count < 1 ) ;
return regmap_spmi_base_gather_write ( context , data , 1 , data + 1 ,
count - 1 ) ;
}
2017-08-11 09:57:36 +03:00
static const struct regmap_bus regmap_spmi_base = {
2014-02-12 23:44:27 +04:00
. read = regmap_spmi_base_read ,
. write = regmap_spmi_base_write ,
. gather_write = regmap_spmi_base_gather_write ,
. reg_format_endian_default = REGMAP_ENDIAN_NATIVE ,
. val_format_endian_default = REGMAP_ENDIAN_NATIVE ,
} ;
2015-07-08 09:30:18 +03:00
struct regmap * __regmap_init_spmi_base ( struct spmi_device * sdev ,
const struct regmap_config * config ,
struct lock_class_key * lock_key ,
const char * lock_name )
2014-02-12 23:44:27 +04:00
{
2015-07-08 09:30:18 +03:00
return __regmap_init ( & sdev - > dev , & regmap_spmi_base , sdev , config ,
lock_key , lock_name ) ;
2014-02-12 23:44:27 +04:00
}
2015-07-08 09:30:18 +03:00
EXPORT_SYMBOL_GPL ( __regmap_init_spmi_base ) ;
2014-02-12 23:44:27 +04:00
2015-07-08 09:30:18 +03:00
struct regmap * __devm_regmap_init_spmi_base ( struct spmi_device * sdev ,
const struct regmap_config * config ,
struct lock_class_key * lock_key ,
const char * lock_name )
2014-02-12 23:44:27 +04:00
{
2015-07-08 09:30:18 +03:00
return __devm_regmap_init ( & sdev - > dev , & regmap_spmi_base , sdev , config ,
lock_key , lock_name ) ;
2014-02-12 23:44:27 +04:00
}
2015-07-08 09:30:18 +03:00
EXPORT_SYMBOL_GPL ( __devm_regmap_init_spmi_base ) ;
2014-02-12 23:44:27 +04:00
static int regmap_spmi_ext_read ( void * context ,
const void * reg , size_t reg_size ,
void * val , size_t val_size )
{
int err = 0 ;
size_t len ;
u16 addr ;
2013-10-28 22:12:35 +04:00
BUG_ON ( reg_size ! = 2 ) ;
2014-02-12 23:44:27 +04:00
addr = * ( u16 * ) reg ;
/*
* Split accesses into two to take advantage of the more
* bandwidth - efficient ' Extended Register Read ' command when possible
*/
while ( addr < = 0xFF & & val_size ) {
len = min_t ( size_t , val_size , 16 ) ;
err = spmi_ext_register_read ( context , addr , val , len ) ;
if ( err )
goto err_out ;
addr + = len ;
val + = len ;
val_size - = len ;
}
while ( val_size ) {
len = min_t ( size_t , val_size , 8 ) ;
2016-04-15 09:37:26 +03:00
err = spmi_ext_register_readl ( context , addr , val , len ) ;
2014-02-12 23:44:27 +04:00
if ( err )
goto err_out ;
addr + = len ;
val + = len ;
val_size - = len ;
}
err_out :
return err ;
2013-10-28 22:12:35 +04:00
}
2014-02-12 23:44:27 +04:00
static int regmap_spmi_ext_gather_write ( void * context ,
const void * reg , size_t reg_size ,
const void * val , size_t val_size )
2013-10-28 22:12:35 +04:00
{
2014-02-12 23:44:27 +04:00
int err = 0 ;
size_t len ;
u16 addr ;
2013-10-28 22:12:35 +04:00
BUG_ON ( reg_size ! = 2 ) ;
2014-02-12 23:44:27 +04:00
addr = * ( u16 * ) reg ;
while ( addr < = 0xFF & & val_size ) {
len = min_t ( size_t , val_size , 16 ) ;
err = spmi_ext_register_write ( context , addr , val , len ) ;
if ( err )
goto err_out ;
addr + = len ;
val + = len ;
val_size - = len ;
}
while ( val_size ) {
len = min_t ( size_t , val_size , 8 ) ;
err = spmi_ext_register_writel ( context , addr , val , len ) ;
if ( err )
goto err_out ;
addr + = len ;
val + = len ;
val_size - = len ;
}
err_out :
return err ;
2013-10-28 22:12:35 +04:00
}
2014-02-12 23:44:27 +04:00
static int regmap_spmi_ext_write ( void * context , const void * data ,
size_t count )
2013-10-28 22:12:35 +04:00
{
BUG_ON ( count < 2 ) ;
2014-02-12 23:44:27 +04:00
return regmap_spmi_ext_gather_write ( context , data , 2 , data + 2 ,
count - 2 ) ;
2013-10-28 22:12:35 +04:00
}
2017-08-11 09:57:36 +03:00
static const struct regmap_bus regmap_spmi_ext = {
2014-02-12 23:44:27 +04:00
. read = regmap_spmi_ext_read ,
. write = regmap_spmi_ext_write ,
. gather_write = regmap_spmi_ext_gather_write ,
2013-10-28 22:12:35 +04:00
. reg_format_endian_default = REGMAP_ENDIAN_NATIVE ,
. val_format_endian_default = REGMAP_ENDIAN_NATIVE ,
} ;
2015-07-08 09:30:18 +03:00
struct regmap * __regmap_init_spmi_ext ( struct spmi_device * sdev ,
const struct regmap_config * config ,
struct lock_class_key * lock_key ,
const char * lock_name )
2013-10-28 22:12:35 +04:00
{
2015-07-08 09:30:18 +03:00
return __regmap_init ( & sdev - > dev , & regmap_spmi_ext , sdev , config ,
lock_key , lock_name ) ;
2013-10-28 22:12:35 +04:00
}
2015-07-08 09:30:18 +03:00
EXPORT_SYMBOL_GPL ( __regmap_init_spmi_ext ) ;
2013-10-28 22:12:35 +04:00
2015-07-08 09:30:18 +03:00
struct regmap * __devm_regmap_init_spmi_ext ( struct spmi_device * sdev ,
const struct regmap_config * config ,
struct lock_class_key * lock_key ,
const char * lock_name )
2013-10-28 22:12:35 +04:00
{
2015-07-08 09:30:18 +03:00
return __devm_regmap_init ( & sdev - > dev , & regmap_spmi_ext , sdev , config ,
lock_key , lock_name ) ;
2013-10-28 22:12:35 +04:00
}
2015-07-08 09:30:18 +03:00
EXPORT_SYMBOL_GPL ( __devm_regmap_init_spmi_ext ) ;
2013-10-28 22:12:35 +04:00
MODULE_LICENSE ( " GPL " ) ;