2013-05-03 10:58:00 +01:00
/*
* Copyright ( C ) 2013 Oskar Andero < oskar . andero @ gmail . com >
2014-10-08 20:42:00 +02:00
* Copyright ( C ) 2014 Rose Technology
* Allan Bendorff Jensen < abj @ rosetechnology . dk >
* Soren Andersen < san @ rosetechnology . dk >
*
* Driver for following ADC chips from Microchip Technology ' s :
* 10 Bit converter
* MCP3001
* MCP3002
* MCP3004
* MCP3008
* - - - - - - - - - - - -
* 12 bit converter
* MCP3201
* MCP3202
* MCP3204
* MCP3208
* - - - - - - - - - - - -
2013-05-03 10:58:00 +01:00
*
* Datasheet can be found here :
2014-10-08 20:42:00 +02:00
* http : //ww1.microchip.com/downloads/en/DeviceDoc/21293C.pdf mcp3001
* http : //ww1.microchip.com/downloads/en/DeviceDoc/21294E.pdf mcp3002
* http : //ww1.microchip.com/downloads/en/DeviceDoc/21295d.pdf mcp3004/08
* http : //ww1.microchip.com/downloads/en/DeviceDoc/21290D.pdf mcp3201
* http : //ww1.microchip.com/downloads/en/DeviceDoc/21034D.pdf mcp3202
* http : //ww1.microchip.com/downloads/en/DeviceDoc/21298c.pdf mcp3204/08
2015-07-14 15:36:21 +02:00
* http : //ww1.microchip.com/downloads/en/DeviceDoc/21700E.pdf mcp3301
2013-05-03 10:58:00 +01: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/err.h>
2014-10-08 20:42:00 +02:00
# include <linux/delay.h>
2013-05-03 10:58:00 +01:00
# include <linux/spi/spi.h>
# include <linux/module.h>
# include <linux/iio/iio.h>
# include <linux/regulator/consumer.h>
enum {
2014-10-08 20:42:00 +02:00
mcp3001 ,
mcp3002 ,
mcp3004 ,
mcp3008 ,
mcp3201 ,
mcp3202 ,
2013-05-03 10:58:00 +01:00
mcp3204 ,
mcp3208 ,
2015-07-14 15:36:21 +02:00
mcp3301 ,
2013-05-03 10:58:00 +01:00
} ;
2014-10-08 20:42:00 +02:00
struct mcp320x_chip_info {
const struct iio_chan_spec * channels ;
unsigned int num_channels ;
unsigned int resolution ;
} ;
2013-05-03 10:58:00 +01:00
struct mcp320x {
struct spi_device * spi ;
struct spi_message msg ;
struct spi_transfer transfer [ 2 ] ;
struct regulator * reg ;
struct mutex lock ;
2014-10-08 20:42:00 +02:00
const struct mcp320x_chip_info * chip_info ;
2015-05-06 11:49:17 -05:00
u8 tx_buf ____cacheline_aligned ;
u8 rx_buf [ 2 ] ;
2013-05-03 10:58:00 +01:00
} ;
2014-10-08 20:42:00 +02:00
static int mcp320x_channel_to_tx_data ( int device_index ,
const unsigned int channel , bool differential )
{
int start_bit = 1 ;
switch ( device_index ) {
case mcp3002 :
case mcp3202 :
return ( ( start_bit < < 4 ) | ( ! differential < < 3 ) |
( channel < < 2 ) ) ;
case mcp3004 :
case mcp3204 :
case mcp3008 :
case mcp3208 :
return ( ( start_bit < < 6 ) | ( ! differential < < 5 ) |
( channel < < 2 ) ) ;
default :
return - EINVAL ;
}
}
static int mcp320x_adc_conversion ( struct mcp320x * adc , u8 channel ,
bool differential , int device_index )
2013-05-03 10:58:00 +01:00
{
int ret ;
2017-09-09 20:32:41 +02:00
memset ( & adc - > rx_buf , 0 , sizeof ( adc - > rx_buf ) ) ;
if ( adc - > chip_info - > num_channels > 1 )
adc - > tx_buf = mcp320x_channel_to_tx_data ( device_index , channel ,
differential ) ;
2014-10-08 20:42:00 +02:00
2017-09-09 20:32:41 +02:00
ret = spi_sync ( adc - > spi , & adc - > msg ) ;
if ( ret < 0 )
return ret ;
2013-05-03 10:58:00 +01:00
2014-10-08 20:42:00 +02:00
switch ( device_index ) {
case mcp3001 :
return ( adc - > rx_buf [ 0 ] < < 5 | adc - > rx_buf [ 1 ] > > 3 ) ;
case mcp3002 :
case mcp3004 :
case mcp3008 :
return ( adc - > rx_buf [ 0 ] < < 2 | adc - > rx_buf [ 1 ] > > 6 ) ;
case mcp3201 :
return ( adc - > rx_buf [ 0 ] < < 7 | adc - > rx_buf [ 1 ] > > 1 ) ;
case mcp3202 :
case mcp3204 :
case mcp3208 :
return ( adc - > rx_buf [ 0 ] < < 4 | adc - > rx_buf [ 1 ] > > 4 ) ;
2015-07-14 15:36:21 +02:00
case mcp3301 :
return sign_extend32 ( ( adc - > rx_buf [ 0 ] & 0x1f ) < < 8 | adc - > rx_buf [ 1 ] , 12 ) ;
2014-10-08 20:42:00 +02:00
default :
return - EINVAL ;
}
2013-05-03 10:58:00 +01:00
}
static int mcp320x_read_raw ( struct iio_dev * indio_dev ,
struct iio_chan_spec const * channel , int * val ,
int * val2 , long mask )
{
struct mcp320x * adc = iio_priv ( indio_dev ) ;
int ret = - EINVAL ;
2014-10-08 20:42:00 +02:00
int device_index = 0 ;
2013-05-03 10:58:00 +01:00
mutex_lock ( & adc - > lock ) ;
2014-10-08 20:42:00 +02:00
device_index = spi_get_device_id ( adc - > spi ) - > driver_data ;
2013-05-03 10:58:00 +01:00
switch ( mask ) {
case IIO_CHAN_INFO_RAW :
2014-10-08 20:42:00 +02:00
ret = mcp320x_adc_conversion ( adc , channel - > address ,
channel - > differential , device_index ) ;
2013-05-03 10:58:00 +01:00
if ( ret < 0 )
goto out ;
* val = ret ;
ret = IIO_VAL_INT ;
break ;
case IIO_CHAN_INFO_SCALE :
ret = regulator_get_voltage ( adc - > reg ) ;
if ( ret < 0 )
goto out ;
2014-10-08 20:42:00 +02:00
/* convert regulator output voltage to mV */
2013-05-03 10:58:00 +01:00
* val = ret / 1000 ;
2014-10-08 20:42:00 +02:00
* val2 = adc - > chip_info - > resolution ;
2013-05-03 10:58:00 +01:00
ret = IIO_VAL_FRACTIONAL_LOG2 ;
break ;
}
out :
mutex_unlock ( & adc - > lock ) ;
return ret ;
}
# define MCP320X_VOLTAGE_CHANNEL(num) \
{ \
. type = IIO_VOLTAGE , \
. indexed = 1 , \
. channel = ( num ) , \
. address = ( num ) , \
. info_mask_separate = BIT ( IIO_CHAN_INFO_RAW ) , \
. info_mask_shared_by_type = BIT ( IIO_CHAN_INFO_SCALE ) \
}
iio: adc: mcp320x: support more differential voltage measurement
mcp320x driver supports the pseudo-differential mode by
in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (0, 1), (2, 3), ...
mcp320x chips except MCP3X01 can also select swapped IN+ and IN-
pairs in the pseudo-differential mode.
i.e. in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (1, 0),
(3, 2), ...
If the voltage level of IN+ is equal to or less than IN-, the
resultant code will be 000h. So it is useful to provide these, too.
Signed-off-by: Akinobu Mita <akinobu.mita@gmail.com>
Cc: Oskar Andero <oskar.andero@gmail.com>
Cc: Jonathan Cameron <jic23@kernel.org>
Cc: Hartmut Knaack <knaack.h@gmx.de>
Cc: Lars-Peter Clausen <lars@metafoo.de>
Cc: Peter Meerwald <pmeerw@pmeerw.net>
Cc: linux-iio@vger.kernel.org
Signed-off-by: Jonathan Cameron <jic23@kernel.org>
2016-01-08 00:40:30 +09:00
# define MCP320X_VOLTAGE_CHANNEL_DIFF(chan1, chan2) \
2013-05-03 10:58:00 +01:00
{ \
. type = IIO_VOLTAGE , \
. indexed = 1 , \
iio: adc: mcp320x: support more differential voltage measurement
mcp320x driver supports the pseudo-differential mode by
in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (0, 1), (2, 3), ...
mcp320x chips except MCP3X01 can also select swapped IN+ and IN-
pairs in the pseudo-differential mode.
i.e. in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (1, 0),
(3, 2), ...
If the voltage level of IN+ is equal to or less than IN-, the
resultant code will be 000h. So it is useful to provide these, too.
Signed-off-by: Akinobu Mita <akinobu.mita@gmail.com>
Cc: Oskar Andero <oskar.andero@gmail.com>
Cc: Jonathan Cameron <jic23@kernel.org>
Cc: Hartmut Knaack <knaack.h@gmx.de>
Cc: Lars-Peter Clausen <lars@metafoo.de>
Cc: Peter Meerwald <pmeerw@pmeerw.net>
Cc: linux-iio@vger.kernel.org
Signed-off-by: Jonathan Cameron <jic23@kernel.org>
2016-01-08 00:40:30 +09:00
. channel = ( chan1 ) , \
. channel2 = ( chan2 ) , \
. address = ( chan1 ) , \
2013-05-03 10:58:00 +01:00
. differential = 1 , \
. info_mask_separate = BIT ( IIO_CHAN_INFO_RAW ) , \
. info_mask_shared_by_type = BIT ( IIO_CHAN_INFO_SCALE ) \
}
2014-10-08 20:42:00 +02:00
static const struct iio_chan_spec mcp3201_channels [ ] = {
iio: adc: mcp320x: support more differential voltage measurement
mcp320x driver supports the pseudo-differential mode by
in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (0, 1), (2, 3), ...
mcp320x chips except MCP3X01 can also select swapped IN+ and IN-
pairs in the pseudo-differential mode.
i.e. in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (1, 0),
(3, 2), ...
If the voltage level of IN+ is equal to or less than IN-, the
resultant code will be 000h. So it is useful to provide these, too.
Signed-off-by: Akinobu Mita <akinobu.mita@gmail.com>
Cc: Oskar Andero <oskar.andero@gmail.com>
Cc: Jonathan Cameron <jic23@kernel.org>
Cc: Hartmut Knaack <knaack.h@gmx.de>
Cc: Lars-Peter Clausen <lars@metafoo.de>
Cc: Peter Meerwald <pmeerw@pmeerw.net>
Cc: linux-iio@vger.kernel.org
Signed-off-by: Jonathan Cameron <jic23@kernel.org>
2016-01-08 00:40:30 +09:00
MCP320X_VOLTAGE_CHANNEL_DIFF ( 0 , 1 ) ,
2014-10-08 20:42:00 +02:00
} ;
static const struct iio_chan_spec mcp3202_channels [ ] = {
MCP320X_VOLTAGE_CHANNEL ( 0 ) ,
MCP320X_VOLTAGE_CHANNEL ( 1 ) ,
iio: adc: mcp320x: support more differential voltage measurement
mcp320x driver supports the pseudo-differential mode by
in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (0, 1), (2, 3), ...
mcp320x chips except MCP3X01 can also select swapped IN+ and IN-
pairs in the pseudo-differential mode.
i.e. in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (1, 0),
(3, 2), ...
If the voltage level of IN+ is equal to or less than IN-, the
resultant code will be 000h. So it is useful to provide these, too.
Signed-off-by: Akinobu Mita <akinobu.mita@gmail.com>
Cc: Oskar Andero <oskar.andero@gmail.com>
Cc: Jonathan Cameron <jic23@kernel.org>
Cc: Hartmut Knaack <knaack.h@gmx.de>
Cc: Lars-Peter Clausen <lars@metafoo.de>
Cc: Peter Meerwald <pmeerw@pmeerw.net>
Cc: linux-iio@vger.kernel.org
Signed-off-by: Jonathan Cameron <jic23@kernel.org>
2016-01-08 00:40:30 +09:00
MCP320X_VOLTAGE_CHANNEL_DIFF ( 0 , 1 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 1 , 0 ) ,
2014-10-08 20:42:00 +02:00
} ;
2013-05-03 10:58:00 +01:00
static const struct iio_chan_spec mcp3204_channels [ ] = {
MCP320X_VOLTAGE_CHANNEL ( 0 ) ,
MCP320X_VOLTAGE_CHANNEL ( 1 ) ,
MCP320X_VOLTAGE_CHANNEL ( 2 ) ,
MCP320X_VOLTAGE_CHANNEL ( 3 ) ,
iio: adc: mcp320x: support more differential voltage measurement
mcp320x driver supports the pseudo-differential mode by
in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (0, 1), (2, 3), ...
mcp320x chips except MCP3X01 can also select swapped IN+ and IN-
pairs in the pseudo-differential mode.
i.e. in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (1, 0),
(3, 2), ...
If the voltage level of IN+ is equal to or less than IN-, the
resultant code will be 000h. So it is useful to provide these, too.
Signed-off-by: Akinobu Mita <akinobu.mita@gmail.com>
Cc: Oskar Andero <oskar.andero@gmail.com>
Cc: Jonathan Cameron <jic23@kernel.org>
Cc: Hartmut Knaack <knaack.h@gmx.de>
Cc: Lars-Peter Clausen <lars@metafoo.de>
Cc: Peter Meerwald <pmeerw@pmeerw.net>
Cc: linux-iio@vger.kernel.org
Signed-off-by: Jonathan Cameron <jic23@kernel.org>
2016-01-08 00:40:30 +09:00
MCP320X_VOLTAGE_CHANNEL_DIFF ( 0 , 1 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 1 , 0 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 2 , 3 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 3 , 2 ) ,
2013-05-03 10:58:00 +01:00
} ;
static const struct iio_chan_spec mcp3208_channels [ ] = {
MCP320X_VOLTAGE_CHANNEL ( 0 ) ,
MCP320X_VOLTAGE_CHANNEL ( 1 ) ,
MCP320X_VOLTAGE_CHANNEL ( 2 ) ,
MCP320X_VOLTAGE_CHANNEL ( 3 ) ,
MCP320X_VOLTAGE_CHANNEL ( 4 ) ,
MCP320X_VOLTAGE_CHANNEL ( 5 ) ,
MCP320X_VOLTAGE_CHANNEL ( 6 ) ,
MCP320X_VOLTAGE_CHANNEL ( 7 ) ,
iio: adc: mcp320x: support more differential voltage measurement
mcp320x driver supports the pseudo-differential mode by
in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (0, 1), (2, 3), ...
mcp320x chips except MCP3X01 can also select swapped IN+ and IN-
pairs in the pseudo-differential mode.
i.e. in_voltage'IN+'-voltage'IN-'_raw where (IN+, IN-) = (1, 0),
(3, 2), ...
If the voltage level of IN+ is equal to or less than IN-, the
resultant code will be 000h. So it is useful to provide these, too.
Signed-off-by: Akinobu Mita <akinobu.mita@gmail.com>
Cc: Oskar Andero <oskar.andero@gmail.com>
Cc: Jonathan Cameron <jic23@kernel.org>
Cc: Hartmut Knaack <knaack.h@gmx.de>
Cc: Lars-Peter Clausen <lars@metafoo.de>
Cc: Peter Meerwald <pmeerw@pmeerw.net>
Cc: linux-iio@vger.kernel.org
Signed-off-by: Jonathan Cameron <jic23@kernel.org>
2016-01-08 00:40:30 +09:00
MCP320X_VOLTAGE_CHANNEL_DIFF ( 0 , 1 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 1 , 0 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 2 , 3 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 3 , 2 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 4 , 5 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 5 , 4 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 6 , 7 ) ,
MCP320X_VOLTAGE_CHANNEL_DIFF ( 7 , 6 ) ,
2013-05-03 10:58:00 +01:00
} ;
static const struct iio_info mcp320x_info = {
. read_raw = mcp320x_read_raw ,
} ;
2014-10-08 20:42:00 +02:00
static const struct mcp320x_chip_info mcp320x_chip_infos [ ] = {
[ mcp3001 ] = {
. channels = mcp3201_channels ,
. num_channels = ARRAY_SIZE ( mcp3201_channels ) ,
. resolution = 10
} ,
[ mcp3002 ] = {
. channels = mcp3202_channels ,
. num_channels = ARRAY_SIZE ( mcp3202_channels ) ,
. resolution = 10
} ,
[ mcp3004 ] = {
. channels = mcp3204_channels ,
. num_channels = ARRAY_SIZE ( mcp3204_channels ) ,
. resolution = 10
} ,
[ mcp3008 ] = {
. channels = mcp3208_channels ,
. num_channels = ARRAY_SIZE ( mcp3208_channels ) ,
. resolution = 10
} ,
[ mcp3201 ] = {
. channels = mcp3201_channels ,
. num_channels = ARRAY_SIZE ( mcp3201_channels ) ,
. resolution = 12
} ,
[ mcp3202 ] = {
. channels = mcp3202_channels ,
. num_channels = ARRAY_SIZE ( mcp3202_channels ) ,
. resolution = 12
} ,
2013-05-03 10:58:00 +01:00
[ mcp3204 ] = {
. channels = mcp3204_channels ,
2014-10-08 20:42:00 +02:00
. num_channels = ARRAY_SIZE ( mcp3204_channels ) ,
. resolution = 12
2013-05-03 10:58:00 +01:00
} ,
[ mcp3208 ] = {
. channels = mcp3208_channels ,
2014-10-08 20:42:00 +02:00
. num_channels = ARRAY_SIZE ( mcp3208_channels ) ,
. resolution = 12
2013-05-03 10:58:00 +01:00
} ,
2015-07-14 15:36:21 +02:00
[ mcp3301 ] = {
. channels = mcp3201_channels ,
. num_channels = ARRAY_SIZE ( mcp3201_channels ) ,
. resolution = 13
} ,
2013-05-03 10:58:00 +01:00
} ;
static int mcp320x_probe ( struct spi_device * spi )
{
struct iio_dev * indio_dev ;
struct mcp320x * adc ;
2014-10-08 20:42:00 +02:00
const struct mcp320x_chip_info * chip_info ;
2013-05-03 10:58:00 +01:00
int ret ;
2013-07-23 09:58:00 +01:00
indio_dev = devm_iio_device_alloc ( & spi - > dev , sizeof ( * adc ) ) ;
2013-05-03 10:58:00 +01:00
if ( ! indio_dev )
return - ENOMEM ;
adc = iio_priv ( indio_dev ) ;
adc - > spi = spi ;
indio_dev - > dev . parent = & spi - > dev ;
2016-07-02 17:26:33 -07:00
indio_dev - > dev . of_node = spi - > dev . of_node ;
2013-05-03 10:58:00 +01:00
indio_dev - > name = spi_get_device_id ( spi ) - > name ;
indio_dev - > modes = INDIO_DIRECT_MODE ;
indio_dev - > info = & mcp320x_info ;
2014-10-08 20:42:00 +02:00
chip_info = & mcp320x_chip_infos [ spi_get_device_id ( spi ) - > driver_data ] ;
2013-05-03 10:58:00 +01:00
indio_dev - > channels = chip_info - > channels ;
indio_dev - > num_channels = chip_info - > num_channels ;
2015-07-10 22:55:30 +02:00
adc - > chip_info = chip_info ;
2013-05-03 10:58:00 +01:00
adc - > transfer [ 0 ] . tx_buf = & adc - > tx_buf ;
adc - > transfer [ 0 ] . len = sizeof ( adc - > tx_buf ) ;
adc - > transfer [ 1 ] . rx_buf = adc - > rx_buf ;
adc - > transfer [ 1 ] . len = sizeof ( adc - > rx_buf ) ;
2017-09-09 20:32:41 +02:00
if ( chip_info - > num_channels = = 1 )
/* single-channel converters are rx only (no MOSI pin) */
spi_message_init_with_transfers ( & adc - > msg ,
& adc - > transfer [ 1 ] , 1 ) ;
else
spi_message_init_with_transfers ( & adc - > msg , adc - > transfer ,
ARRAY_SIZE ( adc - > transfer ) ) ;
2013-05-03 10:58:00 +01:00
2013-07-23 09:58:00 +01:00
adc - > reg = devm_regulator_get ( & spi - > dev , " vref " ) ;
if ( IS_ERR ( adc - > reg ) )
return PTR_ERR ( adc - > reg ) ;
2013-05-03 10:58:00 +01:00
ret = regulator_enable ( adc - > reg ) ;
if ( ret < 0 )
2013-07-23 09:58:00 +01:00
return ret ;
2013-05-03 10:58:00 +01:00
mutex_init ( & adc - > lock ) ;
ret = iio_device_register ( indio_dev ) ;
if ( ret < 0 )
goto reg_disable ;
return 0 ;
reg_disable :
regulator_disable ( adc - > reg ) ;
return ret ;
}
static int mcp320x_remove ( struct spi_device * spi )
{
struct iio_dev * indio_dev = spi_get_drvdata ( spi ) ;
struct mcp320x * adc = iio_priv ( indio_dev ) ;
iio_device_unregister ( indio_dev ) ;
regulator_disable ( adc - > reg ) ;
return 0 ;
}
2014-10-08 20:42:00 +02:00
# if defined(CONFIG_OF)
static const struct of_device_id mcp320x_dt_ids [ ] = {
2015-10-14 14:54:39 +02:00
/* NOTE: The use of compatibles with no vendor prefix is deprecated. */
2017-09-09 20:32:41 +02:00
{ . compatible = " mcp3001 " } ,
{ . compatible = " mcp3002 " } ,
{ . compatible = " mcp3004 " } ,
{ . compatible = " mcp3008 " } ,
{ . compatible = " mcp3201 " } ,
{ . compatible = " mcp3202 " } ,
{ . compatible = " mcp3204 " } ,
{ . compatible = " mcp3208 " } ,
{ . compatible = " mcp3301 " } ,
{ . compatible = " microchip,mcp3001 " } ,
{ . compatible = " microchip,mcp3002 " } ,
{ . compatible = " microchip,mcp3004 " } ,
{ . compatible = " microchip,mcp3008 " } ,
{ . compatible = " microchip,mcp3201 " } ,
{ . compatible = " microchip,mcp3202 " } ,
{ . compatible = " microchip,mcp3204 " } ,
{ . compatible = " microchip,mcp3208 " } ,
{ . compatible = " microchip,mcp3301 " } ,
{ }
2014-10-08 20:42:00 +02:00
} ;
MODULE_DEVICE_TABLE ( of , mcp320x_dt_ids ) ;
# endif
2013-05-03 10:58:00 +01:00
static const struct spi_device_id mcp320x_id [ ] = {
2014-10-08 20:42:00 +02:00
{ " mcp3001 " , mcp3001 } ,
{ " mcp3002 " , mcp3002 } ,
{ " mcp3004 " , mcp3004 } ,
{ " mcp3008 " , mcp3008 } ,
{ " mcp3201 " , mcp3201 } ,
{ " mcp3202 " , mcp3202 } ,
2013-05-03 10:58:00 +01:00
{ " mcp3204 " , mcp3204 } ,
{ " mcp3208 " , mcp3208 } ,
2015-07-14 15:36:21 +02:00
{ " mcp3301 " , mcp3301 } ,
2013-05-03 10:58:00 +01:00
{ }
} ;
MODULE_DEVICE_TABLE ( spi , mcp320x_id ) ;
static struct spi_driver mcp320x_driver = {
. driver = {
. name = " mcp320x " ,
2015-08-20 09:07:26 +02:00
. of_match_table = of_match_ptr ( mcp320x_dt_ids ) ,
2013-05-03 10:58:00 +01:00
} ,
. probe = mcp320x_probe ,
. remove = mcp320x_remove ,
. id_table = mcp320x_id ,
} ;
module_spi_driver ( mcp320x_driver ) ;
MODULE_AUTHOR ( " Oskar Andero <oskar.andero@gmail.com> " ) ;
2014-10-08 20:42:00 +02:00
MODULE_DESCRIPTION ( " Microchip Technology MCP3x01/02/04/08 " ) ;
2013-05-03 10:58:00 +01:00
MODULE_LICENSE ( " GPL v2 " ) ;