summaryrefslogtreecommitdiffstats
path: root/raul/SRMWQueue.h
diff options
context:
space:
mode:
authorDavid Robillard <d@drobilla.net>2007-01-22 04:07:53 +0000
committerDavid Robillard <d@drobilla.net>2007-01-22 04:07:53 +0000
commitae90063afb7ddabcb505f8390b3b90bc7fea96ca (patch)
tree361c26eed5293d70ce662549172533967e7b48de /raul/SRMWQueue.h
parentca32b189529ffc7c1d42af85f005dea54189218f (diff)
downloadraul-ae90063afb7ddabcb505f8390b3b90bc7fea96ca.tar.gz
raul-ae90063afb7ddabcb505f8390b3b90bc7fea96ca.tar.bz2
raul-ae90063afb7ddabcb505f8390b3b90bc7fea96ca.zip
Added atomic int/pointer classes to Raul.
Added multi-writer queue to Raul. Renamed Queue SRSWQueue (single-reader single-writer). Updated patchage/ingen for Raul changes. git-svn-id: http://svn.drobilla.net/lad/raul@264 a436a847-0d15-0410-975c-d299462d15a1
Diffstat (limited to 'raul/SRMWQueue.h')
-rw-r--r--raul/SRMWQueue.h201
1 files changed, 201 insertions, 0 deletions
diff --git a/raul/SRMWQueue.h b/raul/SRMWQueue.h
new file mode 100644
index 0000000..2a4955c
--- /dev/null
+++ b/raul/SRMWQueue.h
@@ -0,0 +1,201 @@
+/* This file is part of Ingen. Copyright (C) 2006 Dave Robillard.
+ *
+ * Ingen is free software; you can redistribute it and/or modify it under the
+ * terms of the GNU General Public License as published by the Free Software
+ * Foundation; either version 2 of the License, or (at your option) any later
+ * version.
+ *
+ * Ingen is distributed in the hope that it will be useful, but WITHOUT ANY
+ * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
+ * FOR A PARTICULAR PURPOSE. See the GNU General Public License for details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+#ifndef RAUL_SRMW_QUEUE_H
+#define RAUL_SRMW_QUEUE_H
+
+#include <cassert>
+#include <cstdlib>
+#include <boost/utility.hpp>
+#include "raul/AtomicInt.h"
+
+#include <iostream>
+using namespace std;
+
+
+/** Realtime-safe single-reader multi-writer queue (aka lock-free ringbuffer)
+ *
+ * Implemented as a dequeue in a fixed array. Both push and pop are realtime
+ * safe, but only push is threadsafe. In other words, multiple threads can push
+ * data into this queue for a single thread to consume.
+ *
+ * The interface is intentionally as similar to std::queue as possible, but
+ * note the additional thread restrictions imposed (e.g. empty() is only
+ * legal to call in the read thread).
+ *
+ * Obey the threading restrictions documented here, or horrible nasty (possibly
+ * undetected) errors will occur.
+ *
+ * This is slightly more expensive than the SRMWSRMWQueue. Use that if you don't
+ * require multiple writers.
+ *
+ * \ingroup raul
+ */
+template <typename T>
+class SRMWQueue : boost::noncopyable
+{
+public:
+ SRMWQueue(size_t size);
+ ~SRMWQueue();
+
+
+ // Any thread:
+
+ inline size_t capacity() const { return _size-1; }
+
+
+ // Write thread(s):
+
+ inline bool full() const;
+ inline bool push(const T& obj);
+
+
+ // Read thread:
+
+ inline bool empty() const;
+ inline T& front() const;
+ inline void pop();
+
+private:
+
+ // Note that _front doesn't need to be an AtomicInt since it's only accessed
+ // by the (single) reader thread
+
+ int _front; ///< Circular index of element at front of queue (READER ONLY)
+ AtomicInt _back; ///< Circular index 1 past element at back of queue (WRITERS ONLY)
+ AtomicInt _space; ///< Remaining free space for new elements (all threads)
+ const int _size; ///< Size of @ref _objects (you can store _size-1 objects)
+ T* const _objects; ///< Fixed array containing queued elements
+};
+
+
+template<typename T>
+SRMWQueue<T>::SRMWQueue(size_t size)
+ : _front(0)
+ , _back(0)
+ , _space(size)
+ , _size(size+1)
+ , _objects((T*)calloc(_size, sizeof(T)))
+{
+ assert(size > 1);
+ assert(_size-1 == _space.get());
+}
+
+
+template <typename T>
+SRMWQueue<T>::~SRMWQueue()
+{
+ free(_objects);
+}
+
+
+/** Return whether the queue is full.
+ *
+ * Write thread(s) only.
+ */
+template <typename T>
+inline bool
+SRMWQueue<T>::full() const
+{
+ return ( _space.get() <= 0 );
+}
+
+
+/** Push an item onto the back of the SRMWQueue - realtime-safe, not thread-safe.
+ *
+ * Write thread(s) only.
+ *
+ * @returns true if @a elem was successfully pushed onto the queue,
+ * false otherwise (queue is full).
+ */
+template <typename T>
+inline bool
+SRMWQueue<T>::push(const T& elem)
+{
+ const int old_space = _space.exchange_and_add(-1);
+
+ const bool already_full = ( old_space <= 0 );
+
+ /* Technically right here pop could be called in the reader thread and
+ * make space available, but no harm in failing anyway - this queue
+ * really isn't designed to be filled... */
+
+ if (already_full) {
+
+ /* if multiple threads simultaneously get here, _space may be 0
+ * or negative. The next call to pop() will set _space back to
+ * a sane value. Note that _space is not exposed, so this is okay
+ * (... assuming this code is correct) */
+
+ return false;
+
+ } else {
+
+ const int write_index = _back.exchange_and_add(1) % _size;
+ _objects[write_index] = elem;
+
+ return true;
+
+ }
+}
+
+
+/** Return whether the queue is empty.
+ *
+ * Read thread only.
+ */
+template <typename T>
+inline bool
+SRMWQueue<T>::empty() const
+{
+ return ( _space == capacity() );
+}
+
+
+/** Return the element at the front of the queue without removing it.
+ *
+ * It is a fatal error to call front() when the queue is empty.
+ * Read thread only.
+ */
+template <typename T>
+inline T&
+SRMWQueue<T>::front() const
+{
+ return _objects[_front];
+}
+
+
+/** Pop an item off the front of the queue - realtime-safe, NOT thread-safe.
+ *
+ * It is a fatal error to call pop() if the queue is empty.
+ * Read thread only.
+ *
+ * @return true if queue is now empty, otherwise false.
+ */
+template <typename T>
+inline void
+SRMWQueue<T>::pop()
+{
+ _front = (_front + 1) % (_size);
+
+ if (_space.get() < 0)
+ _space = 1;
+ else
+ ++_space;
+}
+
+
+#endif // RAUL_SRMW_QUEUE_H