2012-02-15 19:48:01 +00:00
/* The industrial I/O core in kernel channel mapping
*
* Copyright ( c ) 2011 Jonathan Cameron
*
* 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>
# include <linux/export.h>
# include <linux/slab.h>
# include <linux/mutex.h>
2013-02-07 17:09:00 +00:00
# include <linux/of.h>
2012-02-15 19:48:01 +00:00
2012-04-25 15:54:58 +01:00
# include <linux/iio/iio.h>
2012-02-15 19:48:01 +00:00
# include "iio_core.h"
2012-04-25 15:54:58 +01:00
# include <linux/iio/machine.h>
# include <linux/iio/driver.h>
# include <linux/iio/consumer.h>
2012-02-15 19:48:01 +00:00
struct iio_map_internal {
struct iio_dev * indio_dev ;
struct iio_map * map ;
struct list_head l ;
} ;
static LIST_HEAD ( iio_map_list ) ;
static DEFINE_MUTEX ( iio_map_list_lock ) ;
int iio_map_array_register ( struct iio_dev * indio_dev , struct iio_map * maps )
{
int i = 0 , ret = 0 ;
struct iio_map_internal * mapi ;
if ( maps = = NULL )
return 0 ;
mutex_lock ( & iio_map_list_lock ) ;
while ( maps [ i ] . consumer_dev_name ! = NULL ) {
mapi = kzalloc ( sizeof ( * mapi ) , GFP_KERNEL ) ;
if ( mapi = = NULL ) {
ret = - ENOMEM ;
goto error_ret ;
}
mapi - > map = & maps [ i ] ;
mapi - > indio_dev = indio_dev ;
2017-06-27 09:46:01 -07:00
list_add_tail ( & mapi - > l , & iio_map_list ) ;
2012-02-15 19:48:01 +00:00
i + + ;
}
error_ret :
mutex_unlock ( & iio_map_list_lock ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_map_array_register ) ;
2013-01-31 21:43:00 +00:00
/*
* Remove all map entries associated with the given iio device
2012-02-15 19:48:01 +00:00
*/
2013-01-31 21:43:00 +00:00
int iio_map_array_unregister ( struct iio_dev * indio_dev )
2012-02-15 19:48:01 +00:00
{
2013-01-31 21:43:00 +00:00
int ret = - ENODEV ;
2015-12-08 09:45:56 -08:00
struct iio_map_internal * mapi , * next ;
2012-02-15 19:48:01 +00:00
mutex_lock ( & iio_map_list_lock ) ;
2015-12-08 09:45:56 -08:00
list_for_each_entry_safe ( mapi , next , & iio_map_list , l ) {
2013-01-31 21:43:00 +00:00
if ( indio_dev = = mapi - > indio_dev ) {
list_del ( & mapi - > l ) ;
kfree ( mapi ) ;
ret = 0 ;
2012-02-15 19:48:01 +00:00
}
}
mutex_unlock ( & iio_map_list_lock ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_map_array_unregister ) ;
static const struct iio_chan_spec
2012-05-01 21:04:24 +01:00
* iio_chan_spec_from_name ( const struct iio_dev * indio_dev , const char * name )
2012-02-15 19:48:01 +00:00
{
int i ;
const struct iio_chan_spec * chan = NULL ;
for ( i = 0 ; i < indio_dev - > num_channels ; i + + )
if ( indio_dev - > channels [ i ] . datasheet_name & &
strcmp ( name , indio_dev - > channels [ i ] . datasheet_name ) = = 0 ) {
chan = & indio_dev - > channels [ i ] ;
break ;
}
return chan ;
}
2013-02-07 17:09:00 +00:00
# ifdef CONFIG_OF
static int iio_dev_node_match ( struct device * dev , void * data )
{
return dev - > of_node = = data & & dev - > type = = & iio_device_type ;
}
2014-10-22 18:29:43 +03:00
/**
* __of_iio_simple_xlate - translate iiospec to the IIO channel index
* @ indio_dev : pointer to the iio_dev structure
* @ iiospec : IIO specifier as found in the device tree
*
* This is simple translation function , suitable for the most 1 : 1 mapped
* channels in IIO chips . This function performs only one sanity check :
* whether IIO index is less than num_channels ( that is specified in the
* iio_dev ) .
*/
static int __of_iio_simple_xlate ( struct iio_dev * indio_dev ,
const struct of_phandle_args * iiospec )
{
if ( ! iiospec - > args_count )
return 0 ;
2015-01-01 18:13:24 +00:00
if ( iiospec - > args [ 0 ] > = indio_dev - > num_channels ) {
dev_err ( & indio_dev - > dev , " invalid channel index %u \n " ,
iiospec - > args [ 0 ] ) ;
2014-10-22 18:29:43 +03:00
return - EINVAL ;
2015-01-01 18:13:24 +00:00
}
2014-10-22 18:29:43 +03:00
return iiospec - > args [ 0 ] ;
}
2013-02-07 17:09:00 +00:00
static int __of_iio_channel_get ( struct iio_channel * channel ,
struct device_node * np , int index )
{
struct device * idev ;
struct iio_dev * indio_dev ;
int err ;
struct of_phandle_args iiospec ;
err = of_parse_phandle_with_args ( np , " io-channels " ,
" #io-channel-cells " ,
index , & iiospec ) ;
if ( err )
return err ;
idev = bus_find_device ( & iio_bus_type , NULL , iiospec . np ,
iio_dev_node_match ) ;
of_node_put ( iiospec . np ) ;
if ( idev = = NULL )
return - EPROBE_DEFER ;
indio_dev = dev_to_iio_dev ( idev ) ;
channel - > indio_dev = indio_dev ;
2014-10-22 18:29:43 +03:00
if ( indio_dev - > info - > of_xlate )
index = indio_dev - > info - > of_xlate ( indio_dev , & iiospec ) ;
else
index = __of_iio_simple_xlate ( indio_dev , & iiospec ) ;
if ( index < 0 )
2013-02-07 17:09:00 +00:00
goto err_put ;
channel - > channel = & indio_dev - > channels [ index ] ;
return 0 ;
err_put :
iio_device_put ( indio_dev ) ;
2014-10-22 18:29:43 +03:00
return index ;
2013-02-07 17:09:00 +00:00
}
static struct iio_channel * of_iio_channel_get ( struct device_node * np , int index )
{
struct iio_channel * channel ;
int err ;
if ( index < 0 )
return ERR_PTR ( - EINVAL ) ;
channel = kzalloc ( sizeof ( * channel ) , GFP_KERNEL ) ;
if ( channel = = NULL )
return ERR_PTR ( - ENOMEM ) ;
err = __of_iio_channel_get ( channel , np , index ) ;
if ( err )
goto err_free_channel ;
return channel ;
err_free_channel :
kfree ( channel ) ;
return ERR_PTR ( err ) ;
}
static struct iio_channel * of_iio_channel_get_by_name ( struct device_node * np ,
const char * name )
{
struct iio_channel * chan = NULL ;
/* Walk up the tree of devices looking for a matching iio channel */
while ( np ) {
int index = 0 ;
/*
* For named iio channels , first look up the name in the
* " io-channel-names " property . If it cannot be found , the
* index will be an error code , and of_iio_channel_get ( )
* will fail .
*/
if ( name )
index = of_property_match_string ( np , " io-channel-names " ,
name ) ;
chan = of_iio_channel_get ( np , index ) ;
2014-08-25 09:04:00 +01:00
if ( ! IS_ERR ( chan ) | | PTR_ERR ( chan ) = = - EPROBE_DEFER )
2013-02-07 17:09:00 +00:00
break ;
else if ( name & & index > = 0 ) {
2017-07-18 16:43:08 -05:00
pr_err ( " ERROR: could not get IIO channel %pOF:%s(%i) \n " ,
np , name ? name : " " , index ) ;
2014-11-06 12:11:00 +00:00
return NULL ;
2013-02-07 17:09:00 +00:00
}
/*
* No matching IIO channel found on this node .
* If the parent node has a " io-channel-ranges " property ,
* then we can try one of its channels .
*/
np = np - > parent ;
if ( np & & ! of_get_property ( np , " io-channel-ranges " , NULL ) )
2014-11-06 12:11:00 +00:00
return NULL ;
2013-02-07 17:09:00 +00:00
}
2014-11-06 12:11:00 +00:00
2013-02-07 17:09:00 +00:00
return chan ;
}
static struct iio_channel * of_iio_channel_get_all ( struct device * dev )
{
struct iio_channel * chans ;
int i , mapind , nummaps = 0 ;
int ret ;
do {
ret = of_parse_phandle_with_args ( dev - > of_node ,
" io-channels " ,
" #io-channel-cells " ,
nummaps , NULL ) ;
if ( ret < 0 )
break ;
} while ( + + nummaps ) ;
if ( nummaps = = 0 ) /* no error, return NULL to search map table */
return NULL ;
/* NULL terminated array to save passing size */
chans = kcalloc ( nummaps + 1 , sizeof ( * chans ) , GFP_KERNEL ) ;
if ( chans = = NULL )
return ERR_PTR ( - ENOMEM ) ;
/* Search for OF matches */
for ( mapind = 0 ; mapind < nummaps ; mapind + + ) {
ret = __of_iio_channel_get ( & chans [ mapind ] , dev - > of_node ,
mapind ) ;
if ( ret )
goto error_free_chans ;
}
return chans ;
error_free_chans :
for ( i = 0 ; i < mapind ; i + + )
iio_device_put ( chans [ i ] . indio_dev ) ;
kfree ( chans ) ;
return ERR_PTR ( ret ) ;
}
# else /* CONFIG_OF */
static inline struct iio_channel *
of_iio_channel_get_by_name ( struct device_node * np , const char * name )
{
return NULL ;
}
static inline struct iio_channel * of_iio_channel_get_all ( struct device * dev )
{
return NULL ;
}
# endif /* CONFIG_OF */
2012-02-15 19:48:01 +00:00
2013-02-04 20:26:00 +00:00
static struct iio_channel * iio_channel_get_sys ( const char * name ,
const char * channel_name )
2012-02-15 19:48:01 +00:00
{
struct iio_map_internal * c_i = NULL , * c = NULL ;
struct iio_channel * channel ;
2012-09-18 05:56:00 +01:00
int err ;
2012-02-15 19:48:01 +00:00
if ( name = = NULL & & channel_name = = NULL )
return ERR_PTR ( - ENODEV ) ;
/* first find matching entry the channel map */
mutex_lock ( & iio_map_list_lock ) ;
list_for_each_entry ( c_i , & iio_map_list , l ) {
if ( ( name & & strcmp ( name , c_i - > map - > consumer_dev_name ) ! = 0 ) | |
( channel_name & &
strcmp ( channel_name , c_i - > map - > consumer_channel ) ! = 0 ) )
continue ;
c = c_i ;
2012-06-04 10:50:03 +02:00
iio_device_get ( c - > indio_dev ) ;
2012-02-15 19:48:01 +00:00
break ;
}
mutex_unlock ( & iio_map_list_lock ) ;
if ( c = = NULL )
return ERR_PTR ( - ENODEV ) ;
2012-09-14 02:24:00 +01:00
channel = kzalloc ( sizeof ( * channel ) , GFP_KERNEL ) ;
2012-09-18 05:56:00 +01:00
if ( channel = = NULL ) {
err = - ENOMEM ;
2012-09-18 05:55:00 +01:00
goto error_no_mem ;
2012-09-18 05:56:00 +01:00
}
2012-02-15 19:48:01 +00:00
channel - > indio_dev = c - > indio_dev ;
2012-09-17 09:44:00 +01:00
if ( c - > map - > adc_channel_label ) {
2012-02-15 19:48:01 +00:00
channel - > channel =
iio_chan_spec_from_name ( channel - > indio_dev ,
c - > map - > adc_channel_label ) ;
2012-09-18 05:56:00 +01:00
if ( channel - > channel = = NULL ) {
err = - EINVAL ;
2012-09-17 09:44:00 +01:00
goto error_no_chan ;
2012-09-18 05:56:00 +01:00
}
2012-09-17 09:44:00 +01:00
}
2012-02-15 19:48:01 +00:00
return channel ;
2012-09-17 09:44:00 +01:00
error_no_chan :
kfree ( channel ) ;
2012-09-18 05:55:00 +01:00
error_no_mem :
iio_device_put ( c - > indio_dev ) ;
2012-09-18 05:56:00 +01:00
return ERR_PTR ( err ) ;
2012-02-15 19:48:01 +00:00
}
2013-02-04 20:26:00 +00:00
struct iio_channel * iio_channel_get ( struct device * dev ,
const char * channel_name )
{
const char * name = dev ? dev_name ( dev ) : NULL ;
2013-02-07 17:09:00 +00:00
struct iio_channel * channel ;
2013-02-04 20:26:00 +00:00
2013-02-07 17:09:00 +00:00
if ( dev ) {
channel = of_iio_channel_get_by_name ( dev - > of_node ,
channel_name ) ;
if ( channel ! = NULL )
return channel ;
}
2014-11-06 12:11:00 +00:00
2013-02-04 20:26:00 +00:00
return iio_channel_get_sys ( name , channel_name ) ;
}
2012-05-01 21:04:24 +01:00
EXPORT_SYMBOL_GPL ( iio_channel_get ) ;
2012-02-15 19:48:01 +00:00
2012-05-01 21:04:24 +01:00
void iio_channel_release ( struct iio_channel * channel )
2012-02-15 19:48:01 +00:00
{
2016-01-26 12:25:21 +03:00
if ( ! channel )
return ;
2012-06-04 10:50:03 +02:00
iio_device_put ( channel - > indio_dev ) ;
2012-02-15 19:48:01 +00:00
kfree ( channel ) ;
}
2012-05-01 21:04:24 +01:00
EXPORT_SYMBOL_GPL ( iio_channel_release ) ;
2012-02-15 19:48:01 +00:00
2016-04-06 16:01:06 +05:30
static void devm_iio_channel_free ( struct device * dev , void * res )
{
struct iio_channel * channel = * ( struct iio_channel * * ) res ;
iio_channel_release ( channel ) ;
}
static int devm_iio_channel_match ( struct device * dev , void * res , void * data )
{
struct iio_channel * * r = res ;
if ( ! r | | ! * r ) {
WARN_ON ( ! r | | ! * r ) ;
return 0 ;
}
return * r = = data ;
}
struct iio_channel * devm_iio_channel_get ( struct device * dev ,
const char * channel_name )
{
struct iio_channel * * ptr , * channel ;
ptr = devres_alloc ( devm_iio_channel_free , sizeof ( * ptr ) , GFP_KERNEL ) ;
if ( ! ptr )
return ERR_PTR ( - ENOMEM ) ;
channel = iio_channel_get ( dev , channel_name ) ;
if ( IS_ERR ( channel ) ) {
devres_free ( ptr ) ;
return channel ;
}
* ptr = channel ;
devres_add ( dev , ptr ) ;
return channel ;
}
EXPORT_SYMBOL_GPL ( devm_iio_channel_get ) ;
void devm_iio_channel_release ( struct device * dev , struct iio_channel * channel )
{
WARN_ON ( devres_release ( dev , devm_iio_channel_free ,
devm_iio_channel_match , channel ) ) ;
}
EXPORT_SYMBOL_GPL ( devm_iio_channel_release ) ;
2013-01-31 21:43:00 +00:00
struct iio_channel * iio_channel_get_all ( struct device * dev )
2012-02-15 19:48:01 +00:00
{
2013-01-31 21:43:00 +00:00
const char * name ;
2012-02-15 19:48:01 +00:00
struct iio_channel * chans ;
struct iio_map_internal * c = NULL ;
int nummaps = 0 ;
int mapind = 0 ;
int i , ret ;
2013-01-31 21:43:00 +00:00
if ( dev = = NULL )
2012-02-15 19:48:01 +00:00
return ERR_PTR ( - EINVAL ) ;
2013-02-07 17:09:00 +00:00
chans = of_iio_channel_get_all ( dev ) ;
if ( chans )
return chans ;
2013-01-31 21:43:00 +00:00
name = dev_name ( dev ) ;
2012-02-15 19:48:01 +00:00
mutex_lock ( & iio_map_list_lock ) ;
/* first count the matching maps */
list_for_each_entry ( c , & iio_map_list , l )
if ( name & & strcmp ( name , c - > map - > consumer_dev_name ) ! = 0 )
continue ;
else
nummaps + + ;
if ( nummaps = = 0 ) {
ret = - ENODEV ;
goto error_ret ;
}
/* NULL terminated array to save passing size */
chans = kzalloc ( sizeof ( * chans ) * ( nummaps + 1 ) , GFP_KERNEL ) ;
if ( chans = = NULL ) {
ret = - ENOMEM ;
goto error_ret ;
}
/* for each map fill in the chans element */
list_for_each_entry ( c , & iio_map_list , l ) {
if ( name & & strcmp ( name , c - > map - > consumer_dev_name ) ! = 0 )
continue ;
chans [ mapind ] . indio_dev = c - > indio_dev ;
2012-06-30 20:06:00 +01:00
chans [ mapind ] . data = c - > map - > consumer_data ;
2012-02-15 19:48:01 +00:00
chans [ mapind ] . channel =
iio_chan_spec_from_name ( chans [ mapind ] . indio_dev ,
c - > map - > adc_channel_label ) ;
if ( chans [ mapind ] . channel = = NULL ) {
ret = - EINVAL ;
goto error_free_chans ;
}
2012-06-04 10:50:03 +02:00
iio_device_get ( chans [ mapind ] . indio_dev ) ;
2012-02-15 19:48:01 +00:00
mapind + + ;
}
if ( mapind = = 0 ) {
ret = - ENODEV ;
goto error_free_chans ;
}
2012-07-11 07:34:00 +01:00
mutex_unlock ( & iio_map_list_lock ) ;
2012-02-15 19:48:01 +00:00
return chans ;
error_free_chans :
for ( i = 0 ; i < nummaps ; i + + )
2012-06-04 10:50:03 +02:00
iio_device_put ( chans [ i ] . indio_dev ) ;
2012-02-15 19:48:01 +00:00
kfree ( chans ) ;
error_ret :
mutex_unlock ( & iio_map_list_lock ) ;
return ERR_PTR ( ret ) ;
}
2012-05-01 21:04:24 +01:00
EXPORT_SYMBOL_GPL ( iio_channel_get_all ) ;
2012-02-15 19:48:01 +00:00
2012-05-01 21:04:24 +01:00
void iio_channel_release_all ( struct iio_channel * channels )
2012-02-15 19:48:01 +00:00
{
struct iio_channel * chan = & channels [ 0 ] ;
while ( chan - > indio_dev ) {
2012-06-04 10:50:03 +02:00
iio_device_put ( chan - > indio_dev ) ;
2012-02-15 19:48:01 +00:00
chan + + ;
}
kfree ( channels ) ;
}
2012-05-01 21:04:24 +01:00
EXPORT_SYMBOL_GPL ( iio_channel_release_all ) ;
2012-02-15 19:48:01 +00:00
2016-04-06 16:01:07 +05:30
static void devm_iio_channel_free_all ( struct device * dev , void * res )
{
struct iio_channel * channels = * ( struct iio_channel * * ) res ;
iio_channel_release_all ( channels ) ;
}
struct iio_channel * devm_iio_channel_get_all ( struct device * dev )
{
struct iio_channel * * ptr , * channels ;
ptr = devres_alloc ( devm_iio_channel_free_all , sizeof ( * ptr ) , GFP_KERNEL ) ;
if ( ! ptr )
return ERR_PTR ( - ENOMEM ) ;
channels = iio_channel_get_all ( dev ) ;
if ( IS_ERR ( channels ) ) {
devres_free ( ptr ) ;
return channels ;
}
* ptr = channels ;
devres_add ( dev , ptr ) ;
return channels ;
}
EXPORT_SYMBOL_GPL ( devm_iio_channel_get_all ) ;
void devm_iio_channel_release_all ( struct device * dev ,
struct iio_channel * channels )
{
WARN_ON ( devres_release ( dev , devm_iio_channel_free_all ,
devm_iio_channel_match , channels ) ) ;
}
EXPORT_SYMBOL_GPL ( devm_iio_channel_release_all ) ;
2012-09-17 13:17:00 +01:00
static int iio_channel_read ( struct iio_channel * chan , int * val , int * val2 ,
enum iio_chan_info_enum info )
{
int unused ;
2014-04-29 00:51:00 +01:00
int vals [ INDIO_MAX_RAW_ELEMENTS ] ;
int ret ;
int val_len = 2 ;
2012-09-17 13:17:00 +01:00
if ( val2 = = NULL )
val2 = & unused ;
2016-04-14 21:36:34 +02:00
if ( ! iio_channel_has_info ( chan - > channel , info ) )
2015-01-01 12:46:48 +00:00
return - EINVAL ;
2014-04-29 00:51:00 +01:00
if ( chan - > indio_dev - > info - > read_raw_multi ) {
ret = chan - > indio_dev - > info - > read_raw_multi ( chan - > indio_dev ,
chan - > channel , INDIO_MAX_RAW_ELEMENTS ,
vals , & val_len , info ) ;
* val = vals [ 0 ] ;
* val2 = vals [ 1 ] ;
} else
ret = chan - > indio_dev - > info - > read_raw ( chan - > indio_dev ,
chan - > channel , val , val2 , info ) ;
return ret ;
2012-09-17 13:17:00 +01:00
}
2012-05-01 21:04:24 +01:00
int iio_read_channel_raw ( struct iio_channel * chan , int * val )
2012-02-15 19:48:01 +00:00
{
2012-09-17 13:17:00 +01:00
int ret ;
2012-02-15 19:48:01 +00:00
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
2012-09-17 13:17:00 +01:00
ret = iio_channel_read ( chan , val , NULL , IIO_CHAN_INFO_RAW ) ;
2012-02-15 19:48:01 +00:00
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
return ret ;
}
2012-05-01 21:04:24 +01:00
EXPORT_SYMBOL_GPL ( iio_read_channel_raw ) ;
2012-02-15 19:48:01 +00:00
2014-10-03 17:25:00 +01:00
int iio_read_channel_average_raw ( struct iio_channel * chan , int * val )
{
int ret ;
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_channel_read ( chan , val , NULL , IIO_CHAN_INFO_AVERAGE_RAW ) ;
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_read_channel_average_raw ) ;
2012-09-17 13:17:00 +01:00
static int iio_convert_raw_to_processed_unlocked ( struct iio_channel * chan ,
int raw , int * processed , unsigned int scale )
{
int scale_type , scale_val , scale_val2 , offset ;
s64 raw64 = raw ;
int ret ;
2013-06-03 09:04:00 +01:00
ret = iio_channel_read ( chan , & offset , NULL , IIO_CHAN_INFO_OFFSET ) ;
2013-07-01 17:40:00 +01:00
if ( ret > = 0 )
2012-09-17 13:17:00 +01:00
raw64 + = offset ;
scale_type = iio_channel_read ( chan , & scale_val , & scale_val2 ,
IIO_CHAN_INFO_SCALE ) ;
2017-01-11 11:48:39 +01:00
if ( scale_type < 0 ) {
/*
* Just pass raw values as processed if no scaling is
* available .
*/
* processed = raw ;
return 0 ;
}
2012-09-17 13:17:00 +01:00
switch ( scale_type ) {
case IIO_VAL_INT :
* processed = raw64 * scale_val ;
break ;
case IIO_VAL_INT_PLUS_MICRO :
if ( scale_val2 < 0 )
* processed = - raw64 * scale_val ;
else
* processed = raw64 * scale_val ;
* processed + = div_s64 ( raw64 * ( s64 ) scale_val2 * scale ,
1000000LL ) ;
break ;
case IIO_VAL_INT_PLUS_NANO :
if ( scale_val2 < 0 )
* processed = - raw64 * scale_val ;
else
* processed = raw64 * scale_val ;
* processed + = div_s64 ( raw64 * ( s64 ) scale_val2 * scale ,
1000000000LL ) ;
break ;
case IIO_VAL_FRACTIONAL :
* processed = div_s64 ( raw64 * ( s64 ) scale_val * scale ,
scale_val2 ) ;
break ;
2012-10-16 17:29:00 +01:00
case IIO_VAL_FRACTIONAL_LOG2 :
* processed = ( raw64 * ( s64 ) scale_val * scale ) > > scale_val2 ;
break ;
2012-09-17 13:17:00 +01:00
default :
return - EINVAL ;
}
return 0 ;
}
int iio_convert_raw_to_processed ( struct iio_channel * chan , int raw ,
int * processed , unsigned int scale )
{
int ret ;
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_convert_raw_to_processed_unlocked ( chan , raw , processed ,
scale ) ;
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_convert_raw_to_processed ) ;
2016-09-23 23:04:07 -07:00
static int iio_read_channel_attribute ( struct iio_channel * chan ,
int * val , int * val2 ,
enum iio_chan_info_enum attribute )
{
int ret ;
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_channel_read ( chan , val , val2 , attribute ) ;
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
return ret ;
}
int iio_read_channel_offset ( struct iio_channel * chan , int * val , int * val2 )
{
return iio_read_channel_attribute ( chan , val , val2 , IIO_CHAN_INFO_OFFSET ) ;
}
EXPORT_SYMBOL_GPL ( iio_read_channel_offset ) ;
2012-09-17 13:17:00 +01:00
int iio_read_channel_processed ( struct iio_channel * chan , int * val )
{
int ret ;
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
if ( iio_channel_has_info ( chan - > channel , IIO_CHAN_INFO_PROCESSED ) ) {
ret = iio_channel_read ( chan , val , NULL ,
IIO_CHAN_INFO_PROCESSED ) ;
} else {
ret = iio_channel_read ( chan , val , NULL , IIO_CHAN_INFO_RAW ) ;
if ( ret < 0 )
goto err_unlock ;
ret = iio_convert_raw_to_processed_unlocked ( chan , * val , val , 1 ) ;
}
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_read_channel_processed ) ;
2012-05-01 21:04:24 +01:00
int iio_read_channel_scale ( struct iio_channel * chan , int * val , int * val2 )
2012-02-15 19:48:01 +00:00
{
2016-09-23 23:04:07 -07:00
return iio_read_channel_attribute ( chan , val , val2 , IIO_CHAN_INFO_SCALE ) ;
2012-02-15 19:48:01 +00:00
}
2012-05-01 21:04:24 +01:00
EXPORT_SYMBOL_GPL ( iio_read_channel_scale ) ;
2016-11-08 12:58:52 +01:00
static int iio_channel_read_avail ( struct iio_channel * chan ,
const int * * vals , int * type , int * length ,
enum iio_chan_info_enum info )
{
if ( ! iio_channel_has_available ( chan - > channel , info ) )
return - EINVAL ;
return chan - > indio_dev - > info - > read_avail ( chan - > indio_dev , chan - > channel ,
vals , type , length , info ) ;
}
int iio_read_avail_channel_raw ( struct iio_channel * chan ,
const int * * vals , int * length )
{
int ret ;
int type ;
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( ! chan - > indio_dev - > info ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_channel_read_avail ( chan ,
vals , & type , length , IIO_CHAN_INFO_RAW ) ;
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
2017-04-20 23:01:57 +02:00
if ( ret > = 0 & & type ! = IIO_VAL_INT )
2016-11-08 12:58:52 +01:00
/* raw values are assumed to be IIO_VAL_INT */
ret = - EINVAL ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_read_avail_channel_raw ) ;
static int iio_channel_read_max ( struct iio_channel * chan ,
int * val , int * val2 , int * type ,
enum iio_chan_info_enum info )
{
int unused ;
const int * vals ;
int length ;
int ret ;
if ( ! val2 )
val2 = & unused ;
ret = iio_channel_read_avail ( chan , & vals , type , & length , info ) ;
switch ( ret ) {
case IIO_AVAIL_RANGE :
switch ( * type ) {
case IIO_VAL_INT :
* val = vals [ 2 ] ;
break ;
default :
* val = vals [ 4 ] ;
* val2 = vals [ 5 ] ;
}
return 0 ;
case IIO_AVAIL_LIST :
if ( length < = 0 )
return - EINVAL ;
switch ( * type ) {
case IIO_VAL_INT :
* val = vals [ - - length ] ;
while ( length ) {
if ( vals [ - - length ] > * val )
* val = vals [ length ] ;
}
break ;
default :
/* FIXME: learn about max for other iio values */
return - EINVAL ;
}
return 0 ;
default :
return ret ;
}
}
int iio_read_max_channel_raw ( struct iio_channel * chan , int * val )
{
int ret ;
int type ;
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( ! chan - > indio_dev - > info ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_channel_read_max ( chan , val , NULL , & type , IIO_CHAN_INFO_RAW ) ;
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_read_max_channel_raw ) ;
2012-02-15 19:48:01 +00:00
2012-05-01 21:04:24 +01:00
int iio_get_channel_type ( struct iio_channel * chan , enum iio_chan_type * type )
2012-02-15 19:48:01 +00:00
{
int ret = 0 ;
/* Need to verify underlying driver has not gone away */
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
* type = chan - > channel - > type ;
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
return ret ;
}
2012-05-01 21:04:24 +01:00
EXPORT_SYMBOL_GPL ( iio_get_channel_type ) ;
2014-11-27 01:42:45 +03:00
static int iio_channel_write ( struct iio_channel * chan , int val , int val2 ,
enum iio_chan_info_enum info )
{
return chan - > indio_dev - > info - > write_raw ( chan - > indio_dev ,
chan - > channel , val , val2 , info ) ;
}
int iio_write_channel_raw ( struct iio_channel * chan , int val )
{
int ret ;
mutex_lock ( & chan - > indio_dev - > info_exist_lock ) ;
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_channel_write ( chan , val , 0 , IIO_CHAN_INFO_RAW ) ;
err_unlock :
mutex_unlock ( & chan - > indio_dev - > info_exist_lock ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_write_channel_raw ) ;
2017-05-14 21:51:08 +02:00
unsigned int iio_get_channel_ext_info_count ( struct iio_channel * chan )
{
const struct iio_chan_spec_ext_info * ext_info ;
unsigned int i = 0 ;
if ( ! chan - > channel - > ext_info )
return i ;
for ( ext_info = chan - > channel - > ext_info ; ext_info - > name ; ext_info + + )
+ + i ;
return i ;
}
EXPORT_SYMBOL_GPL ( iio_get_channel_ext_info_count ) ;
static const struct iio_chan_spec_ext_info * iio_lookup_ext_info (
const struct iio_channel * chan ,
const char * attr )
{
const struct iio_chan_spec_ext_info * ext_info ;
if ( ! chan - > channel - > ext_info )
return NULL ;
for ( ext_info = chan - > channel - > ext_info ; ext_info - > name ; + + ext_info ) {
if ( ! strcmp ( attr , ext_info - > name ) )
return ext_info ;
}
return NULL ;
}
ssize_t iio_read_channel_ext_info ( struct iio_channel * chan ,
const char * attr , char * buf )
{
const struct iio_chan_spec_ext_info * ext_info ;
ext_info = iio_lookup_ext_info ( chan , attr ) ;
if ( ! ext_info )
return - EINVAL ;
return ext_info - > read ( chan - > indio_dev , ext_info - > private ,
chan - > channel , buf ) ;
}
EXPORT_SYMBOL_GPL ( iio_read_channel_ext_info ) ;
ssize_t iio_write_channel_ext_info ( struct iio_channel * chan , const char * attr ,
const char * buf , size_t len )
{
const struct iio_chan_spec_ext_info * ext_info ;
ext_info = iio_lookup_ext_info ( chan , attr ) ;
if ( ! ext_info )
return - EINVAL ;
return ext_info - > write ( chan - > indio_dev , ext_info - > private ,
chan - > channel , buf , len ) ;
}
EXPORT_SYMBOL_GPL ( iio_write_channel_ext_info ) ;