ring_buffer.c 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /*
  2. * @brief Common ring buffer support functions
  3. *
  4. * @note
  5. * Copyright(C) NXP Semiconductors, 2012
  6. * All rights reserved.
  7. *
  8. * @par
  9. * Software that is described herein is for illustrative purposes only
  10. * which provides customers with programming information regarding the
  11. * LPC products. This software is supplied "AS IS" without any warranties of
  12. * any kind, and NXP Semiconductors and its licensor disclaim any and
  13. * all warranties, express or implied, including all implied warranties of
  14. * merchantability, fitness for a particular purpose and non-infringement of
  15. * intellectual property rights. NXP Semiconductors assumes no responsibility
  16. * or liability for the use of the software, conveys no license or rights under any
  17. * patent, copyright, mask work right, or any other intellectual property rights in
  18. * or to any products. NXP Semiconductors reserves the right to make changes
  19. * in the software without notification. NXP Semiconductors also makes no
  20. * representation or warranty that such application will be suitable for the
  21. * specified use without further testing or modification.
  22. *
  23. * @par
  24. * Permission to use, copy, modify, and distribute this software and its
  25. * documentation is hereby granted, under NXP Semiconductors' and its
  26. * licensor's relevant copyrights in the software, without fee, provided that it
  27. * is used in conjunction with NXP Semiconductors microcontrollers. This
  28. * copyright, permission, and disclaimer notice must appear in all copies of
  29. * this code.
  30. */
  31. #include <string.h>
  32. #include "ring_buffer.h"
  33. /*****************************************************************************
  34. * Private types/enumerations/variables
  35. ****************************************************************************/
  36. #define RB_INDH(rb) ((rb)->head & ((rb)->count - 1))
  37. #define RB_INDT(rb) ((rb)->tail & ((rb)->count - 1))
  38. /*****************************************************************************
  39. * Public types/enumerations/variables
  40. ****************************************************************************/
  41. /*****************************************************************************
  42. * Private functions
  43. ****************************************************************************/
  44. /*****************************************************************************
  45. * Public functions
  46. ****************************************************************************/
  47. /* Initialize ring buffer */
  48. int RingBuffer_Init(RINGBUFF_T *RingBuff, void *buffer, int itemSize, int count)
  49. {
  50. RingBuff->data = buffer;
  51. RingBuff->count = count;
  52. RingBuff->itemSz = itemSize;
  53. RingBuff->head = RingBuff->tail = 0;
  54. return 1;
  55. }
  56. /* Insert a single item into Ring Buffer */
  57. int RingBuffer_Insert(RINGBUFF_T *RingBuff, const void *data)
  58. {
  59. uint8_t *ptr = RingBuff->data;
  60. /* We cannot insert when queue is full */
  61. if (RingBuffer_IsFull(RingBuff))
  62. return 0;
  63. ptr += RB_INDH(RingBuff) * RingBuff->itemSz;
  64. memcpy(ptr, data, RingBuff->itemSz);
  65. RingBuff->head++;
  66. return 1;
  67. }
  68. /* Insert multiple items into Ring Buffer */
  69. int RingBuffer_InsertMult(RINGBUFF_T *RingBuff, const void *data, int num)
  70. {
  71. uint8_t *ptr = RingBuff->data;
  72. int cnt1, cnt2;
  73. /* We cannot insert when queue is full */
  74. if (RingBuffer_IsFull(RingBuff))
  75. return 0;
  76. /* Calculate the segment lengths */
  77. cnt1 = cnt2 = RingBuffer_GetFree(RingBuff);
  78. if (RB_INDH(RingBuff) + cnt1 >= RingBuff->count)
  79. cnt1 = RingBuff->count - RB_INDH(RingBuff);
  80. cnt2 -= cnt1;
  81. cnt1 = MIN(cnt1, num);
  82. num -= cnt1;
  83. cnt2 = MIN(cnt2, num);
  84. num -= cnt2;
  85. /* Write segment 1 */
  86. ptr += RB_INDH(RingBuff) * RingBuff->itemSz;
  87. memcpy(ptr, data, cnt1 * RingBuff->itemSz);
  88. RingBuff->head += cnt1;
  89. /* Write segment 2 */
  90. ptr = (uint8_t *) RingBuff->data + RB_INDH(RingBuff) * RingBuff->itemSz;
  91. data = (const uint8_t *) data + cnt1 * RingBuff->itemSz;
  92. memcpy(ptr, data, cnt2 * RingBuff->itemSz);
  93. RingBuff->head += cnt2;
  94. return cnt1 + cnt2;
  95. }
  96. /* Pop single item from Ring Buffer */
  97. int RingBuffer_Pop(RINGBUFF_T *RingBuff, void *data)
  98. {
  99. uint8_t *ptr = RingBuff->data;
  100. /* We cannot pop when queue is empty */
  101. if (RingBuffer_IsEmpty(RingBuff))
  102. return 0;
  103. ptr += RB_INDT(RingBuff) * RingBuff->itemSz;
  104. memcpy(data, ptr, RingBuff->itemSz);
  105. RingBuff->tail++;
  106. return 1;
  107. }
  108. /* Pop multiple items from Ring buffer */
  109. int RingBuffer_PopMult(RINGBUFF_T *RingBuff, void *data, int num)
  110. {
  111. uint8_t *ptr = RingBuff->data;
  112. int cnt1, cnt2;
  113. /* We cannot insert when queue is empty */
  114. if (RingBuffer_IsEmpty(RingBuff))
  115. return 0;
  116. /* Calculate the segment lengths */
  117. cnt1 = cnt2 = RingBuffer_GetCount(RingBuff);
  118. if (RB_INDT(RingBuff) + cnt1 >= RingBuff->count)
  119. cnt1 = RingBuff->count - RB_INDT(RingBuff);
  120. cnt2 -= cnt1;
  121. cnt1 = MIN(cnt1, num);
  122. num -= cnt1;
  123. cnt2 = MIN(cnt2, num);
  124. num -= cnt2;
  125. /* Write segment 1 */
  126. ptr += RB_INDT(RingBuff) * RingBuff->itemSz;
  127. memcpy(data, ptr, cnt1 * RingBuff->itemSz);
  128. RingBuff->tail += cnt1;
  129. /* Write segment 2 */
  130. ptr = (uint8_t *) RingBuff->data + RB_INDT(RingBuff) * RingBuff->itemSz;
  131. data = (uint8_t *) data + cnt1 * RingBuff->itemSz;
  132. memcpy(data, ptr, cnt2 * RingBuff->itemSz);
  133. RingBuff->tail += cnt2;
  134. return cnt1 + cnt2;
  135. }