2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2012-02-15 23:48:01 +04:00
/* The industrial I/O core in kernel channel mapping
*
* Copyright ( c ) 2011 Jonathan Cameron
*/
# include <linux/err.h>
# include <linux/export.h>
# include <linux/slab.h>
# include <linux/mutex.h>
2013-02-07 21:09:00 +04:00
# include <linux/of.h>
2012-02-15 23:48:01 +04:00
2012-04-25 18:54:58 +04:00
# include <linux/iio/iio.h>
2021-04-26 20:49:08 +03:00
# include <linux/iio/iio-opaque.h>
2012-02-15 23:48:01 +04:00
# include "iio_core.h"
2012-04-25 18:54:58 +04:00
# include <linux/iio/machine.h>
# include <linux/iio/driver.h>
# include <linux/iio/consumer.h>
2012-02-15 23:48:01 +04: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 ) ;
2020-11-28 16:44:18 +03:00
static int iio_map_array_unregister_locked ( struct iio_dev * indio_dev )
{
int ret = - ENODEV ;
struct iio_map_internal * mapi , * next ;
list_for_each_entry_safe ( mapi , next , & iio_map_list , l ) {
if ( indio_dev = = mapi - > indio_dev ) {
list_del ( & mapi - > l ) ;
kfree ( mapi ) ;
ret = 0 ;
}
}
return ret ;
}
2012-02-15 23:48:01 +04:00
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 19:46:01 +03:00
list_add_tail ( & mapi - > l , & iio_map_list ) ;
2012-02-15 23:48:01 +04:00
i + + ;
}
error_ret :
2020-11-28 16:44:19 +03:00
if ( ret )
iio_map_array_unregister_locked ( indio_dev ) ;
2012-02-15 23:48:01 +04:00
mutex_unlock ( & iio_map_list_lock ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_map_array_register ) ;
2013-02-01 01:43:00 +04:00
/*
* Remove all map entries associated with the given iio device
2012-02-15 23:48:01 +04:00
*/
2013-02-01 01:43:00 +04:00
int iio_map_array_unregister ( struct iio_dev * indio_dev )
2012-02-15 23:48:01 +04:00
{
2020-11-28 16:44:18 +03:00
int ret ;
2012-02-15 23:48:01 +04:00
mutex_lock ( & iio_map_list_lock ) ;
2020-11-28 16:44:18 +03:00
ret = iio_map_array_unregister_locked ( indio_dev ) ;
2012-02-15 23:48:01 +04:00
mutex_unlock ( & iio_map_list_lock ) ;
2020-11-28 16:44:18 +03:00
2012-02-15 23:48:01 +04:00
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_map_array_unregister ) ;
2021-09-03 10:29:13 +03:00
static void iio_map_array_unregister_cb ( void * indio_dev )
{
iio_map_array_unregister ( indio_dev ) ;
}
int devm_iio_map_array_register ( struct device * dev , struct iio_dev * indio_dev , struct iio_map * maps )
{
int ret ;
ret = iio_map_array_register ( indio_dev , maps ) ;
if ( ret )
return ret ;
return devm_add_action_or_reset ( dev , iio_map_array_unregister_cb , indio_dev ) ;
}
EXPORT_SYMBOL_GPL ( devm_iio_map_array_register ) ;
2012-02-15 23:48:01 +04:00
static const struct iio_chan_spec
2012-05-02 00:04:24 +04:00
* iio_chan_spec_from_name ( const struct iio_dev * indio_dev , const char * name )
2012-02-15 23:48:01 +04: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 21:09:00 +04:00
# ifdef CONFIG_OF
2019-06-14 20:53:59 +03:00
static int iio_dev_node_match ( struct device * dev , const void * data )
2013-02-07 21:09:00 +04:00
{
return dev - > of_node = = data & & dev - > type = = & iio_device_type ;
}
2014-10-22 19:29:43 +04: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 21:13:24 +03: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 19:29:43 +04:00
return - EINVAL ;
2015-01-01 21:13:24 +03:00
}
2014-10-22 19:29:43 +04:00
return iiospec - > args [ 0 ] ;
}
2013-02-07 21:09:00 +04: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 19:29:43 +04: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 21:09:00 +04:00
goto err_put ;
channel - > channel = & indio_dev - > channels [ index ] ;
return 0 ;
err_put :
iio_device_put ( indio_dev ) ;
2014-10-22 19:29:43 +04:00
return index ;
2013-02-07 21:09:00 +04: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 ) ;
}
2020-12-04 05:55:01 +03:00
struct iio_channel * of_iio_channel_get_by_name ( struct device_node * np ,
const char * name )
2013-02-07 21:09:00 +04:00
{
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 12:04:00 +04:00
if ( ! IS_ERR ( chan ) | | PTR_ERR ( chan ) = = - EPROBE_DEFER )
2013-02-07 21:09:00 +04:00
break ;
else if ( name & & index > = 0 ) {
2017-07-19 00:43:08 +03:00
pr_err ( " ERROR: could not get IIO channel %pOF:%s(%i) \n " ,
np , name ? name : " " , index ) ;
2014-11-06 15:11:00 +03:00
return NULL ;
2013-02-07 21:09:00 +04: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 15:11:00 +03:00
return NULL ;
2013-02-07 21:09:00 +04:00
}
2014-11-06 15:11:00 +03:00
2013-02-07 21:09:00 +04:00
return chan ;
}
2020-12-04 05:55:01 +03:00
EXPORT_SYMBOL_GPL ( of_iio_channel_get_by_name ) ;
2013-02-07 21:09:00 +04:00
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_all ( struct device * dev )
{
return NULL ;
}
# endif /* CONFIG_OF */
2012-02-15 23:48:01 +04:00
2013-02-05 00:26:00 +04:00
static struct iio_channel * iio_channel_get_sys ( const char * name ,
const char * channel_name )
2012-02-15 23:48:01 +04:00
{
struct iio_map_internal * c_i = NULL , * c = NULL ;
struct iio_channel * channel ;
2012-09-18 08:56:00 +04:00
int err ;
2012-02-15 23:48:01 +04: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 12:50:03 +04:00
iio_device_get ( c - > indio_dev ) ;
2012-02-15 23:48:01 +04:00
break ;
}
mutex_unlock ( & iio_map_list_lock ) ;
if ( c = = NULL )
return ERR_PTR ( - ENODEV ) ;
2012-09-14 05:24:00 +04:00
channel = kzalloc ( sizeof ( * channel ) , GFP_KERNEL ) ;
2012-09-18 08:56:00 +04:00
if ( channel = = NULL ) {
err = - ENOMEM ;
2012-09-18 08:55:00 +04:00
goto error_no_mem ;
2012-09-18 08:56:00 +04:00
}
2012-02-15 23:48:01 +04:00
channel - > indio_dev = c - > indio_dev ;
2012-09-17 12:44:00 +04:00
if ( c - > map - > adc_channel_label ) {
2012-02-15 23:48:01 +04:00
channel - > channel =
iio_chan_spec_from_name ( channel - > indio_dev ,
c - > map - > adc_channel_label ) ;
2012-09-18 08:56:00 +04:00
if ( channel - > channel = = NULL ) {
err = - EINVAL ;
2012-09-17 12:44:00 +04:00
goto error_no_chan ;
2012-09-18 08:56:00 +04:00
}
2012-09-17 12:44:00 +04:00
}
2012-02-15 23:48:01 +04:00
return channel ;
2012-09-17 12:44:00 +04:00
error_no_chan :
kfree ( channel ) ;
2012-09-18 08:55:00 +04:00
error_no_mem :
iio_device_put ( c - > indio_dev ) ;
2012-09-18 08:56:00 +04:00
return ERR_PTR ( err ) ;
2012-02-15 23:48:01 +04:00
}
2013-02-05 00:26:00 +04: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 21:09:00 +04:00
struct iio_channel * channel ;
2013-02-05 00:26:00 +04:00
2013-02-07 21:09:00 +04:00
if ( dev ) {
channel = of_iio_channel_get_by_name ( dev - > of_node ,
channel_name ) ;
if ( channel ! = NULL )
return channel ;
}
2014-11-06 15:11:00 +03:00
2013-02-05 00:26:00 +04:00
return iio_channel_get_sys ( name , channel_name ) ;
}
2012-05-02 00:04:24 +04:00
EXPORT_SYMBOL_GPL ( iio_channel_get ) ;
2012-02-15 23:48:01 +04:00
2012-05-02 00:04:24 +04:00
void iio_channel_release ( struct iio_channel * channel )
2012-02-15 23:48:01 +04:00
{
2016-01-26 12:25:21 +03:00
if ( ! channel )
return ;
2012-06-04 12:50:03 +04:00
iio_device_put ( channel - > indio_dev ) ;
2012-02-15 23:48:01 +04:00
kfree ( channel ) ;
}
2012-05-02 00:04:24 +04:00
EXPORT_SYMBOL_GPL ( iio_channel_release ) ;
2012-02-15 23:48:01 +04:00
2021-04-08 14:38:16 +03:00
static void devm_iio_channel_free ( void * iio_channel )
2016-04-06 13:31:06 +03:00
{
2021-04-08 14:38:16 +03:00
iio_channel_release ( iio_channel ) ;
2016-04-06 13:31:06 +03:00
}
struct iio_channel * devm_iio_channel_get ( struct device * dev ,
const char * channel_name )
{
2021-04-08 14:38:16 +03:00
struct iio_channel * channel ;
int ret ;
2016-04-06 13:31:06 +03:00
channel = iio_channel_get ( dev , channel_name ) ;
2021-04-08 14:38:16 +03:00
if ( IS_ERR ( channel ) )
2016-04-06 13:31:06 +03:00
return channel ;
2021-04-08 14:38:16 +03:00
ret = devm_add_action_or_reset ( dev , devm_iio_channel_free , channel ) ;
if ( ret )
return ERR_PTR ( ret ) ;
2016-04-06 13:31:06 +03:00
return channel ;
}
EXPORT_SYMBOL_GPL ( devm_iio_channel_get ) ;
2020-12-04 05:55:01 +03:00
struct iio_channel * devm_of_iio_channel_get_by_name ( struct device * dev ,
struct device_node * np ,
const char * channel_name )
{
2021-04-08 14:38:16 +03:00
struct iio_channel * channel ;
int ret ;
2020-12-04 05:55:01 +03:00
channel = of_iio_channel_get_by_name ( np , channel_name ) ;
2021-04-08 14:38:16 +03:00
if ( IS_ERR ( channel ) )
2020-12-04 05:55:01 +03:00
return channel ;
2021-04-08 14:38:16 +03:00
ret = devm_add_action_or_reset ( dev , devm_iio_channel_free , channel ) ;
if ( ret )
return ERR_PTR ( ret ) ;
2020-12-04 05:55:01 +03:00
return channel ;
}
EXPORT_SYMBOL_GPL ( devm_of_iio_channel_get_by_name ) ;
2013-02-01 01:43:00 +04:00
struct iio_channel * iio_channel_get_all ( struct device * dev )
2012-02-15 23:48:01 +04:00
{
2013-02-01 01:43:00 +04:00
const char * name ;
2012-02-15 23:48:01 +04:00
struct iio_channel * chans ;
struct iio_map_internal * c = NULL ;
int nummaps = 0 ;
int mapind = 0 ;
int i , ret ;
2013-02-01 01:43:00 +04:00
if ( dev = = NULL )
2012-02-15 23:48:01 +04:00
return ERR_PTR ( - EINVAL ) ;
2013-02-07 21:09:00 +04:00
chans = of_iio_channel_get_all ( dev ) ;
if ( chans )
return chans ;
2013-02-01 01:43:00 +04:00
name = dev_name ( dev ) ;
2012-02-15 23:48:01 +04: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 */
treewide: kzalloc() -> kcalloc()
The kzalloc() function has a 2-factor argument form, kcalloc(). This
patch replaces cases of:
kzalloc(a * b, gfp)
with:
kcalloc(a * b, gfp)
as well as handling cases of:
kzalloc(a * b * c, gfp)
with:
kzalloc(array3_size(a, b, c), gfp)
as it's slightly less ugly than:
kzalloc_array(array_size(a, b), c, gfp)
This does, however, attempt to ignore constant size factors like:
kzalloc(4 * 1024, gfp)
though any constants defined via macros get caught up in the conversion.
Any factors with a sizeof() of "unsigned char", "char", and "u8" were
dropped, since they're redundant.
The Coccinelle script used for this was:
// Fix redundant parens around sizeof().
@@
type TYPE;
expression THING, E;
@@
(
kzalloc(
- (sizeof(TYPE)) * E
+ sizeof(TYPE) * E
, ...)
|
kzalloc(
- (sizeof(THING)) * E
+ sizeof(THING) * E
, ...)
)
// Drop single-byte sizes and redundant parens.
@@
expression COUNT;
typedef u8;
typedef __u8;
@@
(
kzalloc(
- sizeof(u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * COUNT
+ COUNT
, ...)
)
// 2-factor product with sizeof(type/expression) and identifier or constant.
@@
type TYPE;
expression THING;
identifier COUNT_ID;
constant COUNT_CONST;
@@
(
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_ID)
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_ID
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_CONST)
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_CONST
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_ID)
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_ID
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_CONST)
+ COUNT_CONST, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_CONST
+ COUNT_CONST, sizeof(THING)
, ...)
)
// 2-factor product, only identifiers.
@@
identifier SIZE, COUNT;
@@
- kzalloc
+ kcalloc
(
- SIZE * COUNT
+ COUNT, SIZE
, ...)
// 3-factor product with 1 sizeof(type) or sizeof(expression), with
// redundant parens removed.
@@
expression THING;
identifier STRIDE, COUNT;
type TYPE;
@@
(
kzalloc(
- sizeof(TYPE) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
)
// 3-factor product with 2 sizeof(variable), with redundant parens removed.
@@
expression THING1, THING2;
identifier COUNT;
type TYPE1, TYPE2;
@@
(
kzalloc(
- sizeof(TYPE1) * sizeof(TYPE2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
)
// 3-factor product, only identifiers, with redundant parens removed.
@@
identifier STRIDE, SIZE, COUNT;
@@
(
kzalloc(
- (COUNT) * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
)
// Any remaining multi-factor products, first at least 3-factor products,
// when they're not all constants...
@@
expression E1, E2, E3;
constant C1, C2, C3;
@@
(
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(
- (E1) * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * (E3)
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- E1 * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
)
// And then all remaining 2 factors products when they're not all constants,
// keeping sizeof() as the second factor argument.
@@
expression THING, E1, E2;
type TYPE;
constant C1, C2, C3;
@@
(
kzalloc(sizeof(THING) * C2, ...)
|
kzalloc(sizeof(TYPE) * C2, ...)
|
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(C1 * C2, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (E2)
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * E2
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (E2)
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * E2
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * E2
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * (E2)
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- E1 * E2
+ E1, E2
, ...)
)
Signed-off-by: Kees Cook <keescook@chromium.org>
2018-06-13 00:03:40 +03:00
chans = kcalloc ( nummaps + 1 , sizeof ( * chans ) , GFP_KERNEL ) ;
2012-02-15 23:48:01 +04:00
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 23:06:00 +04:00
chans [ mapind ] . data = c - > map - > consumer_data ;
2012-02-15 23:48:01 +04: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 12:50:03 +04:00
iio_device_get ( chans [ mapind ] . indio_dev ) ;
2012-02-15 23:48:01 +04:00
mapind + + ;
}
if ( mapind = = 0 ) {
ret = - ENODEV ;
goto error_free_chans ;
}
2012-07-11 10:34:00 +04:00
mutex_unlock ( & iio_map_list_lock ) ;
2012-02-15 23:48:01 +04:00
return chans ;
error_free_chans :
for ( i = 0 ; i < nummaps ; i + + )
2012-06-04 12:50:03 +04:00
iio_device_put ( chans [ i ] . indio_dev ) ;
2012-02-15 23:48:01 +04:00
kfree ( chans ) ;
error_ret :
mutex_unlock ( & iio_map_list_lock ) ;
return ERR_PTR ( ret ) ;
}
2012-05-02 00:04:24 +04:00
EXPORT_SYMBOL_GPL ( iio_channel_get_all ) ;
2012-02-15 23:48:01 +04:00
2012-05-02 00:04:24 +04:00
void iio_channel_release_all ( struct iio_channel * channels )
2012-02-15 23:48:01 +04:00
{
struct iio_channel * chan = & channels [ 0 ] ;
while ( chan - > indio_dev ) {
2012-06-04 12:50:03 +04:00
iio_device_put ( chan - > indio_dev ) ;
2012-02-15 23:48:01 +04:00
chan + + ;
}
kfree ( channels ) ;
}
2012-05-02 00:04:24 +04:00
EXPORT_SYMBOL_GPL ( iio_channel_release_all ) ;
2012-02-15 23:48:01 +04:00
2021-04-08 14:38:16 +03:00
static void devm_iio_channel_free_all ( void * iio_channels )
2016-04-06 13:31:07 +03:00
{
2021-04-08 14:38:16 +03:00
iio_channel_release_all ( iio_channels ) ;
2016-04-06 13:31:07 +03:00
}
struct iio_channel * devm_iio_channel_get_all ( struct device * dev )
{
2021-04-08 14:38:16 +03:00
struct iio_channel * channels ;
int ret ;
2016-04-06 13:31:07 +03:00
channels = iio_channel_get_all ( dev ) ;
2021-04-08 14:38:16 +03:00
if ( IS_ERR ( channels ) )
2016-04-06 13:31:07 +03:00
return channels ;
2021-04-08 14:38:16 +03:00
ret = devm_add_action_or_reset ( dev , devm_iio_channel_free_all ,
channels ) ;
if ( ret )
return ERR_PTR ( ret ) ;
2016-04-06 13:31:07 +03:00
return channels ;
}
EXPORT_SYMBOL_GPL ( devm_iio_channel_get_all ) ;
2012-09-17 16:17:00 +04: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 03:51:00 +04:00
int vals [ INDIO_MAX_RAW_ELEMENTS ] ;
int ret ;
int val_len = 2 ;
2012-09-17 16:17:00 +04:00
if ( val2 = = NULL )
val2 = & unused ;
2016-04-14 22:36:34 +03:00
if ( ! iio_channel_has_info ( chan - > channel , info ) )
2015-01-01 15:46:48 +03:00
return - EINVAL ;
2014-04-29 03:51:00 +04: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 16:17:00 +04:00
}
2012-05-02 00:04:24 +04:00
int iio_read_channel_raw ( struct iio_channel * chan , int * val )
2012-02-15 23:48:01 +04:00
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2012-09-17 16:17:00 +04:00
int ret ;
2012-02-15 23:48:01 +04:00
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2012-02-15 23:48:01 +04:00
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
2012-09-17 16:17:00 +04:00
ret = iio_channel_read ( chan , val , NULL , IIO_CHAN_INFO_RAW ) ;
2012-02-15 23:48:01 +04:00
err_unlock :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2012-02-15 23:48:01 +04:00
return ret ;
}
2012-05-02 00:04:24 +04:00
EXPORT_SYMBOL_GPL ( iio_read_channel_raw ) ;
2012-02-15 23:48:01 +04:00
2014-10-03 20:25:00 +04:00
int iio_read_channel_average_raw ( struct iio_channel * chan , int * val )
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2014-10-03 20:25:00 +04:00
int ret ;
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2014-10-03 20:25:00 +04:00
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 :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2014-10-03 20:25:00 +04:00
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_read_channel_average_raw ) ;
2012-09-17 16:17:00 +04: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 12:04:00 +04:00
ret = iio_channel_read ( chan , & offset , NULL , IIO_CHAN_INFO_OFFSET ) ;
2013-07-01 20:40:00 +04:00
if ( ret > = 0 )
2012-09-17 16:17:00 +04:00
raw64 + = offset ;
scale_type = iio_channel_read ( chan , & scale_val , & scale_val2 ,
IIO_CHAN_INFO_SCALE ) ;
2017-01-11 13:48:39 +03:00
if ( scale_type < 0 ) {
/*
* Just pass raw values as processed if no scaling is
* available .
*/
* processed = raw ;
return 0 ;
}
2012-09-17 16:17:00 +04: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 20:29:00 +04:00
case IIO_VAL_FRACTIONAL_LOG2 :
* processed = ( raw64 * ( s64 ) scale_val * scale ) > > scale_val2 ;
break ;
2012-09-17 16:17:00 +04:00
default :
return - EINVAL ;
}
return 0 ;
}
int iio_convert_raw_to_processed ( struct iio_channel * chan , int raw ,
int * processed , unsigned int scale )
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2012-09-17 16:17:00 +04:00
int ret ;
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2012-09-17 16:17:00 +04:00
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_convert_raw_to_processed_unlocked ( chan , raw , processed ,
scale ) ;
err_unlock :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2012-09-17 16:17:00 +04:00
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_convert_raw_to_processed ) ;
2018-01-10 13:13:06 +03:00
int iio_read_channel_attribute ( struct iio_channel * chan , int * val , int * val2 ,
enum iio_chan_info_enum attribute )
2016-09-24 09:04:07 +03:00
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2016-09-24 09:04:07 +03:00
int ret ;
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2016-09-24 09:04:07 +03:00
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_channel_read ( chan , val , val2 , attribute ) ;
err_unlock :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2016-09-24 09:04:07 +03:00
return ret ;
}
2018-01-10 13:13:06 +03:00
EXPORT_SYMBOL_GPL ( iio_read_channel_attribute ) ;
2016-09-24 09:04:07 +03:00
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 ) ;
2021-03-08 13:02:18 +03:00
int iio_read_channel_processed_scale ( struct iio_channel * chan , int * val ,
unsigned int scale )
2012-09-17 16:17:00 +04:00
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2012-09-17 16:17:00 +04:00
int ret ;
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2012-09-17 16:17:00 +04:00
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 ) ;
2021-03-23 15:27:05 +03:00
if ( ret < 0 )
2021-03-08 13:02:18 +03:00
goto err_unlock ;
* val * = scale ;
2012-09-17 16:17:00 +04:00
} else {
ret = iio_channel_read ( chan , val , NULL , IIO_CHAN_INFO_RAW ) ;
if ( ret < 0 )
goto err_unlock ;
2021-03-08 13:02:18 +03:00
ret = iio_convert_raw_to_processed_unlocked ( chan , * val , val ,
scale ) ;
2012-09-17 16:17:00 +04:00
}
err_unlock :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2012-09-17 16:17:00 +04:00
return ret ;
}
2021-03-08 13:02:18 +03:00
EXPORT_SYMBOL_GPL ( iio_read_channel_processed_scale ) ;
int iio_read_channel_processed ( struct iio_channel * chan , int * val )
{
/* This is just a special case with scale factor 1 */
return iio_read_channel_processed_scale ( chan , val , 1 ) ;
}
2012-09-17 16:17:00 +04:00
EXPORT_SYMBOL_GPL ( iio_read_channel_processed ) ;
2012-05-02 00:04:24 +04:00
int iio_read_channel_scale ( struct iio_channel * chan , int * val , int * val2 )
2012-02-15 23:48:01 +04:00
{
2016-09-24 09:04:07 +03:00
return iio_read_channel_attribute ( chan , val , val2 , IIO_CHAN_INFO_SCALE ) ;
2012-02-15 23:48:01 +04:00
}
2012-05-02 00:04:24 +04:00
EXPORT_SYMBOL_GPL ( iio_read_channel_scale ) ;
2016-11-08 14:58:52 +03: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 ) ;
}
2019-03-23 20:28:06 +03:00
int iio_read_avail_channel_attribute ( struct iio_channel * chan ,
const int * * vals , int * type , int * length ,
enum iio_chan_info_enum attribute )
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2019-03-23 20:28:06 +03:00
int ret ;
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2019-03-23 20:28:06 +03:00
if ( ! chan - > indio_dev - > info ) {
ret = - ENODEV ;
goto err_unlock ;
}
ret = iio_channel_read_avail ( chan , vals , type , length , attribute ) ;
err_unlock :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2019-03-23 20:28:06 +03:00
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_read_avail_channel_attribute ) ;
2016-11-08 14:58:52 +03:00
int iio_read_avail_channel_raw ( struct iio_channel * chan ,
const int * * vals , int * length )
{
int ret ;
int type ;
2019-03-23 20:28:07 +03:00
ret = iio_read_avail_channel_attribute ( chan , vals , & type , length ,
IIO_CHAN_INFO_RAW ) ;
2016-11-08 14:58:52 +03:00
2017-04-21 00:01:57 +03:00
if ( ret > = 0 & & type ! = IIO_VAL_INT )
2016-11-08 14:58:52 +03: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 )
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2016-11-08 14:58:52 +03:00
int ret ;
int type ;
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2016-11-08 14:58:52 +03:00
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 :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2016-11-08 14:58:52 +03:00
return ret ;
}
EXPORT_SYMBOL_GPL ( iio_read_max_channel_raw ) ;
2012-02-15 23:48:01 +04:00
2012-05-02 00:04:24 +04:00
int iio_get_channel_type ( struct iio_channel * chan , enum iio_chan_type * type )
2012-02-15 23:48:01 +04:00
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2012-02-15 23:48:01 +04:00
int ret = 0 ;
/* Need to verify underlying driver has not gone away */
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2012-02-15 23:48:01 +04:00
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
* type = chan - > channel - > type ;
err_unlock :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2012-02-15 23:48:01 +04:00
return ret ;
}
2012-05-02 00:04:24 +04: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 ) ;
}
2018-01-10 13:13:06 +03:00
int iio_write_channel_attribute ( struct iio_channel * chan , int val , int val2 ,
enum iio_chan_info_enum attribute )
2014-11-27 01:42:45 +03:00
{
2021-04-26 20:49:08 +03:00
struct iio_dev_opaque * iio_dev_opaque = to_iio_dev_opaque ( chan - > indio_dev ) ;
2014-11-27 01:42:45 +03:00
int ret ;
2021-04-26 20:49:08 +03:00
mutex_lock ( & iio_dev_opaque - > info_exist_lock ) ;
2014-11-27 01:42:45 +03:00
if ( chan - > indio_dev - > info = = NULL ) {
ret = - ENODEV ;
goto err_unlock ;
}
2018-01-10 13:13:06 +03:00
ret = iio_channel_write ( chan , val , val2 , attribute ) ;
2014-11-27 01:42:45 +03:00
err_unlock :
2021-04-26 20:49:08 +03:00
mutex_unlock ( & iio_dev_opaque - > info_exist_lock ) ;
2014-11-27 01:42:45 +03:00
return ret ;
}
2018-01-10 13:13:06 +03:00
EXPORT_SYMBOL_GPL ( iio_write_channel_attribute ) ;
int iio_write_channel_raw ( struct iio_channel * chan , int val )
{
return iio_write_channel_attribute ( chan , val , 0 , IIO_CHAN_INFO_RAW ) ;
}
2014-11-27 01:42:45 +03:00
EXPORT_SYMBOL_GPL ( iio_write_channel_raw ) ;
2017-05-14 22:51:08 +03: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 ) ;