ringbuffer.h 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. /*
  2. * Copyright 2010-2011 Branimir Karadzic. All rights reserved.
  3. * License: http://www.opensource.org/licenses/BSD-2-Clause
  4. */
  5. #ifndef __BX_RINGBUFFER_H__
  6. #define __BX_RINGBUFFER_H__
  7. #include "bx.h"
  8. #include "cpu.h"
  9. #include "uint32_t.h"
  10. namespace bx
  11. {
  12. class RingBufferControl
  13. {
  14. public:
  15. RingBufferControl(uint32_t _size)
  16. : m_size(_size)
  17. , m_current(0)
  18. , m_write(0)
  19. , m_read(0)
  20. {
  21. }
  22. ~RingBufferControl()
  23. {
  24. }
  25. uint32_t available() const
  26. {
  27. return distance(m_read, m_current);
  28. }
  29. uint32_t consume(uint32_t _size) // consumer only
  30. {
  31. const uint32_t maxSize = distance(m_read, m_current);
  32. const uint32_t sizeNoSign = uint32_and(_size, 0x7FFFFFFF);
  33. const uint32_t test = uint32_sub(sizeNoSign, maxSize);
  34. const uint32_t size = uint32_sels(test, _size, maxSize);
  35. const uint32_t advance = uint32_add(m_read, size);
  36. const uint32_t read = uint32_mod(advance, m_size);
  37. m_read = read;
  38. return size;
  39. }
  40. uint32_t reserve(uint32_t _size) // producer only
  41. {
  42. const uint32_t dist = distance(m_write, m_read)-1;
  43. const uint32_t maxSize = uint32_sels(dist, m_size-1, dist);
  44. const uint32_t sizeNoSign = uint32_and(_size, 0x7FFFFFFF);
  45. const uint32_t test = uint32_sub(sizeNoSign, maxSize);
  46. const uint32_t size = uint32_sels(test, _size, maxSize);
  47. const uint32_t advance = uint32_add(m_write, size);
  48. const uint32_t write = uint32_mod(advance, m_size);
  49. m_write = write;
  50. return size;
  51. }
  52. uint32_t commit(uint32_t _size) // producer only
  53. {
  54. const uint32_t maxSize = distance(m_current, m_write);
  55. const uint32_t sizeNoSign = uint32_and(_size, 0x7FFFFFFF);
  56. const uint32_t test = uint32_sub(sizeNoSign, maxSize);
  57. const uint32_t size = uint32_sels(test, _size, maxSize);
  58. const uint32_t advance = uint32_add(m_current, size);
  59. const uint32_t current = uint32_mod(advance, m_size);
  60. m_current = current;
  61. return size;
  62. }
  63. uint32_t distance(uint32_t _from, uint32_t _to) const // both
  64. {
  65. const uint32_t diff = uint32_sub(_to, _from);
  66. const uint32_t le = uint32_add(m_size, diff);
  67. const uint32_t result = uint32_sels(diff, le, diff);
  68. return result;
  69. }
  70. const uint32_t m_size;
  71. uint32_t m_current;
  72. uint32_t m_write;
  73. uint32_t m_read;
  74. };
  75. class SpScRingBufferControl
  76. {
  77. public:
  78. SpScRingBufferControl(uint32_t _size)
  79. : m_size(_size)
  80. , m_current(0)
  81. , m_write(0)
  82. , m_read(0)
  83. {
  84. }
  85. ~SpScRingBufferControl()
  86. {
  87. }
  88. uint32_t available() const
  89. {
  90. return distance(m_read, m_current);
  91. }
  92. uint32_t consume(uint32_t _size) // consumer only
  93. {
  94. const uint32_t maxSize = distance(m_read, m_current);
  95. const uint32_t sizeNoSign = uint32_and(_size, 0x7FFFFFFF);
  96. const uint32_t test = uint32_sub(sizeNoSign, maxSize);
  97. const uint32_t size = uint32_sels(test, _size, maxSize);
  98. const uint32_t advance = uint32_add(m_read, size);
  99. const uint32_t read = uint32_mod(advance, m_size);
  100. m_read = read;
  101. return size;
  102. }
  103. uint32_t reserve(uint32_t _size) // producer only
  104. {
  105. const uint32_t dist = distance(m_write, m_read)-1;
  106. const uint32_t maxSize = uint32_sels(dist, m_size-1, dist);
  107. const uint32_t sizeNoSign = uint32_and(_size, 0x7FFFFFFF);
  108. const uint32_t test = uint32_sub(sizeNoSign, maxSize);
  109. const uint32_t size = uint32_sels(test, _size, maxSize);
  110. const uint32_t advance = uint32_add(m_write, size);
  111. const uint32_t write = uint32_mod(advance, m_size);
  112. m_write = write;
  113. return size;
  114. }
  115. uint32_t commit(uint32_t _size) // producer only
  116. {
  117. const uint32_t maxSize = distance(m_current, m_write);
  118. const uint32_t sizeNoSign = uint32_and(_size, 0x7FFFFFFF);
  119. const uint32_t test = uint32_sub(sizeNoSign, maxSize);
  120. const uint32_t size = uint32_sels(test, _size, maxSize);
  121. const uint32_t advance = uint32_add(m_current, size);
  122. const uint32_t current = uint32_mod(advance, m_size);
  123. // must commit all memory writes before moving m_current pointer
  124. // once m_current pointer moves data is used by consumer thread
  125. memoryBarrier();
  126. m_current = current;
  127. return size;
  128. }
  129. uint32_t distance(uint32_t _from, uint32_t _to) const // both
  130. {
  131. const uint32_t diff = uint32_sub(_to, _from);
  132. const uint32_t le = uint32_add(m_size, diff);
  133. const uint32_t result = uint32_sels(diff, le, diff);
  134. return result;
  135. }
  136. const uint32_t m_size;
  137. uint32_t m_current;
  138. uint32_t m_write;
  139. uint32_t m_read;
  140. };
  141. template <typename Control>
  142. class ReadRingBufferT
  143. {
  144. public:
  145. ReadRingBufferT(Control& _control, const char* _buffer, uint32_t _size)
  146. : m_control(_control)
  147. , m_read(_control.m_read)
  148. , m_end(m_read+_size)
  149. , m_size(_size)
  150. , m_buffer(_buffer)
  151. {
  152. BX_CHECK(_control.available() >= _size, "%d >= %d", _control.available(), _size);
  153. }
  154. ~ReadRingBufferT()
  155. {
  156. }
  157. void end()
  158. {
  159. m_control.consume(m_size);
  160. }
  161. void read(char* _data, uint32_t _len)
  162. {
  163. const uint32_t end = (m_read + _len) % m_control.m_size;
  164. uint32_t wrap = 0;
  165. const char* from = &m_buffer[m_read];
  166. if (end < m_read)
  167. {
  168. wrap = m_control.m_size - m_read;
  169. memcpy(_data, from, wrap);
  170. _data += wrap;
  171. from = (const char*)&m_buffer[0];
  172. }
  173. memcpy(_data, from, _len-wrap);
  174. m_read = end;
  175. }
  176. void skip(uint32_t _len)
  177. {
  178. m_read += _len;
  179. m_read %= m_control.m_size;
  180. }
  181. private:
  182. template <typename Ty>
  183. friend class WriteRingBufferT;
  184. ReadRingBufferT();
  185. ReadRingBufferT(const Control&);
  186. void operator=(const Control&);
  187. Control& m_control;
  188. uint32_t m_read;
  189. uint32_t m_end;
  190. const uint32_t m_size;
  191. const char* m_buffer;
  192. };
  193. typedef ReadRingBufferT<RingBufferControl> ReadRingBuffer;
  194. typedef ReadRingBufferT<SpScRingBufferControl> SpScReadRingBuffer;
  195. template <typename Control>
  196. class WriteRingBufferT
  197. {
  198. public:
  199. WriteRingBufferT(Control& _control, char* _buffer, uint32_t _size)
  200. : m_control(_control)
  201. , m_size(_size)
  202. , m_buffer(_buffer)
  203. {
  204. uint32_t size = m_control.reserve(_size);
  205. BX_CHECK(size == _size, "%d == %d", size, _size);
  206. m_write = m_control.m_current;
  207. m_end = m_write+_size;
  208. }
  209. ~WriteRingBufferT()
  210. {
  211. }
  212. void end()
  213. {
  214. m_control.commit(m_size);
  215. }
  216. void write(const char* _data, uint32_t _len)
  217. {
  218. const uint32_t end = (m_write + _len) % m_control.m_size;
  219. uint32_t wrap = 0;
  220. char* to = &m_buffer[m_write];
  221. if (end < m_write)
  222. {
  223. wrap = m_control.m_size - m_write;
  224. memcpy(to, _data, wrap);
  225. _data += wrap;
  226. to = (char*)&m_buffer[0];
  227. }
  228. memcpy(to, _data, _len-wrap);
  229. m_write = end;
  230. }
  231. void write(ReadRingBufferT<Control>& _read, uint32_t _len)
  232. {
  233. const uint32_t end = (_read.m_read + _len) % _read.m_control.m_size;
  234. uint32_t wrap = 0;
  235. const char* from = &_read.m_buffer[_read.m_read];
  236. if (end < _read.m_read)
  237. {
  238. wrap = _read.m_control.m_size - _read.m_read;
  239. write(from, wrap);
  240. from = (const char*)&_read.m_buffer[0];
  241. }
  242. write(from, _len-wrap);
  243. _read.m_read = end;
  244. }
  245. void skip(uint32_t _len)
  246. {
  247. m_write += _len;
  248. m_write %= m_control.m_size;
  249. }
  250. private:
  251. WriteRingBufferT();
  252. WriteRingBufferT(const WriteRingBufferT<Control>&);
  253. void operator=(const WriteRingBufferT<Control>&);
  254. Control& m_control;
  255. uint32_t m_write;
  256. uint32_t m_end;
  257. const uint32_t m_size;
  258. char* m_buffer;
  259. };
  260. typedef WriteRingBufferT<RingBufferControl> WriteRingBuffer;
  261. typedef WriteRingBufferT<SpScRingBufferControl> SpScWriteRingBuffer;
  262. } // namespace bx
  263. #endif // __BX_RINGBUFFER_H__