nghttp2_buf.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494
  1. /*
  2. * nghttp2 - HTTP/2 C Library
  3. *
  4. * Copyright (c) 2014 Tatsuhiro Tsujikawa
  5. *
  6. * Permission is hereby granted, free of charge, to any person obtaining
  7. * a copy of this software and associated documentation files (the
  8. * "Software"), to deal in the Software without restriction, including
  9. * without limitation the rights to use, copy, modify, merge, publish,
  10. * distribute, sublicense, and/or sell copies of the Software, and to
  11. * permit persons to whom the Software is furnished to do so, subject to
  12. * the following conditions:
  13. *
  14. * The above copyright notice and this permission notice shall be
  15. * included in all copies or substantial portions of the Software.
  16. *
  17. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  18. * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  19. * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  20. * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  21. * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  22. * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  23. * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  24. */
  25. #include "nghttp2_buf.h"
  26. #include <stdio.h>
  27. #include "nghttp2_helper.h"
  28. void nghttp2_buf_init(nghttp2_buf *buf) {
  29. buf->begin = NULL;
  30. buf->end = NULL;
  31. buf->pos = NULL;
  32. buf->last = NULL;
  33. buf->mark = NULL;
  34. }
  35. int nghttp2_buf_init2(nghttp2_buf *buf, size_t initial, nghttp2_mem *mem) {
  36. nghttp2_buf_init(buf);
  37. return nghttp2_buf_reserve(buf, initial, mem);
  38. }
  39. void nghttp2_buf_free(nghttp2_buf *buf, nghttp2_mem *mem) {
  40. if (buf == NULL) {
  41. return;
  42. }
  43. nghttp2_mem_free(mem, buf->begin);
  44. buf->begin = NULL;
  45. }
  46. int nghttp2_buf_reserve(nghttp2_buf *buf, size_t new_cap, nghttp2_mem *mem) {
  47. uint8_t *ptr;
  48. size_t cap;
  49. cap = nghttp2_buf_cap(buf);
  50. if (cap >= new_cap) {
  51. return 0;
  52. }
  53. new_cap = nghttp2_max(new_cap, cap * 2);
  54. ptr = nghttp2_mem_realloc(mem, buf->begin, new_cap);
  55. if (ptr == NULL) {
  56. return NGHTTP2_ERR_NOMEM;
  57. }
  58. buf->pos = ptr + (buf->pos - buf->begin);
  59. buf->last = ptr + (buf->last - buf->begin);
  60. buf->mark = ptr + (buf->mark - buf->begin);
  61. buf->begin = ptr;
  62. buf->end = ptr + new_cap;
  63. return 0;
  64. }
  65. void nghttp2_buf_reset(nghttp2_buf *buf) {
  66. buf->pos = buf->last = buf->mark = buf->begin;
  67. }
  68. void nghttp2_buf_wrap_init(nghttp2_buf *buf, uint8_t *begin, size_t len) {
  69. buf->begin = buf->pos = buf->last = buf->mark = begin;
  70. buf->end = begin + len;
  71. }
  72. static int buf_chain_new(nghttp2_buf_chain **chain, size_t chunk_length,
  73. nghttp2_mem *mem) {
  74. int rv;
  75. *chain = nghttp2_mem_malloc(mem, sizeof(nghttp2_buf_chain));
  76. if (*chain == NULL) {
  77. return NGHTTP2_ERR_NOMEM;
  78. }
  79. (*chain)->next = NULL;
  80. rv = nghttp2_buf_init2(&(*chain)->buf, chunk_length, mem);
  81. if (rv != 0) {
  82. nghttp2_mem_free(mem, *chain);
  83. return NGHTTP2_ERR_NOMEM;
  84. }
  85. return 0;
  86. }
  87. static void buf_chain_del(nghttp2_buf_chain *chain, nghttp2_mem *mem) {
  88. nghttp2_buf_free(&chain->buf, mem);
  89. nghttp2_mem_free(mem, chain);
  90. }
  91. int nghttp2_bufs_init(nghttp2_bufs *bufs, size_t chunk_length, size_t max_chunk,
  92. nghttp2_mem *mem) {
  93. return nghttp2_bufs_init2(bufs, chunk_length, max_chunk, 0, mem);
  94. }
  95. int nghttp2_bufs_init2(nghttp2_bufs *bufs, size_t chunk_length,
  96. size_t max_chunk, size_t offset, nghttp2_mem *mem) {
  97. return nghttp2_bufs_init3(bufs, chunk_length, max_chunk, max_chunk, offset,
  98. mem);
  99. }
  100. int nghttp2_bufs_init3(nghttp2_bufs *bufs, size_t chunk_length,
  101. size_t max_chunk, size_t chunk_keep, size_t offset,
  102. nghttp2_mem *mem) {
  103. int rv;
  104. nghttp2_buf_chain *chain;
  105. if (chunk_keep == 0 || max_chunk < chunk_keep || chunk_length < offset) {
  106. return NGHTTP2_ERR_INVALID_ARGUMENT;
  107. }
  108. rv = buf_chain_new(&chain, chunk_length, mem);
  109. if (rv != 0) {
  110. return rv;
  111. }
  112. bufs->mem = mem;
  113. bufs->offset = offset;
  114. bufs->head = chain;
  115. bufs->cur = bufs->head;
  116. nghttp2_buf_shift_right(&bufs->cur->buf, offset);
  117. bufs->chunk_length = chunk_length;
  118. bufs->chunk_used = 1;
  119. bufs->max_chunk = max_chunk;
  120. bufs->chunk_keep = chunk_keep;
  121. return 0;
  122. }
  123. int nghttp2_bufs_realloc(nghttp2_bufs *bufs, size_t chunk_length) {
  124. int rv;
  125. nghttp2_buf_chain *chain;
  126. if (chunk_length < bufs->offset) {
  127. return NGHTTP2_ERR_INVALID_ARGUMENT;
  128. }
  129. rv = buf_chain_new(&chain, chunk_length, bufs->mem);
  130. if (rv != 0) {
  131. return rv;
  132. }
  133. nghttp2_bufs_free(bufs);
  134. bufs->head = chain;
  135. bufs->cur = bufs->head;
  136. nghttp2_buf_shift_right(&bufs->cur->buf, bufs->offset);
  137. bufs->chunk_length = chunk_length;
  138. bufs->chunk_used = 1;
  139. return 0;
  140. }
  141. void nghttp2_bufs_free(nghttp2_bufs *bufs) {
  142. nghttp2_buf_chain *chain, *next_chain;
  143. if (bufs == NULL) {
  144. return;
  145. }
  146. for (chain = bufs->head; chain;) {
  147. next_chain = chain->next;
  148. buf_chain_del(chain, bufs->mem);
  149. chain = next_chain;
  150. }
  151. bufs->head = NULL;
  152. }
  153. int nghttp2_bufs_wrap_init(nghttp2_bufs *bufs, uint8_t *begin, size_t len,
  154. nghttp2_mem *mem) {
  155. nghttp2_buf_chain *chain;
  156. chain = nghttp2_mem_malloc(mem, sizeof(nghttp2_buf_chain));
  157. if (chain == NULL) {
  158. return NGHTTP2_ERR_NOMEM;
  159. }
  160. chain->next = NULL;
  161. nghttp2_buf_wrap_init(&chain->buf, begin, len);
  162. bufs->mem = mem;
  163. bufs->offset = 0;
  164. bufs->head = chain;
  165. bufs->cur = bufs->head;
  166. bufs->chunk_length = len;
  167. bufs->chunk_used = 1;
  168. bufs->max_chunk = 1;
  169. bufs->chunk_keep = 1;
  170. return 0;
  171. }
  172. void nghttp2_bufs_wrap_free(nghttp2_bufs *bufs) {
  173. if (bufs == NULL) {
  174. return;
  175. }
  176. nghttp2_mem_free(bufs->mem, bufs->head);
  177. bufs->head = NULL;
  178. }
  179. void nghttp2_bufs_seek_last_present(nghttp2_bufs *bufs) {
  180. nghttp2_buf_chain *ci;
  181. for (ci = bufs->cur; ci; ci = ci->next) {
  182. if (nghttp2_buf_len(&ci->buf) == 0) {
  183. return;
  184. } else {
  185. bufs->cur = ci;
  186. }
  187. }
  188. }
  189. size_t nghttp2_bufs_len(nghttp2_bufs *bufs) {
  190. nghttp2_buf_chain *ci;
  191. size_t len;
  192. len = 0;
  193. for (ci = bufs->head; ci; ci = ci->next) {
  194. len += nghttp2_buf_len(&ci->buf);
  195. }
  196. return len;
  197. }
  198. static size_t bufs_avail(nghttp2_bufs *bufs) {
  199. return nghttp2_buf_avail(&bufs->cur->buf) +
  200. (bufs->chunk_length - bufs->offset) *
  201. (bufs->max_chunk - bufs->chunk_used);
  202. }
  203. static int bufs_alloc_chain(nghttp2_bufs *bufs) {
  204. int rv;
  205. nghttp2_buf_chain *chain;
  206. if (bufs->cur->next) {
  207. bufs->cur = bufs->cur->next;
  208. return 0;
  209. }
  210. if (bufs->max_chunk == bufs->chunk_used) {
  211. return NGHTTP2_ERR_BUFFER_ERROR;
  212. }
  213. rv = buf_chain_new(&chain, bufs->chunk_length, bufs->mem);
  214. if (rv != 0) {
  215. return rv;
  216. }
  217. DEBUGF(fprintf(stderr,
  218. "new buffer %zu bytes allocated for bufs %p, used %zu\n",
  219. bufs->chunk_length, bufs, bufs->chunk_used));
  220. ++bufs->chunk_used;
  221. bufs->cur->next = chain;
  222. bufs->cur = chain;
  223. nghttp2_buf_shift_right(&bufs->cur->buf, bufs->offset);
  224. return 0;
  225. }
  226. int nghttp2_bufs_add(nghttp2_bufs *bufs, const void *data, size_t len) {
  227. int rv;
  228. size_t nwrite;
  229. nghttp2_buf *buf;
  230. const uint8_t *p;
  231. if (bufs_avail(bufs) < len) {
  232. return NGHTTP2_ERR_BUFFER_ERROR;
  233. }
  234. p = data;
  235. while (len) {
  236. buf = &bufs->cur->buf;
  237. nwrite = nghttp2_min(nghttp2_buf_avail(buf), len);
  238. if (nwrite == 0) {
  239. rv = bufs_alloc_chain(bufs);
  240. if (rv != 0) {
  241. return rv;
  242. }
  243. continue;
  244. }
  245. buf->last = nghttp2_cpymem(buf->last, p, nwrite);
  246. p += nwrite;
  247. len -= nwrite;
  248. }
  249. return 0;
  250. }
  251. static int bufs_ensure_addb(nghttp2_bufs *bufs) {
  252. int rv;
  253. nghttp2_buf *buf;
  254. buf = &bufs->cur->buf;
  255. if (nghttp2_buf_avail(buf) > 0) {
  256. return 0;
  257. }
  258. rv = bufs_alloc_chain(bufs);
  259. if (rv != 0) {
  260. return rv;
  261. }
  262. return 0;
  263. }
  264. int nghttp2_bufs_addb(nghttp2_bufs *bufs, uint8_t b) {
  265. int rv;
  266. rv = bufs_ensure_addb(bufs);
  267. if (rv != 0) {
  268. return rv;
  269. }
  270. *bufs->cur->buf.last++ = b;
  271. return 0;
  272. }
  273. int nghttp2_bufs_addb_hold(nghttp2_bufs *bufs, uint8_t b) {
  274. int rv;
  275. rv = bufs_ensure_addb(bufs);
  276. if (rv != 0) {
  277. return rv;
  278. }
  279. *bufs->cur->buf.last = b;
  280. return 0;
  281. }
  282. int nghttp2_bufs_orb(nghttp2_bufs *bufs, uint8_t b) {
  283. int rv;
  284. rv = bufs_ensure_addb(bufs);
  285. if (rv != 0) {
  286. return rv;
  287. }
  288. *bufs->cur->buf.last++ |= b;
  289. return 0;
  290. }
  291. int nghttp2_bufs_orb_hold(nghttp2_bufs *bufs, uint8_t b) {
  292. int rv;
  293. rv = bufs_ensure_addb(bufs);
  294. if (rv != 0) {
  295. return rv;
  296. }
  297. *bufs->cur->buf.last |= b;
  298. return 0;
  299. }
  300. ssize_t nghttp2_bufs_remove(nghttp2_bufs *bufs, uint8_t **out) {
  301. size_t len;
  302. nghttp2_buf_chain *chain;
  303. nghttp2_buf *buf;
  304. uint8_t *res;
  305. nghttp2_buf resbuf;
  306. len = 0;
  307. for (chain = bufs->head; chain; chain = chain->next) {
  308. len += nghttp2_buf_len(&chain->buf);
  309. }
  310. if (len == 0) {
  311. res = NULL;
  312. return 0;
  313. }
  314. res = nghttp2_mem_malloc(bufs->mem, len);
  315. if (res == NULL) {
  316. return NGHTTP2_ERR_NOMEM;
  317. }
  318. nghttp2_buf_wrap_init(&resbuf, res, len);
  319. for (chain = bufs->head; chain; chain = chain->next) {
  320. buf = &chain->buf;
  321. resbuf.last = nghttp2_cpymem(resbuf.last, buf->pos, nghttp2_buf_len(buf));
  322. }
  323. *out = res;
  324. return (ssize_t)len;
  325. }
  326. size_t nghttp2_bufs_remove_copy(nghttp2_bufs *bufs, uint8_t *out) {
  327. size_t len;
  328. nghttp2_buf_chain *chain;
  329. nghttp2_buf *buf;
  330. nghttp2_buf resbuf;
  331. len = nghttp2_bufs_len(bufs);
  332. nghttp2_buf_wrap_init(&resbuf, out, len);
  333. for (chain = bufs->head; chain; chain = chain->next) {
  334. buf = &chain->buf;
  335. resbuf.last = nghttp2_cpymem(resbuf.last, buf->pos, nghttp2_buf_len(buf));
  336. }
  337. return len;
  338. }
  339. void nghttp2_bufs_reset(nghttp2_bufs *bufs) {
  340. nghttp2_buf_chain *chain, *ci;
  341. size_t k;
  342. k = bufs->chunk_keep;
  343. for (ci = bufs->head; ci; ci = ci->next) {
  344. nghttp2_buf_reset(&ci->buf);
  345. nghttp2_buf_shift_right(&ci->buf, bufs->offset);
  346. if (--k == 0) {
  347. break;
  348. }
  349. }
  350. if (ci) {
  351. chain = ci->next;
  352. ci->next = NULL;
  353. for (ci = chain; ci;) {
  354. chain = ci->next;
  355. buf_chain_del(ci, bufs->mem);
  356. ci = chain;
  357. }
  358. bufs->chunk_used = bufs->chunk_keep;
  359. }
  360. bufs->cur = bufs->head;
  361. }
  362. int nghttp2_bufs_advance(nghttp2_bufs *bufs) { return bufs_alloc_chain(bufs); }
  363. int nghttp2_bufs_next_present(nghttp2_bufs *bufs) {
  364. nghttp2_buf_chain *chain;
  365. chain = bufs->cur->next;
  366. return chain && nghttp2_buf_len(&chain->buf);
  367. }