ringbuffer.h 7.8 KB

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