rbuf_core.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. /*******************************************************************************
  2. * @file rbuf_core.c
  3. * @author MEMS Application Team
  4. * @version V1.0
  5. * @date 2020-10-15
  6. * @brief ring buffer for interaction RAM
  7. *******************************************************************************/
  8. /******************************************************************************/
  9. //includes
  10. /******************************************************************************/
  11. #include <string.h>
  12. #include <rbuf/rbuf_core.h>
  13. /******************************************************************************/
  14. //constants
  15. /******************************************************************************/
  16. /******************************************************************************/
  17. //typedef
  18. /******************************************************************************/
  19. /******************************************************************************/
  20. //variables
  21. /******************************************************************************/
  22. /******************************************************************************/
  23. //functions
  24. /******************************************************************************/
  25. static unsigned int wrap(unsigned int val, unsigned int max)
  26. {
  27. return val >= max ? (val - max) : val;
  28. }
  29. static unsigned int _get_space(unsigned int size, unsigned int head, unsigned int tail)
  30. {
  31. if (tail < head) {
  32. return head - tail - 1;
  33. } else {
  34. return (size - tail) + head - 1;
  35. }
  36. }
  37. static unsigned int _get_rbuf_space(rbuf_t *buf)
  38. {
  39. return _get_space(buf->size, buf->head, buf->tail);
  40. }
  41. static unsigned int _get_size(rbuf_t *buf, unsigned int size)
  42. {
  43. return buf->hlen ? ROUND_UP(size, 4) + buf->hlen : size;
  44. }
  45. rbuf_t *rbuf_init_buf(char *buf, unsigned int size, unsigned int mode)
  46. {
  47. rbuf_t *rbuf = (rbuf_t*)buf;
  48. rbuf->head = rbuf->tmp_head = 0;
  49. rbuf->tail = rbuf->tmp_tail = 0;
  50. rbuf->buf_off = RBUF_TO_OF(buf) + sizeof(rbuf_t);
  51. rbuf->size = size - sizeof(rbuf_t);
  52. if (mode == RBUF_MSG) {
  53. rbuf->hlen = sizeof(unsigned int);
  54. } else {
  55. rbuf->hlen = 0;
  56. }
  57. rbuf->next = 0;
  58. return rbuf;
  59. }
  60. unsigned int rbuf_get_space(rbuf_t *buf)
  61. {
  62. unsigned int space = _get_rbuf_space(buf);
  63. return (space < buf->hlen) ? 0 : (space - buf->hlen);
  64. }
  65. void* rbuf_put_claim(rbuf_t *buf, unsigned int size, unsigned int *psz)
  66. {
  67. unsigned int space, trail_size, allocated;
  68. unsigned int *pdata;
  69. /* Available size. */
  70. size = _get_size(buf, size);
  71. space = _get_space(buf->size, buf->head, buf->tmp_tail);
  72. trail_size = buf->size - buf->tmp_tail;
  73. /* Data pointer */
  74. pdata = (unsigned int*)RBUF_FR_OF(buf->buf_off + buf->tmp_tail);
  75. if (buf->hlen > 0) {
  76. /* Limit requested size to available size. */
  77. if (space < size) {
  78. return NULL;
  79. }
  80. /* Limit requested size to be continued */
  81. if (trail_size < size) {
  82. if ((space - trail_size) < size) {
  83. return NULL;
  84. }
  85. /* Put Null Msg and Reset tail */
  86. *pdata = 0;
  87. buf->tail = buf->tmp_tail = 0;
  88. /* Next Msg */
  89. pdata = (unsigned int*)RBUF_FR_OF(buf->buf_off);
  90. }
  91. /* Update data and size */
  92. allocated = size - buf->hlen;
  93. *pdata = allocated;
  94. pdata ++;
  95. } else {
  96. /* Limit requested size to available size. */
  97. size = MIN(size, space);
  98. /* Limit allocated size to trail size. */
  99. allocated = MIN(trail_size, size);
  100. if (allocated == 0) {
  101. return NULL;
  102. }
  103. }
  104. if (psz) {
  105. *psz = allocated;
  106. }
  107. buf->tmp_tail = wrap(buf->tmp_tail + allocated + buf->hlen, buf->size);
  108. return pdata;
  109. }
  110. int rbuf_put_finish(rbuf_t *buf, unsigned int size)
  111. {
  112. size = _get_size(buf, size);
  113. if (size > _get_rbuf_space(buf)) {
  114. return -1;
  115. }
  116. buf->tail = wrap(buf->tail + size, buf->size);
  117. buf->tmp_tail = buf->tail;
  118. return 0;
  119. }
  120. void* rbuf_get_claim(rbuf_t *buf, unsigned int size, unsigned int *psz)
  121. {
  122. unsigned int space, granted_size, trail_size;
  123. unsigned int *pdata;
  124. /* Check space */
  125. space = (buf->size - 1) - _get_space(buf->size, buf->tmp_head, buf->tail);
  126. if (space <= 0) {
  127. return NULL;
  128. }
  129. /* Data Pointer */
  130. pdata = (unsigned int*)RBUF_FR_OF(buf->buf_off + buf->tmp_head);
  131. /* Msg process */
  132. if (buf->hlen > 0) {
  133. /* Skip null data and reset head */
  134. if (*pdata == 0) {
  135. buf->head = buf->tmp_head = 0;
  136. /* Check empty */
  137. if (buf->head == buf->tail) {
  138. return NULL;
  139. }
  140. pdata = (unsigned int*)RBUF_FR_OF(buf->buf_off);
  141. }
  142. /* Update data and size */
  143. granted_size = *pdata;
  144. pdata ++;
  145. } else {
  146. /* Limit requested size to available size. */
  147. granted_size = MIN(size, space);
  148. /* Limit allocated size to trail size. */
  149. trail_size = buf->size - buf->tmp_head;
  150. granted_size = MIN(trail_size, granted_size);
  151. }
  152. if (psz) {
  153. *psz = granted_size;
  154. }
  155. buf->tmp_head = wrap(buf->tmp_head + granted_size + buf->hlen, buf->size);
  156. return pdata;
  157. }
  158. int rbuf_get_finish(rbuf_t *buf, unsigned int size)
  159. {
  160. unsigned int allocated;
  161. size = _get_size(buf, size);
  162. allocated = (buf->size - 1) - _get_rbuf_space(buf);
  163. if (size > allocated) {
  164. return -1;
  165. }
  166. buf->head = wrap(buf->head + size, buf->size);
  167. buf->tmp_head = buf->head;
  168. return 0;
  169. }
  170. unsigned int rbuf_get_hdl(rbuf_t *buf, unsigned int size, rbuf_hdl hdl, void *ctx)
  171. {
  172. void *pdata;
  173. /* read data */
  174. pdata = rbuf_get_claim(buf, size, &size);
  175. if (pdata != NULL) {
  176. /* handle data */
  177. size = (*hdl)(ctx, pdata, size);
  178. }
  179. /* finish data */
  180. rbuf_get_finish(buf, size);
  181. return size;
  182. }