2019-05-27 09:55:06 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2012-08-18 20:06:27 +04:00
/*
* NXP SC18IS602 / 603 SPI driver
*
* Copyright ( C ) Guenter Roeck < linux @ roeck - us . net >
*/
# include <linux/kernel.h>
# include <linux/err.h>
# include <linux/module.h>
# include <linux/spi/spi.h>
# include <linux/i2c.h>
# include <linux/delay.h>
# include <linux/pm_runtime.h>
2017-02-22 21:14:20 +03:00
# include <linux/of_device.h>
2012-08-18 20:06:27 +04:00
# include <linux/of.h>
# include <linux/platform_data/sc18is602.h>
2016-08-31 10:31:38 +03:00
# include <linux/gpio/consumer.h>
2012-08-18 20:06:27 +04:00
enum chips { sc18is602 , sc18is602b , sc18is603 } ;
# define SC18IS602_BUFSIZ 200
# define SC18IS602_CLOCK 7372000
# define SC18IS602_MODE_CPHA BIT(2)
# define SC18IS602_MODE_CPOL BIT(3)
# define SC18IS602_MODE_LSB_FIRST BIT(5)
# define SC18IS602_MODE_CLOCK_DIV_4 0x0
# define SC18IS602_MODE_CLOCK_DIV_16 0x1
# define SC18IS602_MODE_CLOCK_DIV_64 0x2
# define SC18IS602_MODE_CLOCK_DIV_128 0x3
struct sc18is602 {
struct spi_master * master ;
struct device * dev ;
u8 ctrl ;
u32 freq ;
u32 speed ;
/* I2C data */
struct i2c_client * client ;
enum chips id ;
u8 buffer [ SC18IS602_BUFSIZ + 1 ] ;
int tlen ; /* Data queued for tx in buffer */
int rindex ; /* Receive data index in buffer */
2016-08-31 10:31:38 +03:00
struct gpio_desc * reset ;
2012-08-18 20:06:27 +04:00
} ;
static int sc18is602_wait_ready ( struct sc18is602 * hw , int len )
{
int i , err ;
int usecs = 1000000 * len / hw - > speed + 1 ;
u8 dummy [ 1 ] ;
for ( i = 0 ; i < 10 ; i + + ) {
err = i2c_master_recv ( hw - > client , dummy , 1 ) ;
if ( err > = 0 )
return 0 ;
usleep_range ( usecs , usecs * 2 ) ;
}
return - ETIMEDOUT ;
}
static int sc18is602_txrx ( struct sc18is602 * hw , struct spi_message * msg ,
struct spi_transfer * t , bool do_transfer )
{
unsigned int len = t - > len ;
int ret ;
if ( hw - > tlen = = 0 ) {
/* First byte (I2C command) is chip select */
2023-03-10 20:32:03 +03:00
hw - > buffer [ 0 ] = 1 < < spi_get_chipselect ( msg - > spi , 0 ) ;
2012-08-18 20:06:27 +04:00
hw - > tlen = 1 ;
hw - > rindex = 0 ;
}
/*
* We can not immediately send data to the chip , since each I2C message
* resembles a full SPI message ( from CS active to CS inactive ) .
* Enqueue messages up to the first read or until do_transfer is true .
*/
if ( t - > tx_buf ) {
memcpy ( & hw - > buffer [ hw - > tlen ] , t - > tx_buf , len ) ;
hw - > tlen + = len ;
if ( t - > rx_buf )
do_transfer = true ;
else
hw - > rindex = hw - > tlen - 1 ;
} else if ( t - > rx_buf ) {
/*
* For receive - only transfers we still need to perform a dummy
* write to receive data from the SPI chip .
* Read data starts at the end of transmit data ( minus 1 to
* account for CS ) .
*/
hw - > rindex = hw - > tlen - 1 ;
memset ( & hw - > buffer [ hw - > tlen ] , 0 , len ) ;
hw - > tlen + = len ;
do_transfer = true ;
}
if ( do_transfer & & hw - > tlen > 1 ) {
ret = sc18is602_wait_ready ( hw , SC18IS602_BUFSIZ ) ;
if ( ret < 0 )
return ret ;
ret = i2c_master_send ( hw - > client , hw - > buffer , hw - > tlen ) ;
if ( ret < 0 )
return ret ;
if ( ret ! = hw - > tlen )
return - EIO ;
if ( t - > rx_buf ) {
int rlen = hw - > rindex + len ;
ret = sc18is602_wait_ready ( hw , hw - > tlen ) ;
if ( ret < 0 )
return ret ;
ret = i2c_master_recv ( hw - > client , hw - > buffer , rlen ) ;
if ( ret < 0 )
return ret ;
if ( ret ! = rlen )
return - EIO ;
memcpy ( t - > rx_buf , & hw - > buffer [ hw - > rindex ] , len ) ;
}
hw - > tlen = 0 ;
}
return len ;
}
static int sc18is602_setup_transfer ( struct sc18is602 * hw , u32 hz , u8 mode )
{
u8 ctrl = 0 ;
int ret ;
if ( mode & SPI_CPHA )
ctrl | = SC18IS602_MODE_CPHA ;
if ( mode & SPI_CPOL )
ctrl | = SC18IS602_MODE_CPOL ;
if ( mode & SPI_LSB_FIRST )
ctrl | = SC18IS602_MODE_LSB_FIRST ;
/* Find the closest clock speed */
if ( hz > = hw - > freq / 4 ) {
ctrl | = SC18IS602_MODE_CLOCK_DIV_4 ;
hw - > speed = hw - > freq / 4 ;
} else if ( hz > = hw - > freq / 16 ) {
ctrl | = SC18IS602_MODE_CLOCK_DIV_16 ;
hw - > speed = hw - > freq / 16 ;
} else if ( hz > = hw - > freq / 64 ) {
ctrl | = SC18IS602_MODE_CLOCK_DIV_64 ;
hw - > speed = hw - > freq / 64 ;
} else {
ctrl | = SC18IS602_MODE_CLOCK_DIV_128 ;
hw - > speed = hw - > freq / 128 ;
}
/*
* Don ' t do anything if the control value did not change . The initial
* value of 0xff for hw - > ctrl ensures that the correct mode will be set
* with the first call to this function .
*/
if ( ctrl = = hw - > ctrl )
return 0 ;
ret = i2c_smbus_write_byte_data ( hw - > client , 0xf0 , ctrl ) ;
if ( ret < 0 )
return ret ;
hw - > ctrl = ctrl ;
return 0 ;
}
static int sc18is602_check_transfer ( struct spi_device * spi ,
struct spi_transfer * t , int tlen )
{
2021-05-20 16:12:37 +03:00
if ( t & & t - > len + tlen > SC18IS602_BUFSIZ + 1 )
2012-08-18 20:06:27 +04:00
return - EINVAL ;
return 0 ;
}
static int sc18is602_transfer_one ( struct spi_master * master ,
struct spi_message * m )
{
struct sc18is602 * hw = spi_master_get_devdata ( master ) ;
struct spi_device * spi = m - > spi ;
struct spi_transfer * t ;
int status = 0 ;
hw - > tlen = 0 ;
list_for_each_entry ( t , & m - > transfers , transfer_list ) {
bool do_transfer ;
status = sc18is602_check_transfer ( spi , t , hw - > tlen ) ;
if ( status < 0 )
break ;
2014-02-28 14:39:33 +04:00
status = sc18is602_setup_transfer ( hw , t - > speed_hz , spi - > mode ) ;
2012-08-18 20:06:27 +04:00
if ( status < 0 )
break ;
do_transfer = t - > cs_change | | list_is_last ( & t - > transfer_list ,
& m - > transfers ) ;
if ( t - > len ) {
status = sc18is602_txrx ( hw , m , t , do_transfer ) ;
if ( status < 0 )
break ;
m - > actual_length + = status ;
}
status = 0 ;
2019-09-26 13:51:37 +03:00
spi_transfer_delay_exec ( t ) ;
2012-08-18 20:06:27 +04:00
}
m - > status = status ;
spi_finalize_current_message ( master ) ;
return status ;
}
2021-05-20 16:12:38 +03:00
static size_t sc18is602_max_transfer_size ( struct spi_device * spi )
{
return SC18IS602_BUFSIZ ;
}
2014-02-11 16:54:17 +04:00
static int sc18is602_setup ( struct spi_device * spi )
{
struct sc18is602 * hw = spi_master_get_devdata ( spi - > master ) ;
/* SC18IS602 does not support CS2 */
2023-03-10 20:32:03 +03:00
if ( hw - > id = = sc18is602 & & ( spi_get_chipselect ( spi , 0 ) = = 2 ) )
2014-02-11 16:54:17 +04:00
return - ENXIO ;
return 0 ;
}
2022-11-19 01:44:58 +03:00
static int sc18is602_probe ( struct i2c_client * client )
2012-08-18 20:06:27 +04:00
{
2022-11-19 01:44:58 +03:00
const struct i2c_device_id * id = i2c_client_get_device_id ( client ) ;
2012-08-18 20:06:27 +04:00
struct device * dev = & client - > dev ;
struct device_node * np = dev - > of_node ;
struct sc18is602_platform_data * pdata = dev_get_platdata ( dev ) ;
struct sc18is602 * hw ;
struct spi_master * master ;
if ( ! i2c_check_functionality ( client - > adapter , I2C_FUNC_I2C |
I2C_FUNC_SMBUS_WRITE_BYTE_DATA ) )
2012-08-23 04:28:55 +04:00
return - EINVAL ;
2012-08-18 20:06:27 +04:00
2020-12-07 11:17:11 +03:00
master = devm_spi_alloc_master ( dev , sizeof ( struct sc18is602 ) ) ;
2012-08-18 20:06:27 +04:00
if ( ! master )
return - ENOMEM ;
hw = spi_master_get_devdata ( master ) ;
i2c_set_clientdata ( client , hw ) ;
2016-08-31 10:31:38 +03:00
/* assert reset and then release */
hw - > reset = devm_gpiod_get_optional ( dev , " reset " , GPIOD_OUT_HIGH ) ;
if ( IS_ERR ( hw - > reset ) )
return PTR_ERR ( hw - > reset ) ;
2016-09-29 05:41:02 +03:00
gpiod_set_value_cansleep ( hw - > reset , 0 ) ;
2016-08-31 10:31:38 +03:00
2012-08-18 20:06:27 +04:00
hw - > master = master ;
hw - > client = client ;
hw - > dev = dev ;
hw - > ctrl = 0xff ;
2017-02-22 21:14:20 +03:00
if ( client - > dev . of_node )
hw - > id = ( enum chips ) of_device_get_match_data ( & client - > dev ) ;
else
hw - > id = id - > driver_data ;
2012-08-18 20:06:27 +04:00
switch ( hw - > id ) {
case sc18is602 :
case sc18is602b :
master - > num_chipselect = 4 ;
hw - > freq = SC18IS602_CLOCK ;
break ;
case sc18is603 :
master - > num_chipselect = 2 ;
if ( pdata ) {
hw - > freq = pdata - > clock_frequency ;
} else {
const __be32 * val ;
int len ;
val = of_get_property ( np , " clock-frequency " , & len ) ;
if ( val & & len > = sizeof ( __be32 ) )
hw - > freq = be32_to_cpup ( val ) ;
}
if ( ! hw - > freq )
hw - > freq = SC18IS602_CLOCK ;
break ;
}
2015-02-06 23:27:54 +03:00
master - > bus_num = np ? - 1 : client - > adapter - > nr ;
2012-08-18 20:06:27 +04:00
master - > mode_bits = SPI_CPHA | SPI_CPOL | SPI_LSB_FIRST ;
2014-01-17 14:00:42 +04:00
master - > bits_per_word_mask = SPI_BPW_MASK ( 8 ) ;
2014-02-11 16:54:17 +04:00
master - > setup = sc18is602_setup ;
2012-08-18 20:06:27 +04:00
master - > transfer_one_message = sc18is602_transfer_one ;
2021-05-20 16:12:38 +03:00
master - > max_transfer_size = sc18is602_max_transfer_size ;
master - > max_message_size = sc18is602_max_transfer_size ;
2012-08-18 20:06:27 +04:00
master - > dev . of_node = np ;
2014-02-28 14:39:33 +04:00
master - > min_speed_hz = hw - > freq / 128 ;
master - > max_speed_hz = hw - > freq / 4 ;
2012-08-18 20:06:27 +04:00
2020-12-07 11:17:11 +03:00
return devm_spi_register_master ( dev , master ) ;
2012-08-18 20:06:27 +04:00
}
static const struct i2c_device_id sc18is602_id [ ] = {
{ " sc18is602 " , sc18is602 } ,
{ " sc18is602b " , sc18is602b } ,
{ " sc18is603 " , sc18is603 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , sc18is602_id ) ;
2023-03-11 01:28:56 +03:00
static const struct of_device_id sc18is602_of_match [ ] __maybe_unused = {
2017-02-22 21:14:20 +03:00
{
. compatible = " nxp,sc18is602 " ,
. data = ( void * ) sc18is602
} ,
{
. compatible = " nxp,sc18is602b " ,
. data = ( void * ) sc18is602b
} ,
{
. compatible = " nxp,sc18is603 " ,
. data = ( void * ) sc18is603
} ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , sc18is602_of_match ) ;
2012-08-18 20:06:27 +04:00
static struct i2c_driver sc18is602_driver = {
. driver = {
. name = " sc18is602 " ,
2017-02-22 21:14:20 +03:00
. of_match_table = of_match_ptr ( sc18is602_of_match ) ,
2012-08-18 20:06:27 +04:00
} ,
2022-11-19 01:44:58 +03:00
. probe_new = sc18is602_probe ,
2012-08-18 20:06:27 +04:00
. id_table = sc18is602_id ,
} ;
module_i2c_driver ( sc18is602_driver ) ;
2020-04-13 18:40:43 +03:00
MODULE_DESCRIPTION ( " SC18IS602/603 SPI Master Driver " ) ;
2012-08-18 20:06:27 +04:00
MODULE_AUTHOR ( " Guenter Roeck " ) ;
MODULE_LICENSE ( " GPL " ) ;