2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2010-10-01 16:41:51 +02:00
/*
* iio / adc / ad799x . c
2014-02-16 11:57:00 +00:00
* Copyright ( C ) 2010 - 2011 Michael Hennerich , Analog Devices Inc .
2010-10-01 16:41:51 +02:00
*
* based on iio / adc / max1363
* Copyright ( C ) 2008 - 2010 Jonathan Cameron
*
* based on linux / drivers / i2c / chips / max123x
* Copyright ( C ) 2002 - 2004 Stefan Eletzhofer
*
* based on linux / drivers / acron / char / pcf8583 . c
* Copyright ( C ) 2000 Russell King
*
* ad799x . c
*
* Support for ad7991 , ad7995 , ad7999 , ad7992 , ad7993 , ad7994 , ad7997 ,
* ad7998 and similar chips .
*/
# include <linux/interrupt.h>
# include <linux/device.h>
# include <linux/kernel.h>
# include <linux/sysfs.h>
# include <linux/i2c.h>
# include <linux/regulator/consumer.h>
# include <linux/slab.h>
# include <linux/types.h>
# include <linux/err.h>
2011-07-03 15:49:50 -04:00
# include <linux/module.h>
2022-10-04 15:48:54 +02:00
# include <linux/mutex.h>
2014-12-06 05:54:00 +00:00
# include <linux/bitops.h>
2010-10-01 16:41:51 +02:00
2012-04-25 15:54:58 +01:00
# include <linux/iio/iio.h>
# include <linux/iio/sysfs.h>
# include <linux/iio/events.h>
# include <linux/iio/buffer.h>
2014-10-03 10:31:00 +01:00
# include <linux/iio/trigger_consumer.h>
# include <linux/iio/triggered_buffer.h>
2011-08-12 17:08:43 +01:00
2014-10-03 10:31:00 +01:00
# define AD799X_CHANNEL_SHIFT 4
2014-12-06 05:54:00 +00:00
2014-10-03 10:31:00 +01:00
/*
* AD7991 , AD7995 and AD7999 defines
*/
# define AD7991_REF_SEL 0x08
# define AD7991_FLTR 0x04
# define AD7991_BIT_TRIAL_DELAY 0x02
# define AD7991_SAMPLE_DELAY 0x01
/*
* AD7992 , AD7993 , AD7994 , AD7997 and AD7998 defines
*/
2014-12-06 05:54:00 +00:00
# define AD7998_FLTR BIT(3)
# define AD7998_ALERT_EN BIT(2)
# define AD7998_BUSY_ALERT BIT(1)
# define AD7998_BUSY_ALERT_POL BIT(0)
2014-10-03 10:31:00 +01:00
# define AD7998_CONV_RES_REG 0x0
# define AD7998_ALERT_STAT_REG 0x1
# define AD7998_CONF_REG 0x2
# define AD7998_CYCLE_TMR_REG 0x3
# define AD7998_DATALOW_REG(x) ((x) * 3 + 0x4)
# define AD7998_DATAHIGH_REG(x) ((x) * 3 + 0x5)
# define AD7998_HYST_REG(x) ((x) * 3 + 0x6)
2014-12-06 05:54:00 +00:00
# define AD7998_CYC_MASK GENMASK(2, 0)
2014-10-03 10:31:00 +01:00
# define AD7998_CYC_DIS 0x0
# define AD7998_CYC_TCONF_32 0x1
# define AD7998_CYC_TCONF_64 0x2
# define AD7998_CYC_TCONF_128 0x3
# define AD7998_CYC_TCONF_256 0x4
# define AD7998_CYC_TCONF_512 0x5
# define AD7998_CYC_TCONF_1024 0x6
# define AD7998_CYC_TCONF_2048 0x7
# define AD7998_ALERT_STAT_CLEAR 0xFF
/*
* AD7997 and AD7997 defines
*/
2014-12-06 05:54:00 +00:00
# define AD7997_8_READ_SINGLE BIT(7)
# define AD7997_8_READ_SEQUENCE (BIT(6) | BIT(5) | BIT(4))
2014-10-03 10:31:00 +01:00
enum {
ad7991 ,
ad7995 ,
ad7999 ,
ad7992 ,
ad7993 ,
ad7994 ,
ad7997 ,
ad7998
} ;
/**
2014-12-06 05:54:00 +00:00
* struct ad799x_chip_config - chip specific information
2014-10-03 10:31:00 +01:00
* @ channel : channel specification
* @ default_config : device default configuration
2014-12-06 05:54:00 +00:00
* @ info : pointer to iio_info struct
2014-10-03 10:31:00 +01:00
*/
2014-12-06 05:54:00 +00:00
struct ad799x_chip_config {
2014-12-06 05:54:00 +00:00
const struct iio_chan_spec channel [ 9 ] ;
2014-10-03 10:31:00 +01:00
u16 default_config ;
const struct iio_info * info ;
} ;
2014-12-06 05:54:00 +00:00
/**
* struct ad799x_chip_info - chip specific information
* @ num_channels : number of channels
* @ noirq_config : device configuration w / o IRQ
* @ irq_config : device configuration w / IRQ
*/
struct ad799x_chip_info {
int num_channels ;
const struct ad799x_chip_config noirq_config ;
const struct ad799x_chip_config irq_config ;
} ;
2014-10-03 10:31:00 +01:00
struct ad799x_state {
struct i2c_client * client ;
2014-12-06 05:54:00 +00:00
const struct ad799x_chip_config * chip_config ;
2014-10-03 10:31:00 +01:00
struct regulator * reg ;
struct regulator * vref ;
2022-10-04 15:48:54 +02:00
/* lock to protect against multiple access to the device */
struct mutex lock ;
2014-10-03 10:31:00 +01:00
unsigned id ;
u16 config ;
u8 * rx_buf ;
unsigned int transfer_size ;
} ;
2014-12-06 05:54:00 +00:00
static int ad799x_write_config ( struct ad799x_state * st , u16 val )
{
switch ( st - > id ) {
case ad7997 :
case ad7998 :
return i2c_smbus_write_word_swapped ( st - > client , AD7998_CONF_REG ,
val ) ;
2014-11-26 11:35:35 +01:00
case ad7992 :
case ad7993 :
case ad7994 :
2014-12-06 05:54:00 +00:00
return i2c_smbus_write_byte_data ( st - > client , AD7998_CONF_REG ,
val ) ;
2014-11-26 11:35:35 +01:00
default :
/* Will be written when doing a conversion */
st - > config = val ;
return 0 ;
2014-12-06 05:54:00 +00:00
}
}
static int ad799x_read_config ( struct ad799x_state * st )
{
switch ( st - > id ) {
case ad7997 :
case ad7998 :
return i2c_smbus_read_word_swapped ( st - > client , AD7998_CONF_REG ) ;
2014-11-26 11:35:35 +01:00
case ad7992 :
case ad7993 :
case ad7994 :
2014-12-06 05:54:00 +00:00
return i2c_smbus_read_byte_data ( st - > client , AD7998_CONF_REG ) ;
2014-11-26 11:35:35 +01:00
default :
/* No readback support */
return st - > config ;
2014-12-06 05:54:00 +00:00
}
}
2019-11-29 17:53:14 +01:00
static int ad799x_update_config ( struct ad799x_state * st , u16 config )
{
int ret ;
ret = ad799x_write_config ( st , config ) ;
if ( ret < 0 )
return ret ;
ret = ad799x_read_config ( st ) ;
if ( ret < 0 )
return ret ;
st - > config = ret ;
return 0 ;
}
2014-10-03 10:31:00 +01:00
static irqreturn_t ad799x_trigger_handler ( int irq , void * p )
{
struct iio_poll_func * pf = p ;
struct iio_dev * indio_dev = pf - > indio_dev ;
struct ad799x_state * st = iio_priv ( indio_dev ) ;
int b_sent ;
u8 cmd ;
switch ( st - > id ) {
case ad7991 :
case ad7995 :
case ad7999 :
cmd = st - > config |
( * indio_dev - > active_scan_mask < < AD799X_CHANNEL_SHIFT ) ;
break ;
case ad7992 :
case ad7993 :
case ad7994 :
cmd = ( * indio_dev - > active_scan_mask < < AD799X_CHANNEL_SHIFT ) |
AD7998_CONV_RES_REG ;
break ;
case ad7997 :
case ad7998 :
cmd = AD7997_8_READ_SEQUENCE | AD7998_CONV_RES_REG ;
break ;
default :
cmd = 0 ;
}
b_sent = i2c_smbus_read_i2c_block_data ( st - > client ,
cmd , st - > transfer_size , st - > rx_buf ) ;
if ( b_sent < 0 )
goto out ;
iio_push_to_buffers_with_timestamp ( indio_dev , st - > rx_buf ,
2016-03-09 19:05:49 +01:00
iio_get_time_ns ( indio_dev ) ) ;
2014-10-03 10:31:00 +01:00
out :
iio_trigger_notify_done ( indio_dev - > trig ) ;
return IRQ_HANDLED ;
}
2010-10-01 16:41:51 +02:00
2014-12-06 05:54:00 +00:00
static int ad799x_update_scan_mode ( struct iio_dev * indio_dev ,
2012-06-18 18:33:56 +02:00
const unsigned long * scan_mask )
2010-10-01 16:41:51 +02:00
{
2012-06-18 18:33:56 +02:00
struct ad799x_state * st = iio_priv ( indio_dev ) ;
2013-03-26 18:43:00 +00:00
kfree ( st - > rx_buf ) ;
st - > rx_buf = kmalloc ( indio_dev - > scan_bytes , GFP_KERNEL ) ;
if ( ! st - > rx_buf )
return - ENOMEM ;
st - > transfer_size = bitmap_weight ( scan_mask , indio_dev - > masklength ) * 2 ;
2012-06-18 18:33:56 +02:00
switch ( st - > id ) {
2014-12-06 05:54:00 +00:00
case ad7992 :
case ad7993 :
case ad7994 :
2012-06-18 18:33:56 +02:00
case ad7997 :
case ad7998 :
2014-12-06 05:54:00 +00:00
st - > config & = ~ ( GENMASK ( 7 , 0 ) < < AD799X_CHANNEL_SHIFT ) ;
st - > config | = ( * scan_mask < < AD799X_CHANNEL_SHIFT ) ;
return ad799x_write_config ( st , st - > config ) ;
2012-06-18 18:33:56 +02:00
default :
2014-12-06 05:54:00 +00:00
return 0 ;
2012-06-18 18:33:56 +02:00
}
2010-10-01 16:41:51 +02:00
}
2011-05-18 14:41:52 +01:00
static int ad799x_scan_direct ( struct ad799x_state * st , unsigned ch )
2010-10-01 16:41:51 +02:00
{
2011-05-18 14:41:52 +01:00
u8 cmd ;
2010-10-01 16:41:51 +02:00
2011-05-18 14:41:52 +01:00
switch ( st - > id ) {
case ad7991 :
case ad7995 :
case ad7999 :
2014-12-06 05:54:00 +00:00
cmd = st - > config | ( BIT ( ch ) < < AD799X_CHANNEL_SHIFT ) ;
2011-05-18 14:41:52 +01:00
break ;
case ad7992 :
case ad7993 :
case ad7994 :
2014-12-06 05:54:00 +00:00
cmd = BIT ( ch ) < < AD799X_CHANNEL_SHIFT ;
2011-05-18 14:41:52 +01:00
break ;
case ad7997 :
case ad7998 :
cmd = ( ch < < AD799X_CHANNEL_SHIFT ) | AD7997_8_READ_SINGLE ;
break ;
default :
return - EINVAL ;
2010-10-01 16:41:51 +02:00
}
2014-12-06 05:54:00 +00:00
return i2c_smbus_read_word_swapped ( st - > client , cmd ) ;
2010-10-01 16:41:51 +02:00
}
2011-10-06 17:14:37 +01:00
static int ad799x_read_raw ( struct iio_dev * indio_dev ,
2011-05-18 14:41:52 +01:00
struct iio_chan_spec const * chan ,
int * val ,
int * val2 ,
long m )
2010-10-01 16:41:51 +02:00
{
2011-05-18 14:41:52 +01:00
int ret ;
2011-10-06 17:14:37 +01:00
struct ad799x_state * st = iio_priv ( indio_dev ) ;
2011-05-18 14:41:52 +01:00
switch ( m ) {
2012-04-15 17:41:18 +01:00
case IIO_CHAN_INFO_RAW :
2016-05-24 12:20:24 -07:00
ret = iio_device_claim_direct_mode ( indio_dev ) ;
if ( ret )
return ret ;
2022-10-04 15:48:54 +02:00
mutex_lock ( & st - > lock ) ;
2016-05-24 12:20:24 -07:00
ret = ad799x_scan_direct ( st , chan - > scan_index ) ;
2022-10-04 15:48:54 +02:00
mutex_unlock ( & st - > lock ) ;
2016-05-24 12:20:24 -07:00
iio_device_release_direct_mode ( indio_dev ) ;
2010-10-01 16:41:51 +02:00
if ( ret < 0 )
2011-05-18 14:41:52 +01:00
return ret ;
2011-09-30 10:05:32 +01:00
* val = ( ret > > chan - > scan_type . shift ) &
2014-12-06 05:54:00 +00:00
GENMASK ( chan - > scan_type . realbits - 1 , 0 ) ;
2011-05-18 14:41:52 +01:00
return IIO_VAL_INT ;
2011-10-26 17:41:36 +01:00
case IIO_CHAN_INFO_SCALE :
2021-09-30 12:42:48 +02:00
if ( st - > vref )
ret = regulator_get_voltage ( st - > vref ) ;
else
ret = regulator_get_voltage ( st - > reg ) ;
2014-02-16 12:02:00 +00:00
if ( ret < 0 )
return ret ;
* val = ret / 1000 ;
2013-09-28 10:31:00 +01:00
* val2 = chan - > scan_type . realbits ;
return IIO_VAL_FRACTIONAL_LOG2 ;
2010-10-01 16:41:51 +02:00
}
2011-05-18 14:41:52 +01:00
return - EINVAL ;
2010-10-01 16:41:51 +02:00
}
2011-09-30 10:05:33 +01:00
static const unsigned int ad7998_frequencies [ ] = {
[ AD7998_CYC_DIS ] = 0 ,
[ AD7998_CYC_TCONF_32 ] = 15625 ,
[ AD7998_CYC_TCONF_64 ] = 7812 ,
[ AD7998_CYC_TCONF_128 ] = 3906 ,
[ AD7998_CYC_TCONF_512 ] = 976 ,
[ AD7998_CYC_TCONF_1024 ] = 488 ,
[ AD7998_CYC_TCONF_2048 ] = 244 ,
} ;
2014-12-06 05:54:00 +00:00
2010-10-01 16:41:51 +02:00
static ssize_t ad799x_read_frequency ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
{
2012-05-12 15:39:42 +02:00
struct iio_dev * indio_dev = dev_to_iio_dev ( dev ) ;
2011-10-06 17:14:37 +01:00
struct ad799x_state * st = iio_priv ( indio_dev ) ;
2010-10-01 16:41:51 +02:00
2014-12-06 05:54:00 +00:00
int ret = i2c_smbus_read_byte_data ( st - > client , AD7998_CYCLE_TMR_REG ) ;
if ( ret < 0 )
2010-10-01 16:41:51 +02:00
return ret ;
2014-12-06 05:54:00 +00:00
return sprintf ( buf , " %u \n " , ad7998_frequencies [ ret & AD7998_CYC_MASK ] ) ;
2010-10-01 16:41:51 +02:00
}
static ssize_t ad799x_write_frequency ( struct device * dev ,
struct device_attribute * attr ,
const char * buf ,
size_t len )
{
2012-05-12 15:39:42 +02:00
struct iio_dev * indio_dev = dev_to_iio_dev ( dev ) ;
2011-10-06 17:14:37 +01:00
struct ad799x_state * st = iio_priv ( indio_dev ) ;
2010-10-01 16:41:51 +02:00
long val ;
2011-09-30 10:05:33 +01:00
int ret , i ;
2010-10-01 16:41:51 +02:00
2013-05-07 00:04:41 +04:00
ret = kstrtol ( buf , 10 , & val ) ;
2010-10-01 16:41:51 +02:00
if ( ret )
return ret ;
2022-10-04 15:48:54 +02:00
mutex_lock ( & st - > lock ) ;
2014-12-06 05:54:00 +00:00
ret = i2c_smbus_read_byte_data ( st - > client , AD7998_CYCLE_TMR_REG ) ;
if ( ret < 0 )
2010-10-01 16:41:51 +02:00
goto error_ret_mutex ;
/* Wipe the bits clean */
2014-12-06 05:54:00 +00:00
ret & = ~ AD7998_CYC_MASK ;
2010-10-01 16:41:51 +02:00
2011-09-30 10:05:33 +01:00
for ( i = 0 ; i < ARRAY_SIZE ( ad7998_frequencies ) ; i + + )
if ( val = = ad7998_frequencies [ i ] )
break ;
if ( i = = ARRAY_SIZE ( ad7998_frequencies ) ) {
2010-10-01 16:41:51 +02:00
ret = - EINVAL ;
goto error_ret_mutex ;
}
2014-12-06 05:54:00 +00:00
ret = i2c_smbus_write_byte_data ( st - > client , AD7998_CYCLE_TMR_REG ,
ret | i ) ;
if ( ret < 0 )
goto error_ret_mutex ;
ret = len ;
2010-10-01 16:41:51 +02:00
error_ret_mutex :
2022-10-04 15:48:54 +02:00
mutex_unlock ( & st - > lock ) ;
2010-10-01 16:41:51 +02:00
2014-12-06 05:54:00 +00:00
return ret ;
2010-10-01 16:41:51 +02:00
}
2011-10-06 17:14:37 +01:00
static int ad799x_read_event_config ( struct iio_dev * indio_dev ,
2013-10-07 15:11:00 +01:00
const struct iio_chan_spec * chan ,
enum iio_event_type type ,
enum iio_event_direction dir )
2011-09-30 10:05:35 +01:00
{
2014-12-06 05:54:00 +00:00
struct ad799x_state * st = iio_priv ( indio_dev ) ;
if ( ! ( st - > config & AD7998_ALERT_EN ) )
return 0 ;
if ( ( st - > config > > AD799X_CHANNEL_SHIFT ) & BIT ( chan - > scan_index ) )
return 1 ;
return 0 ;
2011-09-30 10:05:35 +01:00
}
2014-12-06 05:54:00 +00:00
static int ad799x_write_event_config ( struct iio_dev * indio_dev ,
const struct iio_chan_spec * chan ,
enum iio_event_type type ,
enum iio_event_direction dir ,
int state )
{
struct ad799x_state * st = iio_priv ( indio_dev ) ;
int ret ;
2016-05-24 12:20:24 -07:00
ret = iio_device_claim_direct_mode ( indio_dev ) ;
if ( ret )
return ret ;
2014-12-06 05:54:00 +00:00
2022-10-04 15:48:54 +02:00
mutex_lock ( & st - > lock ) ;
2014-12-06 05:54:00 +00:00
if ( state )
st - > config | = BIT ( chan - > scan_index ) < < AD799X_CHANNEL_SHIFT ;
else
st - > config & = ~ ( BIT ( chan - > scan_index ) < < AD799X_CHANNEL_SHIFT ) ;
if ( st - > config > > AD799X_CHANNEL_SHIFT )
st - > config | = AD7998_ALERT_EN ;
else
st - > config & = ~ AD7998_ALERT_EN ;
ret = ad799x_write_config ( st , st - > config ) ;
2022-10-04 15:48:54 +02:00
mutex_unlock ( & st - > lock ) ;
2016-05-24 12:20:24 -07:00
iio_device_release_direct_mode ( indio_dev ) ;
2014-12-06 05:54:00 +00:00
return ret ;
}
2013-10-07 15:11:00 +01:00
static unsigned int ad799x_threshold_reg ( const struct iio_chan_spec * chan ,
enum iio_event_direction dir ,
enum iio_event_info info )
2013-10-07 15:11:00 +01:00
{
2013-10-07 15:11:00 +01:00
switch ( info ) {
case IIO_EV_INFO_VALUE :
if ( dir = = IIO_EV_DIR_FALLING )
return AD7998_DATALOW_REG ( chan - > channel ) ;
else
return AD7998_DATAHIGH_REG ( chan - > channel ) ;
case IIO_EV_INFO_HYSTERESIS :
return AD7998_HYST_REG ( chan - > channel ) ;
default :
return - EINVAL ;
}
return 0 ;
2013-10-07 15:11:00 +01:00
}
2011-09-30 10:05:35 +01:00
static int ad799x_write_event_value ( struct iio_dev * indio_dev ,
2013-10-07 15:11:00 +01:00
const struct iio_chan_spec * chan ,
enum iio_event_type type ,
enum iio_event_direction dir ,
enum iio_event_info info ,
int val , int val2 )
2011-09-30 10:05:35 +01:00
{
int ret ;
struct ad799x_state * st = iio_priv ( indio_dev ) ;
2014-12-06 05:54:00 +00:00
if ( val < 0 | | val > GENMASK ( chan - > scan_type . realbits - 1 , 0 ) )
2014-12-06 05:54:00 +00:00
return - EINVAL ;
2014-12-06 05:54:00 +00:00
ret = i2c_smbus_write_word_swapped ( st - > client ,
ad799x_threshold_reg ( chan , dir , info ) ,
2014-12-06 05:54:00 +00:00
val < < chan - > scan_type . shift ) ;
2011-09-30 10:05:35 +01:00
return ret ;
}
static int ad799x_read_event_value ( struct iio_dev * indio_dev ,
2013-10-07 15:11:00 +01:00
const struct iio_chan_spec * chan ,
enum iio_event_type type ,
enum iio_event_direction dir ,
enum iio_event_info info ,
int * val , int * val2 )
2011-09-30 10:05:35 +01:00
{
int ret ;
struct ad799x_state * st = iio_priv ( indio_dev ) ;
2014-12-06 05:54:00 +00:00
ret = i2c_smbus_read_word_swapped ( st - > client ,
ad799x_threshold_reg ( chan , dir , info ) ) ;
2011-09-30 10:05:35 +01:00
if ( ret < 0 )
return ret ;
2014-12-06 05:54:00 +00:00
* val = ( ret > > chan - > scan_type . shift ) &
2016-04-14 21:36:38 +02:00
GENMASK ( chan - > scan_type . realbits - 1 , 0 ) ;
2011-09-30 10:05:35 +01:00
2013-10-07 15:11:00 +01:00
return IIO_VAL_INT ;
2011-09-30 10:05:35 +01:00
}
2011-05-18 14:41:15 +01:00
static irqreturn_t ad799x_event_handler ( int irq , void * private )
2010-10-01 16:41:51 +02:00
{
2011-05-18 14:41:15 +01:00
struct iio_dev * indio_dev = private ;
2011-06-27 13:07:20 +01:00
struct ad799x_state * st = iio_priv ( private ) ;
2011-05-18 14:41:15 +01:00
int i , ret ;
2010-10-01 16:41:51 +02:00
2014-12-06 05:54:00 +00:00
ret = i2c_smbus_read_byte_data ( st - > client , AD7998_ALERT_STAT_REG ) ;
if ( ret < = 0 )
2012-07-04 17:09:00 +01:00
goto done ;
2010-10-01 16:41:51 +02:00
2014-12-06 05:54:00 +00:00
if ( i2c_smbus_write_byte_data ( st - > client , AD7998_ALERT_STAT_REG ,
AD7998_ALERT_STAT_CLEAR ) < 0 )
2012-07-04 17:09:00 +01:00
goto done ;
2010-10-01 16:41:51 +02:00
for ( i = 0 ; i < 8 ; i + + ) {
2014-12-06 05:54:00 +00:00
if ( ret & BIT ( i ) )
2011-08-30 12:41:06 +01:00
iio_push_event ( indio_dev ,
2011-05-18 14:41:15 +01:00
i & 0x1 ?
2011-09-27 09:56:41 +01:00
IIO_UNMOD_EVENT_CODE ( IIO_VOLTAGE ,
2011-08-12 17:08:43 +01:00
( i > > 1 ) ,
IIO_EV_TYPE_THRESH ,
IIO_EV_DIR_RISING ) :
2011-09-27 09:56:41 +01:00
IIO_UNMOD_EVENT_CODE ( IIO_VOLTAGE ,
2011-08-12 17:08:43 +01:00
( i > > 1 ) ,
IIO_EV_TYPE_THRESH ,
IIO_EV_DIR_FALLING ) ,
2016-03-09 19:05:49 +01:00
iio_get_time_ns ( indio_dev ) ) ;
2010-10-01 16:41:51 +02:00
}
2012-07-04 17:09:00 +01:00
done :
2011-05-18 14:41:15 +01:00
return IRQ_HANDLED ;
2010-10-01 16:41:51 +02:00
}
static IIO_DEV_ATTR_SAMP_FREQ ( S_IWUSR | S_IRUGO ,
ad799x_read_frequency ,
ad799x_write_frequency ) ;
static IIO_CONST_ATTR_SAMP_FREQ_AVAIL ( " 15625 7812 3906 1953 976 488 244 0 " ) ;
2013-10-07 15:11:00 +01:00
static struct attribute * ad799x_event_attributes [ ] = {
2010-10-01 16:41:51 +02:00
& iio_dev_attr_sampling_frequency . dev_attr . attr ,
& iio_const_attr_sampling_frequency_available . dev_attr . attr ,
NULL ,
} ;
2017-03-29 01:37:45 +05:30
static const struct attribute_group ad799x_event_attrs_group = {
2013-10-07 15:11:00 +01:00
. attrs = ad799x_event_attributes ,
2010-10-01 16:41:51 +02:00
} ;
2011-05-18 14:42:37 +01:00
static const struct iio_info ad7991_info = {
. read_raw = & ad799x_read_raw ,
2016-07-11 13:54:17 +02:00
. update_scan_mode = ad799x_update_scan_mode ,
2011-05-18 14:42:37 +01:00
} ;
2014-12-06 05:54:00 +00:00
static const struct iio_info ad7993_4_7_8_noirq_info = {
. read_raw = & ad799x_read_raw ,
2014-12-06 05:54:00 +00:00
. update_scan_mode = ad799x_update_scan_mode ,
2014-12-06 05:54:00 +00:00
} ;
static const struct iio_info ad7993_4_7_8_irq_info = {
2011-05-18 14:42:37 +01:00
. read_raw = & ad799x_read_raw ,
2013-10-07 15:11:00 +01:00
. event_attrs = & ad799x_event_attrs_group ,
2013-12-07 10:45:00 +00:00
. read_event_config = & ad799x_read_event_config ,
2014-12-06 05:54:00 +00:00
. write_event_config = & ad799x_write_event_config ,
2013-12-07 10:45:00 +00:00
. read_event_value = & ad799x_read_event_value ,
. write_event_value = & ad799x_write_event_value ,
2014-12-06 05:54:00 +00:00
. update_scan_mode = ad799x_update_scan_mode ,
2011-05-18 14:42:37 +01:00
} ;
2013-10-07 15:11:00 +01:00
static const struct iio_event_spec ad799x_events [ ] = {
{
. type = IIO_EV_TYPE_THRESH ,
. dir = IIO_EV_DIR_RISING ,
. mask_separate = BIT ( IIO_EV_INFO_VALUE ) |
BIT ( IIO_EV_INFO_ENABLE ) ,
} , {
. type = IIO_EV_TYPE_THRESH ,
. dir = IIO_EV_DIR_FALLING ,
2014-08-02 00:07:00 +01:00
. mask_separate = BIT ( IIO_EV_INFO_VALUE ) |
2013-10-07 15:11:00 +01:00
BIT ( IIO_EV_INFO_ENABLE ) ,
2013-10-07 15:11:00 +01:00
} , {
. type = IIO_EV_TYPE_THRESH ,
. dir = IIO_EV_DIR_EITHER ,
. mask_separate = BIT ( IIO_EV_INFO_HYSTERESIS ) ,
2013-10-07 15:11:00 +01:00
} ,
} ;
2011-09-30 10:05:35 +01:00
2013-10-07 15:11:00 +01:00
# define _AD799X_CHANNEL(_index, _realbits, _ev_spec, _num_ev_spec) { \
2013-03-26 18:43:00 +00:00
. type = IIO_VOLTAGE , \
. indexed = 1 , \
. channel = ( _index ) , \
. info_mask_separate = BIT ( IIO_CHAN_INFO_RAW ) , \
2013-03-26 18:43:00 +00:00
. info_mask_shared_by_type = BIT ( IIO_CHAN_INFO_SCALE ) , \
2013-03-26 18:43:00 +00:00
. scan_index = ( _index ) , \
2013-12-11 18:45:00 +00:00
. scan_type = { \
. sign = ' u ' , \
. realbits = ( _realbits ) , \
. storagebits = 16 , \
. shift = 12 - ( _realbits ) , \
. endianness = IIO_BE , \
} , \
2013-10-07 15:11:00 +01:00
. event_spec = _ev_spec , \
. num_event_specs = _num_ev_spec , \
2013-03-26 18:43:00 +00:00
}
2013-10-07 15:11:00 +01:00
# define AD799X_CHANNEL(_index, _realbits) \
_AD799X_CHANNEL ( _index , _realbits , NULL , 0 )
# define AD799X_CHANNEL_WITH_EVENTS(_index, _realbits) \
_AD799X_CHANNEL ( _index , _realbits , ad799x_events , \
ARRAY_SIZE ( ad799x_events ) )
2010-10-01 16:41:51 +02:00
static const struct ad799x_chip_info ad799x_chip_info_tbl [ ] = {
[ ad7991 ] = {
2011-05-18 14:41:52 +01:00
. num_channels = 5 ,
2014-12-06 05:54:00 +00:00
. noirq_config = {
. channel = {
AD799X_CHANNEL ( 0 , 12 ) ,
AD799X_CHANNEL ( 1 , 12 ) ,
AD799X_CHANNEL ( 2 , 12 ) ,
AD799X_CHANNEL ( 3 , 12 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 4 ) ,
} ,
. info = & ad7991_info ,
} ,
2010-10-01 16:41:51 +02:00
} ,
[ ad7995 ] = {
2011-05-18 14:41:52 +01:00
. num_channels = 5 ,
2014-12-06 05:54:00 +00:00
. noirq_config = {
. channel = {
AD799X_CHANNEL ( 0 , 10 ) ,
AD799X_CHANNEL ( 1 , 10 ) ,
AD799X_CHANNEL ( 2 , 10 ) ,
AD799X_CHANNEL ( 3 , 10 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 4 ) ,
} ,
. info = & ad7991_info ,
} ,
2010-10-01 16:41:51 +02:00
} ,
[ ad7999 ] = {
2011-05-18 14:41:52 +01:00
. num_channels = 5 ,
2014-12-06 05:54:00 +00:00
. noirq_config = {
. channel = {
AD799X_CHANNEL ( 0 , 8 ) ,
AD799X_CHANNEL ( 1 , 8 ) ,
AD799X_CHANNEL ( 2 , 8 ) ,
AD799X_CHANNEL ( 3 , 8 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 4 ) ,
} ,
. info = & ad7991_info ,
} ,
2010-10-01 16:41:51 +02:00
} ,
[ ad7992 ] = {
2011-05-18 14:41:52 +01:00
. num_channels = 3 ,
2014-12-06 05:54:00 +00:00
. noirq_config = {
. channel = {
AD799X_CHANNEL ( 0 , 12 ) ,
AD799X_CHANNEL ( 1 , 12 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 3 ) ,
} ,
. info = & ad7993_4_7_8_noirq_info ,
} ,
. irq_config = {
. channel = {
AD799X_CHANNEL_WITH_EVENTS ( 0 , 12 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 1 , 12 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 3 ) ,
} ,
2014-12-06 05:54:00 +00:00
. default_config = AD7998_ALERT_EN | AD7998_BUSY_ALERT ,
2014-12-06 05:54:00 +00:00
. info = & ad7993_4_7_8_irq_info ,
} ,
2010-10-01 16:41:51 +02:00
} ,
[ ad7993 ] = {
2011-05-18 14:41:52 +01:00
. num_channels = 5 ,
2014-12-06 05:54:00 +00:00
. noirq_config = {
. channel = {
AD799X_CHANNEL ( 0 , 10 ) ,
AD799X_CHANNEL ( 1 , 10 ) ,
AD799X_CHANNEL ( 2 , 10 ) ,
AD799X_CHANNEL ( 3 , 10 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 4 ) ,
} ,
. info = & ad7993_4_7_8_noirq_info ,
} ,
. irq_config = {
. channel = {
AD799X_CHANNEL_WITH_EVENTS ( 0 , 10 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 1 , 10 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 2 , 10 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 3 , 10 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 4 ) ,
} ,
2014-12-06 05:54:00 +00:00
. default_config = AD7998_ALERT_EN | AD7998_BUSY_ALERT ,
2014-12-06 05:54:00 +00:00
. info = & ad7993_4_7_8_irq_info ,
} ,
2010-10-01 16:41:51 +02:00
} ,
[ ad7994 ] = {
2011-05-18 14:41:52 +01:00
. num_channels = 5 ,
2014-12-06 05:54:00 +00:00
. noirq_config = {
. channel = {
AD799X_CHANNEL ( 0 , 12 ) ,
AD799X_CHANNEL ( 1 , 12 ) ,
AD799X_CHANNEL ( 2 , 12 ) ,
AD799X_CHANNEL ( 3 , 12 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 4 ) ,
} ,
. info = & ad7993_4_7_8_noirq_info ,
} ,
. irq_config = {
. channel = {
AD799X_CHANNEL_WITH_EVENTS ( 0 , 12 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 1 , 12 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 2 , 12 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 3 , 12 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 4 ) ,
} ,
2014-12-06 05:54:00 +00:00
. default_config = AD7998_ALERT_EN | AD7998_BUSY_ALERT ,
2014-12-06 05:54:00 +00:00
. info = & ad7993_4_7_8_irq_info ,
} ,
2010-10-01 16:41:51 +02:00
} ,
[ ad7997 ] = {
2011-05-18 14:41:52 +01:00
. num_channels = 9 ,
2014-12-06 05:54:00 +00:00
. noirq_config = {
. channel = {
AD799X_CHANNEL ( 0 , 10 ) ,
AD799X_CHANNEL ( 1 , 10 ) ,
AD799X_CHANNEL ( 2 , 10 ) ,
AD799X_CHANNEL ( 3 , 10 ) ,
AD799X_CHANNEL ( 4 , 10 ) ,
AD799X_CHANNEL ( 5 , 10 ) ,
AD799X_CHANNEL ( 6 , 10 ) ,
AD799X_CHANNEL ( 7 , 10 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 8 ) ,
} ,
. info = & ad7993_4_7_8_noirq_info ,
} ,
. irq_config = {
. channel = {
AD799X_CHANNEL_WITH_EVENTS ( 0 , 10 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 1 , 10 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 2 , 10 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 3 , 10 ) ,
AD799X_CHANNEL ( 4 , 10 ) ,
AD799X_CHANNEL ( 5 , 10 ) ,
AD799X_CHANNEL ( 6 , 10 ) ,
AD799X_CHANNEL ( 7 , 10 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 8 ) ,
} ,
2014-12-06 05:54:00 +00:00
. default_config = AD7998_ALERT_EN | AD7998_BUSY_ALERT ,
2014-12-06 05:54:00 +00:00
. info = & ad7993_4_7_8_irq_info ,
} ,
2010-10-01 16:41:51 +02:00
} ,
[ ad7998 ] = {
2011-05-18 14:41:52 +01:00
. num_channels = 9 ,
2014-12-06 05:54:00 +00:00
. noirq_config = {
. channel = {
AD799X_CHANNEL ( 0 , 12 ) ,
AD799X_CHANNEL ( 1 , 12 ) ,
AD799X_CHANNEL ( 2 , 12 ) ,
AD799X_CHANNEL ( 3 , 12 ) ,
AD799X_CHANNEL ( 4 , 12 ) ,
AD799X_CHANNEL ( 5 , 12 ) ,
AD799X_CHANNEL ( 6 , 12 ) ,
AD799X_CHANNEL ( 7 , 12 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 8 ) ,
} ,
. info = & ad7993_4_7_8_noirq_info ,
} ,
. irq_config = {
. channel = {
AD799X_CHANNEL_WITH_EVENTS ( 0 , 12 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 1 , 12 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 2 , 12 ) ,
AD799X_CHANNEL_WITH_EVENTS ( 3 , 12 ) ,
AD799X_CHANNEL ( 4 , 12 ) ,
AD799X_CHANNEL ( 5 , 12 ) ,
AD799X_CHANNEL ( 6 , 12 ) ,
AD799X_CHANNEL ( 7 , 12 ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 8 ) ,
} ,
2014-12-06 05:54:00 +00:00
. default_config = AD7998_ALERT_EN | AD7998_BUSY_ALERT ,
2014-12-06 05:54:00 +00:00
. info = & ad7993_4_7_8_irq_info ,
} ,
2010-10-01 16:41:51 +02:00
} ,
} ;
2022-11-18 23:36:35 +01:00
static int ad799x_probe ( struct i2c_client * client )
2010-10-01 16:41:51 +02:00
{
2022-11-18 23:36:35 +01:00
const struct i2c_device_id * id = i2c_client_get_device_id ( client ) ;
2011-09-02 17:14:40 +01:00
int ret ;
2021-09-30 12:42:48 +02:00
int extra_config = 0 ;
2011-05-18 14:41:53 +01:00
struct ad799x_state * st ;
2013-08-31 18:12:00 +01:00
struct iio_dev * indio_dev ;
2014-12-06 05:54:00 +00:00
const struct ad799x_chip_info * chip_info =
& ad799x_chip_info_tbl [ id - > driver_data ] ;
2011-05-18 14:41:53 +01:00
2013-08-31 18:12:00 +01:00
indio_dev = devm_iio_device_alloc ( & client - > dev , sizeof ( * st ) ) ;
2011-05-18 14:41:53 +01:00
if ( indio_dev = = NULL )
return - ENOMEM ;
2010-10-01 16:41:51 +02:00
2011-05-18 14:41:53 +01:00
st = iio_priv ( indio_dev ) ;
2010-10-01 16:41:51 +02:00
/* this is only used for device removal purposes */
2011-05-18 14:41:53 +01:00
i2c_set_clientdata ( client , indio_dev ) ;
2010-10-01 16:41:51 +02:00
st - > id = id - > driver_data ;
2014-12-06 05:54:00 +00:00
if ( client - > irq > 0 & & chip_info - > irq_config . info )
st - > chip_config = & chip_info - > irq_config ;
else
st - > chip_config = & chip_info - > noirq_config ;
2010-10-01 16:41:51 +02:00
/* TODO: Add pdata options for filtering and bit delay */
2013-08-31 18:12:00 +01:00
st - > reg = devm_regulator_get ( & client - > dev , " vcc " ) ;
2014-02-16 12:02:00 +00:00
if ( IS_ERR ( st - > reg ) )
return PTR_ERR ( st - > reg ) ;
ret = regulator_enable ( st - > reg ) ;
if ( ret )
return ret ;
2021-09-30 12:42:48 +02:00
/* check if an external reference is supplied */
st - > vref = devm_regulator_get_optional ( & client - > dev , " vref " ) ;
2014-02-16 12:02:00 +00:00
if ( IS_ERR ( st - > vref ) ) {
2021-09-30 12:42:48 +02:00
if ( PTR_ERR ( st - > vref ) = = - ENODEV ) {
st - > vref = NULL ;
dev_info ( & client - > dev , " Using VCC reference voltage \n " ) ;
} else {
ret = PTR_ERR ( st - > vref ) ;
goto error_disable_reg ;
}
}
if ( st - > vref ) {
/*
* Use external reference voltage if supported by hardware .
* This is optional if voltage / regulator present , use VCC otherwise .
*/
if ( ( st - > id = = ad7991 ) | | ( st - > id = = ad7995 ) | | ( st - > id = = ad7999 ) ) {
dev_info ( & client - > dev , " Using external reference voltage \n " ) ;
extra_config | = AD7991_REF_SEL ;
ret = regulator_enable ( st - > vref ) ;
if ( ret )
goto error_disable_reg ;
} else {
st - > vref = NULL ;
dev_warn ( & client - > dev , " Supplied reference not supported \n " ) ;
}
2010-10-01 16:41:51 +02:00
}
2014-02-16 12:02:00 +00:00
2010-10-01 16:41:51 +02:00
st - > client = client ;
2011-05-18 14:41:53 +01:00
indio_dev - > name = id - > name ;
2014-12-06 05:54:00 +00:00
indio_dev - > info = st - > chip_config - > info ;
2011-05-18 14:42:37 +01:00
2011-05-18 14:41:53 +01:00
indio_dev - > modes = INDIO_DIRECT_MODE ;
2014-12-06 05:54:00 +00:00
indio_dev - > channels = st - > chip_config - > channel ;
indio_dev - > num_channels = chip_info - > num_channels ;
2011-05-18 14:41:53 +01:00
2021-09-30 12:42:48 +02:00
ret = ad799x_update_config ( st , st - > chip_config - > default_config | extra_config ) ;
2019-11-29 17:53:14 +01:00
if ( ret )
2019-09-17 18:09:23 +02:00
goto error_disable_vref ;
2014-12-06 05:54:00 +00:00
2014-10-03 10:31:00 +01:00
ret = iio_triggered_buffer_setup ( indio_dev , NULL ,
& ad799x_trigger_handler , NULL ) ;
2010-10-01 16:41:51 +02:00
if ( ret )
2014-04-30 22:05:00 +01:00
goto error_disable_vref ;
2010-10-01 16:41:51 +02:00
2011-05-18 14:42:37 +01:00
if ( client - > irq > 0 ) {
2014-01-01 23:04:00 +00:00
ret = devm_request_threaded_irq ( & client - > dev ,
client - > irq ,
NULL ,
ad799x_event_handler ,
IRQF_TRIGGER_FALLING |
IRQF_ONESHOT ,
client - > name ,
indio_dev ) ;
2010-10-01 16:41:51 +02:00
if ( ret )
goto error_cleanup_ring ;
}
2022-10-04 15:48:54 +02:00
mutex_init ( & st - > lock ) ;
2011-09-02 17:14:40 +01:00
ret = iio_device_register ( indio_dev ) ;
if ( ret )
2014-01-01 23:04:00 +00:00
goto error_cleanup_ring ;
2010-10-01 16:41:51 +02:00
return 0 ;
2011-05-18 14:41:53 +01:00
2010-10-01 16:41:51 +02:00
error_cleanup_ring :
2014-10-03 10:31:00 +01:00
iio_triggered_buffer_cleanup ( indio_dev ) ;
2014-04-30 22:05:00 +01:00
error_disable_vref :
2021-09-30 12:42:48 +02:00
if ( st - > vref )
regulator_disable ( st - > vref ) ;
2010-10-01 16:41:51 +02:00
error_disable_reg :
2014-04-30 22:05:00 +01:00
regulator_disable ( st - > reg ) ;
2011-05-18 14:41:53 +01:00
2010-10-01 16:41:51 +02:00
return ret ;
}
2022-08-15 10:02:30 +02:00
static void ad799x_remove ( struct i2c_client * client )
2010-10-01 16:41:51 +02:00
{
2011-05-18 14:41:53 +01:00
struct iio_dev * indio_dev = i2c_get_clientdata ( client ) ;
struct ad799x_state * st = iio_priv ( indio_dev ) ;
2010-10-01 16:41:51 +02:00
2011-10-14 14:46:58 +01:00
iio_device_unregister ( indio_dev ) ;
2010-10-01 16:41:51 +02:00
2014-10-03 10:31:00 +01:00
iio_triggered_buffer_cleanup ( indio_dev ) ;
2021-09-30 12:42:48 +02:00
if ( st - > vref )
regulator_disable ( st - > vref ) ;
2014-04-30 22:05:00 +01:00
regulator_disable ( st - > reg ) ;
2013-03-26 18:43:00 +00:00
kfree ( st - > rx_buf ) ;
2010-10-01 16:41:51 +02:00
}
2022-06-21 21:26:45 +01:00
static int ad799x_suspend ( struct device * dev )
2019-11-29 17:53:14 +01:00
{
struct iio_dev * indio_dev = i2c_get_clientdata ( to_i2c_client ( dev ) ) ;
struct ad799x_state * st = iio_priv ( indio_dev ) ;
2021-09-30 12:42:48 +02:00
if ( st - > vref )
regulator_disable ( st - > vref ) ;
2019-11-29 17:53:14 +01:00
regulator_disable ( st - > reg ) ;
return 0 ;
}
2022-06-21 21:26:45 +01:00
static int ad799x_resume ( struct device * dev )
2019-11-29 17:53:14 +01:00
{
struct iio_dev * indio_dev = i2c_get_clientdata ( to_i2c_client ( dev ) ) ;
struct ad799x_state * st = iio_priv ( indio_dev ) ;
int ret ;
ret = regulator_enable ( st - > reg ) ;
if ( ret ) {
dev_err ( dev , " Unable to enable vcc regulator \n " ) ;
return ret ;
}
2021-09-30 12:42:48 +02:00
if ( st - > vref ) {
ret = regulator_enable ( st - > vref ) ;
if ( ret ) {
regulator_disable ( st - > reg ) ;
dev_err ( dev , " Unable to enable vref regulator \n " ) ;
return ret ;
}
2019-11-29 17:53:14 +01:00
}
/* resync config */
ret = ad799x_update_config ( st , st - > config ) ;
if ( ret ) {
2021-09-30 12:42:48 +02:00
if ( st - > vref )
regulator_disable ( st - > vref ) ;
2019-11-29 17:53:14 +01:00
regulator_disable ( st - > reg ) ;
return ret ;
}
return 0 ;
}
2022-06-21 21:26:45 +01:00
static DEFINE_SIMPLE_DEV_PM_OPS ( ad799x_pm_ops , ad799x_suspend , ad799x_resume ) ;
2019-11-29 17:53:14 +01:00
2010-10-01 16:41:51 +02:00
static const struct i2c_device_id ad799x_id [ ] = {
{ " ad7991 " , ad7991 } ,
{ " ad7995 " , ad7995 } ,
{ " ad7999 " , ad7999 } ,
{ " ad7992 " , ad7992 } ,
{ " ad7993 " , ad7993 } ,
{ " ad7994 " , ad7994 } ,
{ " ad7997 " , ad7997 } ,
{ " ad7998 " , ad7998 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , ad799x_id ) ;
static struct i2c_driver ad799x_driver = {
. driver = {
. name = " ad799x " ,
2022-06-21 21:26:45 +01:00
. pm = pm_sleep_ptr ( & ad799x_pm_ops ) ,
2010-10-01 16:41:51 +02:00
} ,
2023-05-15 22:50:48 +02:00
. probe = ad799x_probe ,
2012-11-19 13:21:38 -05:00
. remove = ad799x_remove ,
2010-10-01 16:41:51 +02:00
. id_table = ad799x_id ,
} ;
2011-11-16 10:13:38 +01:00
module_i2c_driver ( ad799x_driver ) ;
2010-10-01 16:41:51 +02:00
2018-08-14 13:23:17 +02:00
MODULE_AUTHOR ( " Michael Hennerich <michael.hennerich@analog.com> " ) ;
2010-10-01 16:41:51 +02:00
MODULE_DESCRIPTION ( " Analog Devices AD799x ADC " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;