hbuf.h 5.1 KB

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