Home | History | Annotate | Download | only in decpp
      1 #ifndef _DETHREADSAFERINGBUFFER_HPP
      2 #define _DETHREADSAFERINGBUFFER_HPP
      3 /*-------------------------------------------------------------------------
      4  * drawElements C++ Base Library
      5  * -----------------------------
      6  *
      7  * Copyright 2014 The Android Open Source Project
      8  *
      9  * Licensed under the Apache License, Version 2.0 (the "License");
     10  * you may not use this file except in compliance with the License.
     11  * You may obtain a copy of the License at
     12  *
     13  *      http://www.apache.org/licenses/LICENSE-2.0
     14  *
     15  * Unless required by applicable law or agreed to in writing, software
     16  * distributed under the License is distributed on an "AS IS" BASIS,
     17  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     18  * See the License for the specific language governing permissions and
     19  * limitations under the License.
     20  *
     21  *//*!
     22  * \file
     23  * \brief Thread-safe ring buffer template.
     24  *//*--------------------------------------------------------------------*/
     25 
     26 #include "deDefs.hpp"
     27 #include "deMutex.hpp"
     28 #include "deSemaphore.hpp"
     29 
     30 #include <vector>
     31 
     32 namespace de
     33 {
     34 
     35 void ThreadSafeRingBuffer_selfTest (void);
     36 
     37 /** Thread-safe ring buffer template. */
     38 template <typename T>
     39 class ThreadSafeRingBuffer
     40 {
     41 public:
     42 					ThreadSafeRingBuffer	(size_t size);
     43 					~ThreadSafeRingBuffer	(void) {}
     44 
     45 	void			pushFront				(const T& elem);
     46 	bool			tryPushFront			(const T& elem);
     47 	T				popBack					(void);
     48 	bool			tryPopBack				(T& dst);
     49 
     50 protected:
     51 	void			pushFrontInternal		(const T& elem);
     52 	T				popBackInternal			(void);
     53 
     54 	const size_t	m_size;
     55 	std::vector<T>	m_elements;
     56 
     57 	size_t			m_front;
     58 	size_t			m_back;
     59 
     60 	Mutex			m_writeMutex;
     61 	Mutex			m_readMutex;
     62 
     63 	Semaphore		m_fill;
     64 	Semaphore		m_empty;
     65 };
     66 
     67 // ThreadSafeRingBuffer implementation.
     68 
     69 template <typename T>
     70 ThreadSafeRingBuffer<T>::ThreadSafeRingBuffer (size_t size)
     71 	: m_size		(size+1)
     72 	, m_elements	(m_size)
     73 	, m_front		(0)
     74 	, m_back		(0)
     75 	, m_fill		(0)
     76 	, m_empty		((int)size)
     77 {
     78 	// Semaphores currently only support INT_MAX
     79 	DE_ASSERT(size > 0 && size < 0x7fffffff);
     80 }
     81 
     82 template <typename T>
     83 inline void ThreadSafeRingBuffer<T>::pushFrontInternal (const T& elem)
     84 {
     85 	m_elements[m_front] = elem;
     86 	m_front = (m_front + 1) % m_size;
     87 }
     88 
     89 template <typename T>
     90 inline T ThreadSafeRingBuffer<T>::popBackInternal ()
     91 {
     92 	const size_t ndx = m_back;
     93 	m_back = (m_back + 1) % m_size;
     94 	return m_elements[ndx];
     95 }
     96 
     97 template <typename T>
     98 void ThreadSafeRingBuffer<T>::pushFront (const T& elem)
     99 {
    100 	m_writeMutex.lock();
    101 	m_empty.decrement();
    102 	pushFrontInternal(elem);
    103 	m_fill.increment();
    104 	m_writeMutex.unlock();
    105 }
    106 
    107 template <typename T>
    108 bool ThreadSafeRingBuffer<T>::tryPushFront (const T& elem)
    109 {
    110 	if (!m_writeMutex.tryLock())
    111 		return false;
    112 
    113 	const bool success = m_empty.tryDecrement();
    114 
    115 	if (success)
    116 	{
    117 		pushFrontInternal(elem);
    118 		m_fill.increment();
    119 	}
    120 
    121 	m_writeMutex.unlock();
    122 	return success;
    123 }
    124 
    125 template <typename T>
    126 T ThreadSafeRingBuffer<T>::popBack ()
    127 {
    128 	m_readMutex.lock();
    129 	m_fill.decrement();
    130 	T elem = popBackInternal();
    131 	m_empty.increment();
    132 	m_readMutex.unlock();
    133 	return elem;
    134 }
    135 
    136 template <typename T>
    137 bool ThreadSafeRingBuffer<T>::tryPopBack (T& dst)
    138 {
    139 	if (!m_readMutex.tryLock())
    140 		return false;
    141 
    142 	bool success = m_fill.tryDecrement();
    143 
    144 	if (success)
    145 	{
    146 		dst = popBackInternal();
    147 		m_empty.increment();
    148 	}
    149 
    150 	m_readMutex.unlock();
    151 
    152 	return success;
    153 }
    154 
    155 } // de
    156 
    157 #endif // _DETHREADSAFERINGBUFFER_HPP
    158