2009-08-18 17:06:27 +00:00
|
|
|
/* The industrial I/O simple minimally locked ring buffer.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2008 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.
|
|
|
|
*
|
|
|
|
* This code is deliberately kept separate from the main industrialio I/O core
|
|
|
|
* as it is intended that in the future a number of different software ring
|
|
|
|
* buffer implementations will exist with different characteristics to suit
|
|
|
|
* different applications.
|
|
|
|
*
|
|
|
|
* This particular one was designed for a data capture application where it was
|
|
|
|
* particularly important that no userspace reads would interrupt the capture
|
|
|
|
* process. To this end the ring is not locked during a read.
|
|
|
|
*
|
|
|
|
* Comments on this buffer design welcomed. It's far from efficient and some of
|
|
|
|
* my understanding of the effects of scheduling on this are somewhat limited.
|
|
|
|
* Frankly, to my mind, this is the current weak point in the industrial I/O
|
|
|
|
* patch set.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _IIO_RING_SW_H_
|
|
|
|
#define _IIO_RING_SW_H_
|
|
|
|
/* NEEDS COMMENTS */
|
|
|
|
/* The intention is that this should be a separate module from the iio core.
|
|
|
|
* This is a bit like supporting algorithms dependent on what the device
|
|
|
|
* driver requests - some may support multiple options */
|
|
|
|
|
|
|
|
|
|
|
|
#include "iio.h"
|
|
|
|
#include "ring_generic.h"
|
|
|
|
|
|
|
|
#if defined CONFIG_IIO_SW_RING || defined CONFIG_IIO_SW_RING_MODULE
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_create_sw_rb() - software ring buffer allocation
|
2009-08-18 17:06:27 +00:00
|
|
|
* @r: pointer to ring buffer pointer
|
|
|
|
**/
|
|
|
|
int iio_create_sw_rb(struct iio_ring_buffer **r);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_init_sw_rb() - initialize the software ring buffer
|
2009-08-18 17:06:27 +00:00
|
|
|
* @r: pointer to a software ring buffer created by an
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_create_sw_rb call
|
|
|
|
* @indio_dev: industrial I/O device structure
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_init_sw_rb(struct iio_ring_buffer *r, struct iio_dev *indio_dev);
|
2009-10-05 02:34:02 +00:00
|
|
|
|
2009-08-18 17:06:27 +00:00
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_exit_sw_rb() - reverse what was done in iio_init_sw_rb
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
void iio_exit_sw_rb(struct iio_ring_buffer *r);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_free_sw_rb() - free memory occupied by the core ring buffer struct
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
void iio_free_sw_rb(struct iio_ring_buffer *r);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_mark_sw_rb_in_use() - reference counting to prevent incorrect chances
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
void iio_mark_sw_rb_in_use(struct iio_ring_buffer *r);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_unmark_sw_rb_in_use() - notify the ring buffer that we don't care anymore
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
void iio_unmark_sw_rb_in_use(struct iio_ring_buffer *r);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_read_last_from_sw_rb() - attempt to read the last stored datum from the rb
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
|
|
|
* @data: where to store the last datum
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_read_last_from_sw_rb(struct iio_ring_buffer *r, u8 *data);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_store_to_sw_rb() - store a new datum to the ring buffer
|
|
|
|
* @r: pointer to ring buffer instance
|
|
|
|
* @data: the datum to be stored including timestamp if relevant
|
|
|
|
* @timestamp: timestamp which will be attached to buffer events if relevant
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_store_to_sw_rb(struct iio_ring_buffer *r, u8 *data, s64 timestamp);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_rip_sw_rb() - attempt to read data from the ring buffer
|
2009-08-18 17:06:27 +00:00
|
|
|
* @r: ring buffer instance
|
|
|
|
* @count: number of datum's to try and read
|
|
|
|
* @data: where the data will be stored.
|
|
|
|
* @dead_offset: how much of the stored data was possibly invalidated by
|
|
|
|
* the end of the copy.
|
|
|
|
**/
|
|
|
|
int iio_rip_sw_rb(struct iio_ring_buffer *r,
|
|
|
|
size_t count,
|
|
|
|
u8 **data,
|
|
|
|
int *dead_offset);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_request_update_sw_rb() - update params if update needed
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_request_update_sw_rb(struct iio_ring_buffer *r);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_mark_update_needed_sw_rb() - tell the ring buffer it needs a param update
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_mark_update_needed_sw_rb(struct iio_ring_buffer *r);
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_get_bpd_sw_rb() - get the datum size in bytes
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_get_bpd_sw_rb(struct iio_ring_buffer *r);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_set_bpd_sw_rb() - set the datum size in bytes
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
|
|
|
* @bpd: bytes per datum value
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_set_bpd_sw_rb(struct iio_ring_buffer *r, size_t bpd);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_get_length_sw_rb() - get how many datums the rb may contain
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_get_length_sw_rb(struct iio_ring_buffer *r);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_set_length_sw_rb() - set how many datums the rb may contain
|
|
|
|
* @r: pointer to a software ring buffer created by an
|
|
|
|
* iio_create_sw_rb call
|
|
|
|
* @length: max number of data items for the ring buffer
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
int iio_set_length_sw_rb(struct iio_ring_buffer *r, int length);
|
|
|
|
|
|
|
|
/**
|
2009-10-05 02:34:02 +00:00
|
|
|
* iio_ring_sw_register_funcs() - helper function to set up rb access
|
|
|
|
* @ra: pointer to @iio_ring_access_funcs
|
2009-08-18 17:06:27 +00:00
|
|
|
**/
|
|
|
|
static inline void iio_ring_sw_register_funcs(struct iio_ring_access_funcs *ra)
|
|
|
|
{
|
|
|
|
ra->mark_in_use = &iio_mark_sw_rb_in_use;
|
|
|
|
ra->unmark_in_use = &iio_unmark_sw_rb_in_use;
|
|
|
|
|
|
|
|
ra->store_to = &iio_store_to_sw_rb;
|
|
|
|
ra->read_last = &iio_read_last_from_sw_rb;
|
|
|
|
ra->rip_lots = &iio_rip_sw_rb;
|
|
|
|
|
|
|
|
ra->mark_param_change = &iio_mark_update_needed_sw_rb;
|
|
|
|
ra->request_update = &iio_request_update_sw_rb;
|
|
|
|
|
|
|
|
ra->get_bpd = &iio_get_bpd_sw_rb;
|
|
|
|
ra->set_bpd = &iio_set_bpd_sw_rb;
|
|
|
|
|
|
|
|
ra->get_length = &iio_get_length_sw_rb;
|
|
|
|
ra->set_length = &iio_set_length_sw_rb;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct iio_sw_ring_buffer - software ring buffer
|
|
|
|
* @buf: generic ring buffer elements
|
|
|
|
* @data: the ring buffer memory
|
|
|
|
* @read_p: read pointer (oldest available)
|
|
|
|
* @write_p: write pointer
|
|
|
|
* @last_written_p: read pointer (newest available)
|
|
|
|
* @half_p: half buffer length behind write_p (event generation)
|
|
|
|
* @use_count: reference count to prevent resizing when in use
|
|
|
|
* @update_needed: flag to indicated change in size requested
|
|
|
|
* @use_lock: lock to prevent change in size when in use
|
|
|
|
*
|
|
|
|
* Note that the first element of all ring buffers must be a
|
|
|
|
* struct iio_ring_buffer.
|
|
|
|
**/
|
|
|
|
|
|
|
|
struct iio_sw_ring_buffer {
|
|
|
|
struct iio_ring_buffer buf;
|
|
|
|
unsigned char *data;
|
|
|
|
unsigned char *read_p;
|
|
|
|
unsigned char *write_p;
|
|
|
|
unsigned char *last_written_p;
|
|
|
|
/* used to act as a point at which to signal an event */
|
|
|
|
unsigned char *half_p;
|
|
|
|
int use_count;
|
|
|
|
int update_needed;
|
|
|
|
spinlock_t use_lock;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define iio_to_sw_ring(r) container_of(r, struct iio_sw_ring_buffer, buf)
|
|
|
|
|
|
|
|
struct iio_ring_buffer *iio_sw_rb_allocate(struct iio_dev *indio_dev);
|
|
|
|
void iio_sw_rb_free(struct iio_ring_buffer *ring);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#else /* CONFIG_IIO_RING_BUFFER*/
|
|
|
|
static inline void iio_ring_sw_register_funcs(struct iio_ring_access_funcs *ra)
|
|
|
|
{};
|
|
|
|
#endif /* !CONFIG_IIO_RING_BUFFER */
|
|
|
|
#endif /* _IIO_RING_SW_H_ */
|