2011-02-11 13:09:10 +00:00
# include <linux/slab.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/device.h>
# include <linux/workqueue.h>
# include <linux/kfifo.h>
# include <linux/mutex.h>
2012-04-25 15:54:58 +01:00
# include <linux/iio/kfifo_buf.h>
2012-06-30 13:52:00 +01:00
# include <linux/sched.h>
2013-09-15 16:31:00 +01:00
# include <linux/poll.h>
2011-02-11 13:09:10 +00:00
2011-08-30 12:32:43 +01:00
struct iio_kfifo {
2011-09-21 11:15:57 +01:00
struct iio_buffer buffer ;
2011-08-30 12:32:43 +01:00
struct kfifo kf ;
2013-10-15 09:30:00 +01:00
struct mutex user_lock ;
2011-08-30 12:32:43 +01:00
int update_needed ;
} ;
2011-09-21 11:15:57 +01:00
# define iio_to_kfifo(r) container_of(r, struct iio_kfifo, buffer)
2011-05-18 14:42:24 +01:00
2011-02-11 13:09:10 +00:00
static inline int __iio_allocate_kfifo ( struct iio_kfifo * buf ,
int bytes_per_datum , int length )
{
if ( ( length = = 0 ) | | ( bytes_per_datum = = 0 ) )
return - EINVAL ;
2012-06-30 13:52:00 +01:00
return __kfifo_alloc ( ( struct __kfifo * ) & buf - > kf , length ,
bytes_per_datum , GFP_KERNEL ) ;
2011-02-11 13:09:10 +00:00
}
2011-09-21 11:15:57 +01:00
static int iio_request_update_kfifo ( struct iio_buffer * r )
2011-02-11 13:09:10 +00:00
{
int ret = 0 ;
struct iio_kfifo * buf = iio_to_kfifo ( r ) ;
2013-10-15 09:30:00 +01:00
mutex_lock ( & buf - > user_lock ) ;
2013-10-15 09:30:00 +01:00
if ( buf - > update_needed ) {
kfifo_free ( & buf - > kf ) ;
ret = __iio_allocate_kfifo ( buf , buf - > buffer . bytes_per_datum ,
2011-09-21 11:15:57 +01:00
buf - > buffer . length ) ;
2015-05-02 14:31:16 +02:00
if ( ret > = 0 )
buf - > update_needed = false ;
2013-10-15 09:30:00 +01:00
} else {
kfifo_reset_out ( & buf - > kf ) ;
}
2013-10-15 09:30:00 +01:00
mutex_unlock ( & buf - > user_lock ) ;
2013-10-15 09:30:00 +01:00
2011-02-11 13:09:10 +00:00
return ret ;
}
2011-12-19 15:23:48 +01:00
static int iio_mark_update_needed_kfifo ( struct iio_buffer * r )
2011-02-11 13:09:10 +00:00
{
2011-12-19 15:23:48 +01:00
struct iio_kfifo * kf = iio_to_kfifo ( r ) ;
kf - > update_needed = true ;
2011-02-11 13:09:10 +00:00
return 0 ;
}
2011-12-19 15:23:48 +01:00
static int iio_set_bytes_per_datum_kfifo ( struct iio_buffer * r , size_t bpd )
2011-02-11 13:09:10 +00:00
{
2011-12-19 15:23:48 +01:00
if ( r - > bytes_per_datum ! = bpd ) {
r - > bytes_per_datum = bpd ;
iio_mark_update_needed_kfifo ( r ) ;
}
2011-02-11 13:09:10 +00:00
return 0 ;
}
2011-09-21 11:15:57 +01:00
static int iio_set_length_kfifo ( struct iio_buffer * r , int length )
2011-02-11 13:09:10 +00:00
{
2012-06-30 13:52:00 +01:00
/* Avoid an invalid state */
if ( length < 2 )
length = 2 ;
2011-02-11 13:09:10 +00:00
if ( r - > length ! = length ) {
r - > length = length ;
2011-12-19 15:23:48 +01:00
iio_mark_update_needed_kfifo ( r ) ;
2011-02-11 13:09:10 +00:00
}
return 0 ;
}
2011-09-21 11:15:57 +01:00
static int iio_store_to_kfifo ( struct iio_buffer * r ,
2013-09-15 17:50:00 +01:00
const void * data )
2011-02-11 13:09:10 +00:00
{
int ret ;
struct iio_kfifo * kf = iio_to_kfifo ( r ) ;
2012-06-30 13:52:00 +01:00
ret = kfifo_in ( & kf - > kf , data , 1 ) ;
if ( ret ! = 1 )
2011-02-11 13:09:10 +00:00
return - EBUSY ;
return 0 ;
}
2011-09-21 11:15:57 +01:00
static int iio_read_first_n_kfifo ( struct iio_buffer * r ,
2011-05-18 14:41:02 +01:00
size_t n , char __user * buf )
2011-02-11 13:09:10 +00:00
{
int ret , copied ;
struct iio_kfifo * kf = iio_to_kfifo ( r ) ;
2013-10-15 09:30:00 +01:00
if ( mutex_lock_interruptible ( & kf - > user_lock ) )
return - ERESTARTSYS ;
2011-12-12 09:33:14 +01:00
2013-10-15 09:30:00 +01:00
if ( ! kfifo_initialized ( & kf - > kf ) | | n < kfifo_esize ( & kf - > kf ) )
ret = - EINVAL ;
else
ret = kfifo_to_user ( & kf - > kf , buf , n , & copied ) ;
mutex_unlock ( & kf - > user_lock ) ;
if ( ret < 0 )
return ret ;
2011-02-11 13:09:10 +00:00
return copied ;
}
2011-05-18 14:42:24 +01:00
2015-03-22 20:33:38 +02:00
static size_t iio_kfifo_buf_data_available ( struct iio_buffer * r )
2013-11-25 14:56:00 +00:00
{
struct iio_kfifo * kf = iio_to_kfifo ( r ) ;
2015-03-22 20:33:38 +02:00
size_t samples ;
2013-11-25 14:56:00 +00:00
mutex_lock ( & kf - > user_lock ) ;
2015-03-22 20:33:38 +02:00
samples = kfifo_len ( & kf - > kf ) ;
2013-11-25 14:56:00 +00:00
mutex_unlock ( & kf - > user_lock ) ;
2015-03-22 20:33:38 +02:00
return samples ;
2013-11-25 14:56:00 +00:00
}
2013-10-04 12:06:00 +01:00
static void iio_kfifo_buffer_release ( struct iio_buffer * buffer )
{
2013-10-15 09:30:00 +01:00
struct iio_kfifo * kf = iio_to_kfifo ( buffer ) ;
2013-10-15 09:30:00 +01:00
mutex_destroy ( & kf - > user_lock ) ;
2013-10-15 09:30:00 +01:00
kfifo_free ( & kf - > kf ) ;
kfree ( kf ) ;
2013-10-04 12:06:00 +01:00
}
2012-01-03 11:02:51 +01:00
static const struct iio_buffer_access_funcs kfifo_access_funcs = {
2011-05-18 14:42:24 +01:00
. store_to = & iio_store_to_kfifo ,
. read_first_n = & iio_read_first_n_kfifo ,
2013-11-25 14:56:00 +00:00
. data_available = iio_kfifo_buf_data_available ,
2011-05-18 14:42:24 +01:00
. request_update = & iio_request_update_kfifo ,
. set_bytes_per_datum = & iio_set_bytes_per_datum_kfifo ,
. set_length = & iio_set_length_kfifo ,
2013-10-04 12:06:00 +01:00
. release = & iio_kfifo_buffer_release ,
2015-05-29 18:14:21 +02:00
. modes = INDIO_BUFFER_SOFTWARE | INDIO_BUFFER_TRIGGERED ,
2011-05-18 14:42:24 +01:00
} ;
2012-01-03 11:02:51 +01:00
2014-12-19 18:39:24 +01:00
struct iio_buffer * iio_kfifo_allocate ( void )
2012-01-03 11:02:51 +01:00
{
struct iio_kfifo * kf ;
2014-12-19 18:39:24 +01:00
kf = kzalloc ( sizeof ( * kf ) , GFP_KERNEL ) ;
2012-01-03 11:02:51 +01:00
if ( ! kf )
return NULL ;
2014-12-19 18:39:24 +01:00
2012-01-03 11:02:51 +01:00
kf - > update_needed = true ;
iio_buffer_init ( & kf - > buffer ) ;
kf - > buffer . access = & kfifo_access_funcs ;
2012-06-30 13:52:00 +01:00
kf - > buffer . length = 2 ;
2013-10-15 09:30:00 +01:00
mutex_init ( & kf - > user_lock ) ;
2014-12-19 18:39:24 +01:00
2012-01-03 11:02:51 +01:00
return & kf - > buffer ;
}
EXPORT_SYMBOL ( iio_kfifo_allocate ) ;
void iio_kfifo_free ( struct iio_buffer * r )
{
2013-10-04 12:06:00 +01:00
iio_buffer_put ( r ) ;
2012-01-03 11:02:51 +01:00
}
EXPORT_SYMBOL ( iio_kfifo_free ) ;
2011-05-18 14:42:24 +01:00
2014-12-19 18:39:25 +01:00
static void devm_iio_kfifo_release ( struct device * dev , void * res )
{
iio_kfifo_free ( * ( struct iio_buffer * * ) res ) ;
}
static int devm_iio_kfifo_match ( struct device * dev , void * res , void * data )
{
struct iio_buffer * * r = res ;
if ( WARN_ON ( ! r | | ! * r ) )
return 0 ;
return * r = = data ;
}
/**
* devm_iio_fifo_allocate - Resource - managed iio_kfifo_allocate ( )
* @ dev : Device to allocate kfifo buffer for
*
* RETURNS :
* Pointer to allocated iio_buffer on success , NULL on failure .
*/
struct iio_buffer * devm_iio_kfifo_allocate ( struct device * dev )
{
struct iio_buffer * * ptr , * r ;
ptr = devres_alloc ( devm_iio_kfifo_release , sizeof ( * ptr ) , GFP_KERNEL ) ;
if ( ! ptr )
return NULL ;
r = iio_kfifo_allocate ( ) ;
if ( r ) {
* ptr = r ;
devres_add ( dev , ptr ) ;
} else {
devres_free ( ptr ) ;
}
return r ;
}
EXPORT_SYMBOL ( devm_iio_kfifo_allocate ) ;
/**
* devm_iio_fifo_free - Resource - managed iio_kfifo_free ( )
* @ dev : Device the buffer belongs to
* @ r : The buffer associated with the device
*/
void devm_iio_kfifo_free ( struct device * dev , struct iio_buffer * r )
{
WARN_ON ( devres_release ( dev , devm_iio_kfifo_release ,
devm_iio_kfifo_match , r ) ) ;
}
EXPORT_SYMBOL ( devm_iio_kfifo_free ) ;
2011-02-11 13:09:10 +00:00
MODULE_LICENSE ( " GPL " ) ;