2010-11-22 14:35:32 +01:00
/*
* AD7887 SPI ADC driver
*
2011-05-18 14:41:50 +01:00
* Copyright 2010 - 2011 Analog Devices Inc .
2010-11-22 14:35:32 +01:00
*
2011-05-18 14:41:50 +01:00
* Licensed under the GPL - 2.
2010-11-22 14:35:32 +01:00
*/
# include <linux/device.h>
# include <linux/kernel.h>
# include <linux/slab.h>
# include <linux/sysfs.h>
# include <linux/spi/spi.h>
# include <linux/regulator/consumer.h>
# include <linux/err.h>
2011-07-03 15:49:50 -04:00
# include <linux/module.h>
2012-11-05 09:56:00 +00:00
# include <linux/interrupt.h>
2014-12-06 06:00:00 +00:00
# include <linux/bitops.h>
2010-11-22 14:35:32 +01:00
2012-04-25 15:54:58 +01:00
# include <linux/iio/iio.h>
# include <linux/iio/sysfs.h>
# include <linux/iio/buffer.h>
2011-08-12 17:08:43 +01:00
2012-11-05 09:56:00 +00:00
# include <linux/iio/trigger_consumer.h>
# include <linux/iio/triggered_buffer.h>
2010-11-22 14:35:32 +01:00
2012-11-05 09:56:00 +00:00
# include <linux/platform_data/ad7887.h>
2010-11-22 14:35:32 +01:00
2014-12-06 06:00:00 +00:00
# define AD7887_REF_DIS BIT(5) /* on-chip reference disable */
# define AD7887_DUAL BIT(4) /* dual-channel mode */
# define AD7887_CH_AIN1 BIT(3) /* convert on channel 1, DUAL=1 */
# define AD7887_CH_AIN0 0 /* convert on channel 0, DUAL=0,1 */
# define AD7887_PM_MODE1 0 /* CS based shutdown */
# define AD7887_PM_MODE2 1 /* full on */
# define AD7887_PM_MODE3 2 /* auto shutdown after conversion */
# define AD7887_PM_MODE4 3 /* standby mode */
2012-11-05 09:56:00 +00:00
enum ad7887_channels {
AD7887_CH0 ,
AD7887_CH0_CH1 ,
AD7887_CH1 ,
} ;
/**
* struct ad7887_chip_info - chip specifc information
* @ int_vref_mv : the internal reference voltage
* @ channel : channel specification
*/
struct ad7887_chip_info {
u16 int_vref_mv ;
struct iio_chan_spec channel [ 3 ] ;
} ;
struct ad7887_state {
struct spi_device * spi ;
const struct ad7887_chip_info * chip_info ;
struct regulator * reg ;
struct spi_transfer xfer [ 4 ] ;
struct spi_message msg [ 3 ] ;
struct spi_message * ring_msg ;
2012-11-05 09:56:00 +00:00
unsigned char tx_cmd_buf [ 4 ] ;
2012-11-05 09:56:00 +00:00
/*
* DMA ( thus cache coherency maintenance ) requires the
* transfer buffers to live in their own cache lines .
* Buffer needs to be large enough to hold two 16 bit samples and a
* 64 bit aligned 64 bit timestamp .
*/
unsigned char data [ ALIGN ( 4 , sizeof ( s64 ) ) + sizeof ( s64 ) ]
____cacheline_aligned ;
} ;
enum ad7887_supported_device_ids {
ID_AD7887
} ;
static int ad7887_ring_preenable ( struct iio_dev * indio_dev )
{
struct ad7887_state * st = iio_priv ( indio_dev ) ;
/* We know this is a single long so can 'cheat' */
switch ( * indio_dev - > active_scan_mask ) {
case ( 1 < < 0 ) :
st - > ring_msg = & st - > msg [ AD7887_CH0 ] ;
break ;
case ( 1 < < 1 ) :
st - > ring_msg = & st - > msg [ AD7887_CH1 ] ;
/* Dummy read: push CH1 setting down to hardware */
spi_sync ( st - > spi , st - > ring_msg ) ;
break ;
case ( ( 1 < < 1 ) | ( 1 < < 0 ) ) :
st - > ring_msg = & st - > msg [ AD7887_CH0_CH1 ] ;
break ;
}
return 0 ;
}
static int ad7887_ring_postdisable ( struct iio_dev * indio_dev )
{
struct ad7887_state * st = iio_priv ( indio_dev ) ;
/* dummy read: restore default CH0 settin */
return spi_sync ( st - > spi , & st - > msg [ AD7887_CH0 ] ) ;
}
/**
* ad7887_trigger_handler ( ) bh of trigger launched polling to ring buffer
*
* Currently there is no option in this driver to disable the saving of
* timestamps within the ring .
* */
static irqreturn_t ad7887_trigger_handler ( int irq , void * p )
{
struct iio_poll_func * pf = p ;
struct iio_dev * indio_dev = pf - > indio_dev ;
struct ad7887_state * st = iio_priv ( indio_dev ) ;
int b_sent ;
b_sent = spi_sync ( st - > spi , st - > ring_msg ) ;
if ( b_sent )
goto done ;
2013-09-19 13:59:00 +01:00
iio_push_to_buffers_with_timestamp ( indio_dev , st - > data ,
2016-03-09 19:05:49 +01:00
iio_get_time_ns ( indio_dev ) ) ;
2012-11-05 09:56:00 +00:00
done :
iio_trigger_notify_done ( indio_dev - > trig ) ;
return IRQ_HANDLED ;
}
static const struct iio_buffer_setup_ops ad7887_ring_setup_ops = {
. preenable = & ad7887_ring_preenable ,
. postenable = & iio_triggered_buffer_postenable ,
. predisable = & iio_triggered_buffer_predisable ,
. postdisable = & ad7887_ring_postdisable ,
} ;
2010-11-22 14:35:32 +01:00
static int ad7887_scan_direct ( struct ad7887_state * st , unsigned ch )
{
int ret = spi_sync ( st - > spi , & st - > msg [ ch ] ) ;
if ( ret )
return ret ;
return ( st - > data [ ( ch * 2 ) ] < < 8 ) | st - > data [ ( ch * 2 ) + 1 ] ;
}
2011-10-06 17:14:37 +01:00
static int ad7887_read_raw ( struct iio_dev * indio_dev ,
2011-05-18 14:41:50 +01:00
struct iio_chan_spec const * chan ,
int * val ,
int * val2 ,
long m )
2010-11-22 14:35:32 +01:00
{
int ret ;
2011-10-06 17:14:37 +01:00
struct ad7887_state * st = iio_priv ( indio_dev ) ;
2010-11-22 14:35:32 +01:00
2011-05-18 14:41:50 +01:00
switch ( m ) {
2012-04-15 17:41:18 +01:00
case IIO_CHAN_INFO_RAW :
2016-05-24 12:18:43 -07:00
ret = iio_device_claim_direct_mode ( indio_dev ) ;
if ( ret )
return ret ;
ret = ad7887_scan_direct ( st , chan - > address ) ;
iio_device_release_direct_mode ( indio_dev ) ;
2011-05-18 14:41:50 +01:00
if ( ret < 0 )
return ret ;
2012-11-05 09:56:00 +00:00
* val = ret > > chan - > scan_type . shift ;
2014-12-06 06:00:00 +00:00
* val & = GENMASK ( chan - > scan_type . realbits - 1 , 0 ) ;
2011-05-18 14:41:50 +01:00
return IIO_VAL_INT ;
2011-10-26 17:41:36 +01:00
case IIO_CHAN_INFO_SCALE :
2012-11-05 09:56:00 +00:00
if ( st - > reg ) {
* val = regulator_get_voltage ( st - > reg ) ;
if ( * val < 0 )
return * val ;
* val / = 1000 ;
} else {
* val = st - > chip_info - > int_vref_mv ;
}
2012-11-05 09:56:00 +00:00
* val2 = chan - > scan_type . realbits ;
2012-11-05 09:56:00 +00:00
return IIO_VAL_FRACTIONAL_LOG2 ;
2011-05-18 14:41:50 +01:00
}
return - EINVAL ;
2010-11-22 14:35:32 +01:00
}
static const struct ad7887_chip_info ad7887_chip_info_tbl [ ] = {
/*
* More devices added in future
*/
[ ID_AD7887 ] = {
2011-09-30 10:05:44 +01:00
. channel [ 0 ] = {
. type = IIO_VOLTAGE ,
. indexed = 1 ,
. channel = 1 ,
2013-02-27 19:05:59 +00:00
. info_mask_separate = BIT ( IIO_CHAN_INFO_RAW ) ,
. info_mask_shared_by_type = BIT ( IIO_CHAN_INFO_SCALE ) ,
2011-09-30 10:05:44 +01:00
. address = 1 ,
. scan_index = 1 ,
2013-12-11 18:45:00 +00:00
. scan_type = {
. sign = ' u ' ,
. realbits = 12 ,
. storagebits = 16 ,
. shift = 0 ,
. endianness = IIO_BE ,
} ,
2011-09-30 10:05:44 +01:00
} ,
. channel [ 1 ] = {
. type = IIO_VOLTAGE ,
. indexed = 1 ,
. channel = 0 ,
2013-02-27 19:05:59 +00:00
. info_mask_separate = BIT ( IIO_CHAN_INFO_RAW ) ,
. info_mask_shared_by_type = BIT ( IIO_CHAN_INFO_SCALE ) ,
2011-09-30 10:05:44 +01:00
. address = 0 ,
. scan_index = 0 ,
2013-12-11 18:45:00 +00:00
. scan_type = {
. sign = ' u ' ,
. realbits = 12 ,
. storagebits = 16 ,
. shift = 0 ,
. endianness = IIO_BE ,
} ,
2011-09-30 10:05:44 +01:00
} ,
2011-05-18 14:41:50 +01:00
. channel [ 2 ] = IIO_CHAN_SOFT_TIMESTAMP ( 2 ) ,
2010-11-22 14:35:32 +01:00
. int_vref_mv = 2500 ,
} ,
} ;
2011-05-18 14:42:37 +01:00
static const struct iio_info ad7887_info = {
. read_raw = & ad7887_read_raw ,
. driver_module = THIS_MODULE ,
} ;
2012-12-21 13:21:43 -08:00
static int ad7887_probe ( struct spi_device * spi )
2010-11-22 14:35:32 +01:00
{
struct ad7887_platform_data * pdata = spi - > dev . platform_data ;
struct ad7887_state * st ;
2013-07-23 09:58:00 +01:00
struct iio_dev * indio_dev ;
2012-11-05 09:56:00 +00:00
uint8_t mode ;
2012-11-05 09:56:00 +00:00
int ret ;
2010-11-22 14:35:32 +01:00
2013-07-23 09:58:00 +01:00
indio_dev = devm_iio_device_alloc ( & spi - > dev , sizeof ( * st ) ) ;
2011-05-18 14:41:51 +01:00
if ( indio_dev = = NULL )
return - ENOMEM ;
st = iio_priv ( indio_dev ) ;
2010-11-22 14:35:32 +01:00
2012-11-05 09:56:00 +00:00
if ( ! pdata | | ! pdata - > use_onchip_ref ) {
2013-07-23 09:58:00 +01:00
st - > reg = devm_regulator_get ( & spi - > dev , " vref " ) ;
if ( IS_ERR ( st - > reg ) )
return PTR_ERR ( st - > reg ) ;
2012-11-05 09:56:00 +00:00
2010-11-22 14:35:32 +01:00
ret = regulator_enable ( st - > reg ) ;
if ( ret )
2013-07-23 09:58:00 +01:00
return ret ;
2010-11-22 14:35:32 +01:00
}
st - > chip_info =
& ad7887_chip_info_tbl [ spi_get_device_id ( spi ) - > driver_data ] ;
2011-05-18 14:41:51 +01:00
spi_set_drvdata ( spi , indio_dev ) ;
2010-11-22 14:35:32 +01:00
st - > spi = spi ;
/* Estabilish that the iio_dev is a child of the spi device */
2011-05-18 14:41:51 +01:00
indio_dev - > dev . parent = & spi - > dev ;
2016-07-02 17:26:33 -07:00
indio_dev - > dev . of_node = spi - > dev . of_node ;
2011-05-18 14:41:51 +01:00
indio_dev - > name = spi_get_device_id ( spi ) - > name ;
2011-05-18 14:42:37 +01:00
indio_dev - > info = & ad7887_info ;
2011-05-18 14:41:51 +01:00
indio_dev - > modes = INDIO_DIRECT_MODE ;
2010-11-22 14:35:32 +01:00
/* Setup default message */
2012-11-05 09:56:00 +00:00
mode = AD7887_PM_MODE4 ;
if ( ! pdata | | ! pdata - > use_onchip_ref )
mode | = AD7887_REF_DIS ;
if ( pdata & & pdata - > en_dual )
mode | = AD7887_DUAL ;
st - > tx_cmd_buf [ 0 ] = AD7887_CH_AIN0 | mode ;
2010-11-22 14:35:32 +01:00
st - > xfer [ 0 ] . rx_buf = & st - > data [ 0 ] ;
st - > xfer [ 0 ] . tx_buf = & st - > tx_cmd_buf [ 0 ] ;
st - > xfer [ 0 ] . len = 2 ;
spi_message_init ( & st - > msg [ AD7887_CH0 ] ) ;
spi_message_add_tail ( & st - > xfer [ 0 ] , & st - > msg [ AD7887_CH0 ] ) ;
if ( pdata & & pdata - > en_dual ) {
2012-11-05 09:56:00 +00:00
st - > tx_cmd_buf [ 2 ] = AD7887_CH_AIN1 | mode ;
2010-11-22 14:35:32 +01:00
st - > xfer [ 1 ] . rx_buf = & st - > data [ 0 ] ;
st - > xfer [ 1 ] . tx_buf = & st - > tx_cmd_buf [ 2 ] ;
st - > xfer [ 1 ] . len = 2 ;
st - > xfer [ 2 ] . rx_buf = & st - > data [ 2 ] ;
2012-11-05 09:56:00 +00:00
st - > xfer [ 2 ] . tx_buf = & st - > tx_cmd_buf [ 0 ] ;
2010-11-22 14:35:32 +01:00
st - > xfer [ 2 ] . len = 2 ;
spi_message_init ( & st - > msg [ AD7887_CH0_CH1 ] ) ;
spi_message_add_tail ( & st - > xfer [ 1 ] , & st - > msg [ AD7887_CH0_CH1 ] ) ;
spi_message_add_tail ( & st - > xfer [ 2 ] , & st - > msg [ AD7887_CH0_CH1 ] ) ;
2012-11-05 09:56:00 +00:00
st - > xfer [ 3 ] . rx_buf = & st - > data [ 2 ] ;
st - > xfer [ 3 ] . tx_buf = & st - > tx_cmd_buf [ 2 ] ;
2010-11-22 14:35:32 +01:00
st - > xfer [ 3 ] . len = 2 ;
spi_message_init ( & st - > msg [ AD7887_CH1 ] ) ;
spi_message_add_tail ( & st - > xfer [ 3 ] , & st - > msg [ AD7887_CH1 ] ) ;
2011-05-18 14:41:51 +01:00
indio_dev - > channels = st - > chip_info - > channel ;
indio_dev - > num_channels = 3 ;
2010-11-22 14:35:32 +01:00
} else {
2011-05-18 14:41:51 +01:00
indio_dev - > channels = & st - > chip_info - > channel [ 1 ] ;
indio_dev - > num_channels = 2 ;
2011-05-18 14:41:50 +01:00
}
2010-11-22 14:35:32 +01:00
2012-11-05 09:56:00 +00:00
ret = iio_triggered_buffer_setup ( indio_dev , & iio_pollfunc_store_time ,
& ad7887_trigger_handler , & ad7887_ring_setup_ops ) ;
2010-11-22 14:35:32 +01:00
if ( ret )
2011-05-18 14:41:51 +01:00
goto error_disable_reg ;
2010-11-22 14:35:32 +01:00
2011-09-02 17:14:40 +01:00
ret = iio_device_register ( indio_dev ) ;
if ( ret )
goto error_unregister_ring ;
return 0 ;
error_unregister_ring :
2012-11-05 09:56:00 +00:00
iio_triggered_buffer_cleanup ( indio_dev ) ;
2010-11-22 14:35:32 +01:00
error_disable_reg :
2012-11-05 09:56:00 +00:00
if ( st - > reg )
2010-11-22 14:35:32 +01:00
regulator_disable ( st - > reg ) ;
2011-05-18 14:41:51 +01:00
2010-11-22 14:35:32 +01:00
return ret ;
}
2012-12-21 13:21:43 -08:00
static int ad7887_remove ( struct spi_device * spi )
2010-11-22 14:35:32 +01:00
{
2011-05-18 14:41:51 +01:00
struct iio_dev * indio_dev = spi_get_drvdata ( spi ) ;
struct ad7887_state * st = iio_priv ( indio_dev ) ;
2011-10-14 14:46:58 +01:00
iio_device_unregister ( indio_dev ) ;
2012-11-05 09:56:00 +00:00
iio_triggered_buffer_cleanup ( indio_dev ) ;
2013-07-23 09:58:00 +01:00
if ( st - > reg )
2010-11-22 14:35:32 +01:00
regulator_disable ( st - > reg ) ;
2011-05-18 14:41:51 +01:00
2010-11-22 14:35:32 +01:00
return 0 ;
}
static const struct spi_device_id ad7887_id [ ] = {
{ " ad7887 " , ID_AD7887 } ,
{ }
} ;
2011-11-16 08:53:31 +01:00
MODULE_DEVICE_TABLE ( spi , ad7887_id ) ;
2010-11-22 14:35:32 +01:00
static struct spi_driver ad7887_driver = {
. driver = {
. name = " ad7887 " ,
} ,
. probe = ad7887_probe ,
2012-12-21 13:21:43 -08:00
. remove = ad7887_remove ,
2010-11-22 14:35:32 +01:00
. id_table = ad7887_id ,
} ;
2011-11-16 10:13:39 +01:00
module_spi_driver ( ad7887_driver ) ;
2010-11-22 14:35:32 +01:00
MODULE_AUTHOR ( " Michael Hennerich <hennerich@blackfin.uclinux.org> " ) ;
MODULE_DESCRIPTION ( " Analog Devices AD7887 ADC " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;