hbuf.h 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. #ifndef HW_BUF_H_
  2. #define HW_BUF_H_
  3. #include <stdio.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6. #include "hdef.h"
  7. typedef struct hbuf_s {
  8. char* base;
  9. size_t len;
  10. #ifdef __cplusplus
  11. hbuf_s() {
  12. base = NULL;
  13. len = 0;
  14. }
  15. hbuf_s(void* data, size_t len) {
  16. this->base = (char*)data;
  17. this->len = len;
  18. }
  19. #endif
  20. } hbuf_t;
  21. typedef struct offset_buf_s {
  22. char* base;
  23. size_t len;
  24. size_t offset;
  25. #ifdef __cplusplus
  26. offset_buf_s() {
  27. base = NULL;
  28. len = offset = 0;
  29. }
  30. offset_buf_s(void* data, size_t len) {
  31. this->base = (char*)data;
  32. this->len = len;
  33. }
  34. #endif
  35. } offset_buf_t;
  36. #ifdef __cplusplus
  37. class HBuf : public hbuf_t {
  38. public:
  39. HBuf() : hbuf_t() {
  40. cleanup_ = false;
  41. }
  42. HBuf(void* data, size_t len) : hbuf_t(data, len) {
  43. cleanup_ = false;
  44. }
  45. HBuf(size_t cap) { resize(cap); }
  46. virtual ~HBuf() {
  47. cleanup();
  48. }
  49. void* data() { return base; }
  50. size_t size() { return len; }
  51. bool isNull() {
  52. return base == NULL || len == 0;
  53. }
  54. void cleanup() {
  55. if (cleanup_) {
  56. if (base) {
  57. free(base);
  58. base = NULL;
  59. }
  60. len = 0;
  61. cleanup_ = false;
  62. }
  63. }
  64. void resize(size_t cap) {
  65. if (cap == len) return;
  66. if (base == NULL) {
  67. base = (char*)malloc(cap);
  68. memset(base, 0, cap);
  69. }
  70. else {
  71. base = (char*)realloc(base, cap);
  72. }
  73. len = cap;
  74. cleanup_ = true;
  75. }
  76. void copy(void* data, size_t len) {
  77. resize(len);
  78. memcpy(base, data, len);
  79. }
  80. void copy(hbuf_t* buf) {
  81. copy(buf->base, buf->len);
  82. }
  83. private:
  84. bool cleanup_;
  85. };
  86. // VL: Variable-Length
  87. class HVLBuf : public HBuf {
  88. public:
  89. HVLBuf() : HBuf() {_offset = _size = 0;}
  90. HVLBuf(void* data, size_t len) : HBuf(data, len) {_offset = 0; _size = len;}
  91. HVLBuf(size_t cap) : HBuf(cap) {_offset = _size = 0;}
  92. virtual ~HVLBuf() {}
  93. char* data() { return base+_offset; }
  94. size_t size() { return _size; }
  95. void push_front(void* ptr, size_t len) {
  96. if (len > this->len - _size) {
  97. this->len = MAX(this->len, len)*2;
  98. base = (char*)realloc(base, this->len);
  99. }
  100. if (_offset < len) {
  101. // move => end
  102. memmove(base+this->len-_size, data(), _size);
  103. _offset = this->len-_size;
  104. }
  105. memcpy(data()-len, ptr, len);
  106. _offset -= len;
  107. _size += len;
  108. }
  109. void push_back(void* ptr, size_t len) {
  110. if (len > this->len - _size) {
  111. this->len = MAX(this->len, len)*2;
  112. base = (char*)realloc(base, this->len);
  113. }
  114. else if (len > this->len - _offset - _size) {
  115. // move => start
  116. memmove(base, data(), _size);
  117. _offset = 0;
  118. }
  119. memcpy(data()+_size, ptr, len);
  120. _size += len;
  121. }
  122. void pop_front(void* ptr, size_t len) {
  123. if (len <= _size) {
  124. if (ptr) {
  125. memcpy(ptr, data(), len);
  126. }
  127. _offset += len;
  128. if (_offset >= len) _offset = 0;
  129. _size -= len;
  130. }
  131. }
  132. void pop_back(void* ptr, size_t len) {
  133. if (len <= _size) {
  134. if (ptr) {
  135. memcpy(ptr, data()+_size-len, len);
  136. }
  137. _size -= len;
  138. }
  139. }
  140. void clear() {
  141. _offset = _size = 0;
  142. }
  143. void prepend(void* ptr, size_t len) {
  144. push_front(ptr, len);
  145. }
  146. void append(void* ptr, size_t len) {
  147. push_back(ptr, len);
  148. }
  149. void insert(void* ptr, size_t len) {
  150. push_back(ptr, len);
  151. }
  152. void remove(size_t len) {
  153. pop_front(NULL, len);
  154. }
  155. private:
  156. size_t _offset;
  157. size_t _size;
  158. };
  159. class HRingBuf : public HBuf {
  160. public:
  161. HRingBuf() : HBuf() {_head = _tail = _size = 0;}
  162. HRingBuf(size_t cap) : HBuf(cap) {_head = _tail = _size = 0;}
  163. virtual ~HRingBuf() {}
  164. char* alloc(size_t len) {
  165. char* ret = NULL;
  166. if (_head < _tail || _size == 0) {
  167. // [_tail, this->len) && [0, _head)
  168. if (this->len - _tail >= len) {
  169. ret = base + _tail;
  170. _tail += len;
  171. if (_tail == this->len) _tail = 0;
  172. }
  173. else if (_head >= len) {
  174. ret = base;
  175. _tail = len;
  176. }
  177. }
  178. else {
  179. // [_tail, _head)
  180. if (_head - _tail >= len) {
  181. ret = base + _tail;
  182. _tail += len;
  183. }
  184. }
  185. _size += ret ? len : 0;
  186. return ret;
  187. }
  188. void free(size_t len) {
  189. _size -= len;
  190. if (len <= this->len - _head) {
  191. _head += len;
  192. if (_head == this->len) _head = 0;
  193. }
  194. else {
  195. _head = len;
  196. }
  197. }
  198. size_t size() {return _size;}
  199. private:
  200. size_t _head;
  201. size_t _tail;
  202. size_t _size;
  203. };
  204. #endif
  205. #endif // HW_BUF_H_