2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2015-08-14 16:54:54 +02:00
/* The industrial I/O callback buffer
*/
2012-06-30 20:06:00 +01:00
# include <linux/kernel.h>
2015-08-14 16:54:54 +02:00
# include <linux/module.h>
2012-06-30 20:06:00 +01:00
# include <linux/slab.h>
# include <linux/err.h>
# include <linux/export.h>
2017-01-02 19:28:31 +00:00
# include <linux/iio/iio.h>
2017-01-02 19:28:34 +00:00
# include <linux/iio/buffer_impl.h>
2012-06-30 20:06:00 +01:00
# include <linux/iio/consumer.h>
struct iio_cb_buffer {
struct iio_buffer buffer ;
2013-09-15 17:50:00 +01:00
int ( * cb ) ( const void * data , void * private ) ;
2012-06-30 20:06:00 +01:00
void * private ;
struct iio_channel * channels ;
2016-08-19 20:17:02 -07:00
struct iio_dev * indio_dev ;
2012-06-30 20:06:00 +01:00
} ;
2013-10-04 12:06:00 +01:00
static struct iio_cb_buffer * buffer_to_cb_buffer ( struct iio_buffer * buffer )
2012-06-30 20:06:00 +01:00
{
2013-10-04 12:06:00 +01:00
return container_of ( buffer , struct iio_cb_buffer , buffer ) ;
}
2012-06-30 20:06:00 +01:00
2013-10-04 12:06:00 +01:00
static int iio_buffer_cb_store_to ( struct iio_buffer * buffer , const void * data )
{
struct iio_cb_buffer * cb_buff = buffer_to_cb_buffer ( buffer ) ;
2012-06-30 20:06:00 +01:00
return cb_buff - > cb ( data , cb_buff - > private ) ;
}
2013-10-04 12:06:00 +01:00
static void iio_buffer_cb_release ( struct iio_buffer * buffer )
{
struct iio_cb_buffer * cb_buff = buffer_to_cb_buffer ( buffer ) ;
2019-03-04 10:55:40 +02:00
bitmap_free ( cb_buff - > buffer . scan_mask ) ;
2013-10-04 12:06:00 +01:00
kfree ( cb_buff ) ;
}
2013-09-15 17:09:00 +01:00
static const struct iio_buffer_access_funcs iio_cb_access = {
2012-06-30 20:06:00 +01:00
. store_to = & iio_buffer_cb_store_to ,
2013-10-04 12:06:00 +01:00
. release = & iio_buffer_cb_release ,
2015-05-29 18:14:21 +02:00
. modes = INDIO_BUFFER_SOFTWARE | INDIO_BUFFER_TRIGGERED ,
2012-06-30 20:06:00 +01:00
} ;
2013-01-31 21:43:00 +00:00
struct iio_cb_buffer * iio_channel_get_all_cb ( struct device * dev ,
2013-09-15 17:50:00 +01:00
int ( * cb ) ( const void * data ,
2012-06-30 20:06:00 +01:00
void * private ) ,
void * private )
{
int ret ;
struct iio_cb_buffer * cb_buff ;
struct iio_channel * chan ;
2020-11-21 17:14:57 +01:00
if ( ! cb ) {
dev_err ( dev , " Invalid arguments: A callback must be provided! \n " ) ;
return ERR_PTR ( - EINVAL ) ;
}
2012-06-30 20:06:00 +01:00
cb_buff = kzalloc ( sizeof ( * cb_buff ) , GFP_KERNEL ) ;
2014-02-16 11:53:00 +00:00
if ( cb_buff = = NULL )
return ERR_PTR ( - ENOMEM ) ;
2012-06-30 20:06:00 +01:00
2013-09-18 21:02:00 +01:00
iio_buffer_init ( & cb_buff - > buffer ) ;
2012-06-30 20:06:00 +01:00
cb_buff - > private = private ;
cb_buff - > cb = cb ;
cb_buff - > buffer . access = & iio_cb_access ;
INIT_LIST_HEAD ( & cb_buff - > buffer . demux_list ) ;
2013-01-31 21:43:00 +00:00
cb_buff - > channels = iio_channel_get_all ( dev ) ;
2012-06-30 20:06:00 +01:00
if ( IS_ERR ( cb_buff - > channels ) ) {
ret = PTR_ERR ( cb_buff - > channels ) ;
goto error_free_cb_buff ;
}
2016-08-19 20:17:02 -07:00
cb_buff - > indio_dev = cb_buff - > channels [ 0 ] . indio_dev ;
2019-03-04 10:55:40 +02:00
cb_buff - > buffer . scan_mask = bitmap_zalloc ( cb_buff - > indio_dev - > masklength ,
GFP_KERNEL ) ;
2012-06-30 20:06:00 +01:00
if ( cb_buff - > buffer . scan_mask = = NULL ) {
ret = - ENOMEM ;
goto error_release_channels ;
}
chan = & cb_buff - > channels [ 0 ] ;
while ( chan - > indio_dev ) {
2016-08-19 20:17:02 -07:00
if ( chan - > indio_dev ! = cb_buff - > indio_dev ) {
2012-06-30 20:06:00 +01:00
ret = - EINVAL ;
2013-05-22 22:41:00 +01:00
goto error_free_scan_mask ;
2012-06-30 20:06:00 +01:00
}
set_bit ( chan - > channel - > scan_index ,
cb_buff - > buffer . scan_mask ) ;
chan + + ;
}
return cb_buff ;
2013-05-22 22:41:00 +01:00
error_free_scan_mask :
2019-03-04 10:55:40 +02:00
bitmap_free ( cb_buff - > buffer . scan_mask ) ;
2012-06-30 20:06:00 +01:00
error_release_channels :
iio_channel_release_all ( cb_buff - > channels ) ;
error_free_cb_buff :
kfree ( cb_buff ) ;
return ERR_PTR ( ret ) ;
}
EXPORT_SYMBOL_GPL ( iio_channel_get_all_cb ) ;
2018-01-10 11:13:13 +01:00
int iio_channel_cb_set_buffer_watermark ( struct iio_cb_buffer * cb_buff ,
size_t watermark )
{
if ( ! watermark )
return - EINVAL ;
cb_buff - > buffer . watermark = watermark ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( iio_channel_cb_set_buffer_watermark ) ;
2012-06-30 20:06:00 +01:00
int iio_channel_start_all_cb ( struct iio_cb_buffer * cb_buff )
{
2016-08-19 20:17:02 -07:00
return iio_update_buffers ( cb_buff - > indio_dev , & cb_buff - > buffer ,
2012-06-30 20:06:00 +01:00
NULL ) ;
}
EXPORT_SYMBOL_GPL ( iio_channel_start_all_cb ) ;
void iio_channel_stop_all_cb ( struct iio_cb_buffer * cb_buff )
{
2016-08-19 20:17:02 -07:00
iio_update_buffers ( cb_buff - > indio_dev , NULL , & cb_buff - > buffer ) ;
2012-06-30 20:06:00 +01:00
}
EXPORT_SYMBOL_GPL ( iio_channel_stop_all_cb ) ;
void iio_channel_release_all_cb ( struct iio_cb_buffer * cb_buff )
{
iio_channel_release_all ( cb_buff - > channels ) ;
2013-10-04 12:06:00 +01:00
iio_buffer_put ( & cb_buff - > buffer ) ;
2012-06-30 20:06:00 +01:00
}
EXPORT_SYMBOL_GPL ( iio_channel_release_all_cb ) ;
struct iio_channel
* iio_channel_cb_get_channels ( const struct iio_cb_buffer * cb_buffer )
{
return cb_buffer - > channels ;
}
EXPORT_SYMBOL_GPL ( iio_channel_cb_get_channels ) ;
2015-08-14 16:54:54 +02:00
2016-08-19 20:17:02 -07:00
struct iio_dev
* iio_channel_cb_get_iio_dev ( const struct iio_cb_buffer * cb_buffer )
{
return cb_buffer - > indio_dev ;
}
EXPORT_SYMBOL_GPL ( iio_channel_cb_get_iio_dev ) ;
2015-08-14 16:54:54 +02:00
MODULE_AUTHOR ( " Jonathan Cameron <jic23@kernel.org> " ) ;
MODULE_DESCRIPTION ( " Industrial I/O callback buffer " ) ;
MODULE_LICENSE ( " GPL " ) ;