memory_buffer_alloc.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751
  1. /*
  2. * Buffer-based memory allocator
  3. *
  4. * Copyright The Mbed TLS Contributors
  5. * SPDX-License-Identifier: Apache-2.0
  6. *
  7. * Licensed under the Apache License, Version 2.0 (the "License"); you may
  8. * not use this file except in compliance with the License.
  9. * You may obtain a copy of the License at
  10. *
  11. * http://www.apache.org/licenses/LICENSE-2.0
  12. *
  13. * Unless required by applicable law or agreed to in writing, software
  14. * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
  15. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  16. * See the License for the specific language governing permissions and
  17. * limitations under the License.
  18. */
  19. #include "common.h"
  20. #if defined(MBEDTLS_MEMORY_BUFFER_ALLOC_C)
  21. #include "mbedtls/memory_buffer_alloc.h"
  22. /* No need for the header guard as MBEDTLS_MEMORY_BUFFER_ALLOC_C
  23. is dependent upon MBEDTLS_PLATFORM_C */
  24. #include "mbedtls/platform.h"
  25. #include "mbedtls/platform_util.h"
  26. #include <string.h>
  27. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  28. #include <execinfo.h>
  29. #endif
  30. #if defined(MBEDTLS_THREADING_C)
  31. #include "mbedtls/threading.h"
  32. #endif
  33. #define MAGIC1 0xFF00AA55
  34. #define MAGIC2 0xEE119966
  35. #define MAX_BT 20
  36. typedef struct _memory_header memory_header;
  37. struct _memory_header {
  38. size_t magic1;
  39. size_t size;
  40. size_t alloc;
  41. memory_header *prev;
  42. memory_header *next;
  43. memory_header *prev_free;
  44. memory_header *next_free;
  45. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  46. char **trace;
  47. size_t trace_count;
  48. #endif
  49. size_t magic2;
  50. };
  51. typedef struct {
  52. unsigned char *buf;
  53. size_t len;
  54. memory_header *first;
  55. memory_header *first_free;
  56. int verify;
  57. #if defined(MBEDTLS_MEMORY_DEBUG)
  58. size_t alloc_count;
  59. size_t free_count;
  60. size_t total_used;
  61. size_t maximum_used;
  62. size_t header_count;
  63. size_t maximum_header_count;
  64. #endif
  65. #if defined(MBEDTLS_THREADING_C)
  66. mbedtls_threading_mutex_t mutex;
  67. #endif
  68. }
  69. buffer_alloc_ctx;
  70. static buffer_alloc_ctx heap;
  71. #if defined(MBEDTLS_MEMORY_DEBUG)
  72. static void debug_header(memory_header *hdr)
  73. {
  74. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  75. size_t i;
  76. #endif
  77. mbedtls_fprintf(stderr, "HDR: PTR(%10zu), PREV(%10zu), NEXT(%10zu), "
  78. "ALLOC(%zu), SIZE(%10zu)\n",
  79. (size_t) hdr, (size_t) hdr->prev, (size_t) hdr->next,
  80. hdr->alloc, hdr->size);
  81. mbedtls_fprintf(stderr, " FPREV(%10zu), FNEXT(%10zu)\n",
  82. (size_t) hdr->prev_free, (size_t) hdr->next_free);
  83. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  84. mbedtls_fprintf(stderr, "TRACE: \n");
  85. for (i = 0; i < hdr->trace_count; i++) {
  86. mbedtls_fprintf(stderr, "%s\n", hdr->trace[i]);
  87. }
  88. mbedtls_fprintf(stderr, "\n");
  89. #endif
  90. }
  91. static void debug_chain(void)
  92. {
  93. memory_header *cur = heap.first;
  94. mbedtls_fprintf(stderr, "\nBlock list\n");
  95. while (cur != NULL) {
  96. debug_header(cur);
  97. cur = cur->next;
  98. }
  99. mbedtls_fprintf(stderr, "Free list\n");
  100. cur = heap.first_free;
  101. while (cur != NULL) {
  102. debug_header(cur);
  103. cur = cur->next_free;
  104. }
  105. }
  106. #endif /* MBEDTLS_MEMORY_DEBUG */
  107. static int verify_header(memory_header *hdr)
  108. {
  109. if (hdr->magic1 != MAGIC1) {
  110. #if defined(MBEDTLS_MEMORY_DEBUG)
  111. mbedtls_fprintf(stderr, "FATAL: MAGIC1 mismatch\n");
  112. #endif
  113. return 1;
  114. }
  115. if (hdr->magic2 != MAGIC2) {
  116. #if defined(MBEDTLS_MEMORY_DEBUG)
  117. mbedtls_fprintf(stderr, "FATAL: MAGIC2 mismatch\n");
  118. #endif
  119. return 1;
  120. }
  121. if (hdr->alloc > 1) {
  122. #if defined(MBEDTLS_MEMORY_DEBUG)
  123. mbedtls_fprintf(stderr, "FATAL: alloc has illegal value\n");
  124. #endif
  125. return 1;
  126. }
  127. if (hdr->prev != NULL && hdr->prev == hdr->next) {
  128. #if defined(MBEDTLS_MEMORY_DEBUG)
  129. mbedtls_fprintf(stderr, "FATAL: prev == next\n");
  130. #endif
  131. return 1;
  132. }
  133. if (hdr->prev_free != NULL && hdr->prev_free == hdr->next_free) {
  134. #if defined(MBEDTLS_MEMORY_DEBUG)
  135. mbedtls_fprintf(stderr, "FATAL: prev_free == next_free\n");
  136. #endif
  137. return 1;
  138. }
  139. return 0;
  140. }
  141. static int verify_chain(void)
  142. {
  143. memory_header *prv = heap.first, *cur;
  144. if (prv == NULL || verify_header(prv) != 0) {
  145. #if defined(MBEDTLS_MEMORY_DEBUG)
  146. mbedtls_fprintf(stderr, "FATAL: verification of first header "
  147. "failed\n");
  148. #endif
  149. return 1;
  150. }
  151. if (heap.first->prev != NULL) {
  152. #if defined(MBEDTLS_MEMORY_DEBUG)
  153. mbedtls_fprintf(stderr, "FATAL: verification failed: "
  154. "first->prev != NULL\n");
  155. #endif
  156. return 1;
  157. }
  158. cur = heap.first->next;
  159. while (cur != NULL) {
  160. if (verify_header(cur) != 0) {
  161. #if defined(MBEDTLS_MEMORY_DEBUG)
  162. mbedtls_fprintf(stderr, "FATAL: verification of header "
  163. "failed\n");
  164. #endif
  165. return 1;
  166. }
  167. if (cur->prev != prv) {
  168. #if defined(MBEDTLS_MEMORY_DEBUG)
  169. mbedtls_fprintf(stderr, "FATAL: verification failed: "
  170. "cur->prev != prv\n");
  171. #endif
  172. return 1;
  173. }
  174. prv = cur;
  175. cur = cur->next;
  176. }
  177. return 0;
  178. }
  179. static void *buffer_alloc_calloc(size_t n, size_t size)
  180. {
  181. memory_header *new, *cur = heap.first_free;
  182. unsigned char *p;
  183. void *ret;
  184. size_t original_len, len;
  185. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  186. void *trace_buffer[MAX_BT];
  187. size_t trace_cnt;
  188. #endif
  189. if (heap.buf == NULL || heap.first == NULL) {
  190. return NULL;
  191. }
  192. original_len = len = n * size;
  193. if (n == 0 || size == 0 || len / n != size) {
  194. return NULL;
  195. } else if (len > (size_t) -MBEDTLS_MEMORY_ALIGN_MULTIPLE) {
  196. return NULL;
  197. }
  198. if (len % MBEDTLS_MEMORY_ALIGN_MULTIPLE) {
  199. len -= len % MBEDTLS_MEMORY_ALIGN_MULTIPLE;
  200. len += MBEDTLS_MEMORY_ALIGN_MULTIPLE;
  201. }
  202. // Find block that fits
  203. //
  204. while (cur != NULL) {
  205. if (cur->size >= len) {
  206. break;
  207. }
  208. cur = cur->next_free;
  209. }
  210. if (cur == NULL) {
  211. return NULL;
  212. }
  213. if (cur->alloc != 0) {
  214. #if defined(MBEDTLS_MEMORY_DEBUG)
  215. mbedtls_fprintf(stderr, "FATAL: block in free_list but allocated "
  216. "data\n");
  217. #endif
  218. mbedtls_exit(1);
  219. }
  220. #if defined(MBEDTLS_MEMORY_DEBUG)
  221. heap.alloc_count++;
  222. #endif
  223. // Found location, split block if > memory_header + 4 room left
  224. //
  225. if (cur->size - len < sizeof(memory_header) +
  226. MBEDTLS_MEMORY_ALIGN_MULTIPLE) {
  227. cur->alloc = 1;
  228. // Remove from free_list
  229. //
  230. if (cur->prev_free != NULL) {
  231. cur->prev_free->next_free = cur->next_free;
  232. } else {
  233. heap.first_free = cur->next_free;
  234. }
  235. if (cur->next_free != NULL) {
  236. cur->next_free->prev_free = cur->prev_free;
  237. }
  238. cur->prev_free = NULL;
  239. cur->next_free = NULL;
  240. #if defined(MBEDTLS_MEMORY_DEBUG)
  241. heap.total_used += cur->size;
  242. if (heap.total_used > heap.maximum_used) {
  243. heap.maximum_used = heap.total_used;
  244. }
  245. #endif
  246. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  247. trace_cnt = backtrace(trace_buffer, MAX_BT);
  248. cur->trace = backtrace_symbols(trace_buffer, trace_cnt);
  249. cur->trace_count = trace_cnt;
  250. #endif
  251. if ((heap.verify & MBEDTLS_MEMORY_VERIFY_ALLOC) && verify_chain() != 0) {
  252. mbedtls_exit(1);
  253. }
  254. ret = (unsigned char *) cur + sizeof(memory_header);
  255. memset(ret, 0, original_len);
  256. return ret;
  257. }
  258. p = ((unsigned char *) cur) + sizeof(memory_header) + len;
  259. new = (memory_header *) p;
  260. new->size = cur->size - len - sizeof(memory_header);
  261. new->alloc = 0;
  262. new->prev = cur;
  263. new->next = cur->next;
  264. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  265. new->trace = NULL;
  266. new->trace_count = 0;
  267. #endif
  268. new->magic1 = MAGIC1;
  269. new->magic2 = MAGIC2;
  270. if (new->next != NULL) {
  271. new->next->prev = new;
  272. }
  273. // Replace cur with new in free_list
  274. //
  275. new->prev_free = cur->prev_free;
  276. new->next_free = cur->next_free;
  277. if (new->prev_free != NULL) {
  278. new->prev_free->next_free = new;
  279. } else {
  280. heap.first_free = new;
  281. }
  282. if (new->next_free != NULL) {
  283. new->next_free->prev_free = new;
  284. }
  285. cur->alloc = 1;
  286. cur->size = len;
  287. cur->next = new;
  288. cur->prev_free = NULL;
  289. cur->next_free = NULL;
  290. #if defined(MBEDTLS_MEMORY_DEBUG)
  291. heap.header_count++;
  292. if (heap.header_count > heap.maximum_header_count) {
  293. heap.maximum_header_count = heap.header_count;
  294. }
  295. heap.total_used += cur->size;
  296. if (heap.total_used > heap.maximum_used) {
  297. heap.maximum_used = heap.total_used;
  298. }
  299. #endif
  300. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  301. trace_cnt = backtrace(trace_buffer, MAX_BT);
  302. cur->trace = backtrace_symbols(trace_buffer, trace_cnt);
  303. cur->trace_count = trace_cnt;
  304. #endif
  305. if ((heap.verify & MBEDTLS_MEMORY_VERIFY_ALLOC) && verify_chain() != 0) {
  306. mbedtls_exit(1);
  307. }
  308. ret = (unsigned char *) cur + sizeof(memory_header);
  309. memset(ret, 0, original_len);
  310. return ret;
  311. }
  312. static void buffer_alloc_free(void *ptr)
  313. {
  314. memory_header *hdr, *old = NULL;
  315. unsigned char *p = (unsigned char *) ptr;
  316. if (ptr == NULL || heap.buf == NULL || heap.first == NULL) {
  317. return;
  318. }
  319. if (p < heap.buf || p >= heap.buf + heap.len) {
  320. #if defined(MBEDTLS_MEMORY_DEBUG)
  321. mbedtls_fprintf(stderr, "FATAL: mbedtls_free() outside of managed "
  322. "space\n");
  323. #endif
  324. mbedtls_exit(1);
  325. }
  326. p -= sizeof(memory_header);
  327. hdr = (memory_header *) p;
  328. if (verify_header(hdr) != 0) {
  329. mbedtls_exit(1);
  330. }
  331. if (hdr->alloc != 1) {
  332. #if defined(MBEDTLS_MEMORY_DEBUG)
  333. mbedtls_fprintf(stderr, "FATAL: mbedtls_free() on unallocated "
  334. "data\n");
  335. #endif
  336. mbedtls_exit(1);
  337. }
  338. hdr->alloc = 0;
  339. #if defined(MBEDTLS_MEMORY_DEBUG)
  340. heap.free_count++;
  341. heap.total_used -= hdr->size;
  342. #endif
  343. #if defined(MBEDTLS_MEMORY_BACKTRACE)
  344. free(hdr->trace);
  345. hdr->trace = NULL;
  346. hdr->trace_count = 0;
  347. #endif
  348. // Regroup with block before
  349. //
  350. if (hdr->prev != NULL && hdr->prev->alloc == 0) {
  351. #if defined(MBEDTLS_MEMORY_DEBUG)
  352. heap.header_count--;
  353. #endif
  354. hdr->prev->size += sizeof(memory_header) + hdr->size;
  355. hdr->prev->next = hdr->next;
  356. old = hdr;
  357. hdr = hdr->prev;
  358. if (hdr->next != NULL) {
  359. hdr->next->prev = hdr;
  360. }
  361. memset(old, 0, sizeof(memory_header));
  362. }
  363. // Regroup with block after
  364. //
  365. if (hdr->next != NULL && hdr->next->alloc == 0) {
  366. #if defined(MBEDTLS_MEMORY_DEBUG)
  367. heap.header_count--;
  368. #endif
  369. hdr->size += sizeof(memory_header) + hdr->next->size;
  370. old = hdr->next;
  371. hdr->next = hdr->next->next;
  372. if (hdr->prev_free != NULL || hdr->next_free != NULL) {
  373. if (hdr->prev_free != NULL) {
  374. hdr->prev_free->next_free = hdr->next_free;
  375. } else {
  376. heap.first_free = hdr->next_free;
  377. }
  378. if (hdr->next_free != NULL) {
  379. hdr->next_free->prev_free = hdr->prev_free;
  380. }
  381. }
  382. hdr->prev_free = old->prev_free;
  383. hdr->next_free = old->next_free;
  384. if (hdr->prev_free != NULL) {
  385. hdr->prev_free->next_free = hdr;
  386. } else {
  387. heap.first_free = hdr;
  388. }
  389. if (hdr->next_free != NULL) {
  390. hdr->next_free->prev_free = hdr;
  391. }
  392. if (hdr->next != NULL) {
  393. hdr->next->prev = hdr;
  394. }
  395. memset(old, 0, sizeof(memory_header));
  396. }
  397. // Prepend to free_list if we have not merged
  398. // (Does not have to stay in same order as prev / next list)
  399. //
  400. if (old == NULL) {
  401. hdr->next_free = heap.first_free;
  402. if (heap.first_free != NULL) {
  403. heap.first_free->prev_free = hdr;
  404. }
  405. heap.first_free = hdr;
  406. }
  407. if ((heap.verify & MBEDTLS_MEMORY_VERIFY_FREE) && verify_chain() != 0) {
  408. mbedtls_exit(1);
  409. }
  410. }
  411. void mbedtls_memory_buffer_set_verify(int verify)
  412. {
  413. heap.verify = verify;
  414. }
  415. int mbedtls_memory_buffer_alloc_verify(void)
  416. {
  417. return verify_chain();
  418. }
  419. #if defined(MBEDTLS_MEMORY_DEBUG)
  420. void mbedtls_memory_buffer_alloc_status(void)
  421. {
  422. mbedtls_fprintf(stderr,
  423. "Current use: %zu blocks / %zu bytes, max: %zu blocks / "
  424. "%zu bytes (total %zu bytes), alloc / free: %zu / %zu\n",
  425. heap.header_count, heap.total_used,
  426. heap.maximum_header_count, heap.maximum_used,
  427. heap.maximum_header_count * sizeof(memory_header)
  428. + heap.maximum_used,
  429. heap.alloc_count, heap.free_count);
  430. if (heap.first->next == NULL) {
  431. mbedtls_fprintf(stderr, "All memory de-allocated in stack buffer\n");
  432. } else {
  433. mbedtls_fprintf(stderr, "Memory currently allocated:\n");
  434. debug_chain();
  435. }
  436. }
  437. void mbedtls_memory_buffer_alloc_max_get(size_t *max_used, size_t *max_blocks)
  438. {
  439. *max_used = heap.maximum_used;
  440. *max_blocks = heap.maximum_header_count;
  441. }
  442. void mbedtls_memory_buffer_alloc_max_reset(void)
  443. {
  444. heap.maximum_used = 0;
  445. heap.maximum_header_count = 0;
  446. }
  447. void mbedtls_memory_buffer_alloc_cur_get(size_t *cur_used, size_t *cur_blocks)
  448. {
  449. *cur_used = heap.total_used;
  450. *cur_blocks = heap.header_count;
  451. }
  452. #endif /* MBEDTLS_MEMORY_DEBUG */
  453. #if defined(MBEDTLS_THREADING_C)
  454. static void *buffer_alloc_calloc_mutexed(size_t n, size_t size)
  455. {
  456. void *buf;
  457. if (mbedtls_mutex_lock(&heap.mutex) != 0) {
  458. return NULL;
  459. }
  460. buf = buffer_alloc_calloc(n, size);
  461. if (mbedtls_mutex_unlock(&heap.mutex)) {
  462. return NULL;
  463. }
  464. return buf;
  465. }
  466. static void buffer_alloc_free_mutexed(void *ptr)
  467. {
  468. /* We have no good option here, but corrupting the heap seems
  469. * worse than losing memory. */
  470. if (mbedtls_mutex_lock(&heap.mutex)) {
  471. return;
  472. }
  473. buffer_alloc_free(ptr);
  474. (void) mbedtls_mutex_unlock(&heap.mutex);
  475. }
  476. #endif /* MBEDTLS_THREADING_C */
  477. void mbedtls_memory_buffer_alloc_init(unsigned char *buf, size_t len)
  478. {
  479. memset(&heap, 0, sizeof(buffer_alloc_ctx));
  480. #if defined(MBEDTLS_THREADING_C)
  481. mbedtls_mutex_init(&heap.mutex);
  482. mbedtls_platform_set_calloc_free(buffer_alloc_calloc_mutexed,
  483. buffer_alloc_free_mutexed);
  484. #else
  485. mbedtls_platform_set_calloc_free(buffer_alloc_calloc, buffer_alloc_free);
  486. #endif
  487. if (len < sizeof(memory_header) + MBEDTLS_MEMORY_ALIGN_MULTIPLE) {
  488. return;
  489. } else if ((size_t) buf % MBEDTLS_MEMORY_ALIGN_MULTIPLE) {
  490. /* Adjust len first since buf is used in the computation */
  491. len -= MBEDTLS_MEMORY_ALIGN_MULTIPLE
  492. - (size_t) buf % MBEDTLS_MEMORY_ALIGN_MULTIPLE;
  493. buf += MBEDTLS_MEMORY_ALIGN_MULTIPLE
  494. - (size_t) buf % MBEDTLS_MEMORY_ALIGN_MULTIPLE;
  495. }
  496. memset(buf, 0, len);
  497. heap.buf = buf;
  498. heap.len = len;
  499. heap.first = (memory_header *) buf;
  500. heap.first->size = len - sizeof(memory_header);
  501. heap.first->magic1 = MAGIC1;
  502. heap.first->magic2 = MAGIC2;
  503. heap.first_free = heap.first;
  504. }
  505. void mbedtls_memory_buffer_alloc_free(void)
  506. {
  507. #if defined(MBEDTLS_THREADING_C)
  508. mbedtls_mutex_free(&heap.mutex);
  509. #endif
  510. mbedtls_platform_zeroize(&heap, sizeof(buffer_alloc_ctx));
  511. }
  512. #if defined(MBEDTLS_SELF_TEST)
  513. static int check_pointer(void *p)
  514. {
  515. if (p == NULL) {
  516. return -1;
  517. }
  518. if ((size_t) p % MBEDTLS_MEMORY_ALIGN_MULTIPLE != 0) {
  519. return -1;
  520. }
  521. return 0;
  522. }
  523. static int check_all_free(void)
  524. {
  525. if (
  526. #if defined(MBEDTLS_MEMORY_DEBUG)
  527. heap.total_used != 0 ||
  528. #endif
  529. heap.first != heap.first_free ||
  530. (void *) heap.first != (void *) heap.buf) {
  531. return -1;
  532. }
  533. return 0;
  534. }
  535. #define TEST_ASSERT(condition) \
  536. if (!(condition)) \
  537. { \
  538. if (verbose != 0) \
  539. mbedtls_printf("failed\n"); \
  540. \
  541. ret = 1; \
  542. goto cleanup; \
  543. }
  544. int mbedtls_memory_buffer_alloc_self_test(int verbose)
  545. {
  546. unsigned char buf[1024];
  547. unsigned char *p, *q, *r, *end;
  548. int ret = 0;
  549. if (verbose != 0) {
  550. mbedtls_printf(" MBA test #1 (basic alloc-free cycle): ");
  551. }
  552. mbedtls_memory_buffer_alloc_init(buf, sizeof(buf));
  553. p = mbedtls_calloc(1, 1);
  554. q = mbedtls_calloc(1, 128);
  555. r = mbedtls_calloc(1, 16);
  556. TEST_ASSERT(check_pointer(p) == 0 &&
  557. check_pointer(q) == 0 &&
  558. check_pointer(r) == 0);
  559. mbedtls_free(r);
  560. mbedtls_free(q);
  561. mbedtls_free(p);
  562. TEST_ASSERT(check_all_free() == 0);
  563. /* Memorize end to compare with the next test */
  564. end = heap.buf + heap.len;
  565. mbedtls_memory_buffer_alloc_free();
  566. if (verbose != 0) {
  567. mbedtls_printf("passed\n");
  568. }
  569. if (verbose != 0) {
  570. mbedtls_printf(" MBA test #2 (buf not aligned): ");
  571. }
  572. mbedtls_memory_buffer_alloc_init(buf + 1, sizeof(buf) - 1);
  573. TEST_ASSERT(heap.buf + heap.len == end);
  574. p = mbedtls_calloc(1, 1);
  575. q = mbedtls_calloc(1, 128);
  576. r = mbedtls_calloc(1, 16);
  577. TEST_ASSERT(check_pointer(p) == 0 &&
  578. check_pointer(q) == 0 &&
  579. check_pointer(r) == 0);
  580. mbedtls_free(r);
  581. mbedtls_free(q);
  582. mbedtls_free(p);
  583. TEST_ASSERT(check_all_free() == 0);
  584. mbedtls_memory_buffer_alloc_free();
  585. if (verbose != 0) {
  586. mbedtls_printf("passed\n");
  587. }
  588. if (verbose != 0) {
  589. mbedtls_printf(" MBA test #3 (full): ");
  590. }
  591. mbedtls_memory_buffer_alloc_init(buf, sizeof(buf));
  592. p = mbedtls_calloc(1, sizeof(buf) - sizeof(memory_header));
  593. TEST_ASSERT(check_pointer(p) == 0);
  594. TEST_ASSERT(mbedtls_calloc(1, 1) == NULL);
  595. mbedtls_free(p);
  596. p = mbedtls_calloc(1, sizeof(buf) - 2 * sizeof(memory_header) - 16);
  597. q = mbedtls_calloc(1, 16);
  598. TEST_ASSERT(check_pointer(p) == 0 && check_pointer(q) == 0);
  599. TEST_ASSERT(mbedtls_calloc(1, 1) == NULL);
  600. mbedtls_free(q);
  601. TEST_ASSERT(mbedtls_calloc(1, 17) == NULL);
  602. mbedtls_free(p);
  603. TEST_ASSERT(check_all_free() == 0);
  604. mbedtls_memory_buffer_alloc_free();
  605. if (verbose != 0) {
  606. mbedtls_printf("passed\n");
  607. }
  608. cleanup:
  609. mbedtls_memory_buffer_alloc_free();
  610. return ret;
  611. }
  612. #endif /* MBEDTLS_SELF_TEST */
  613. #endif /* MBEDTLS_MEMORY_BUFFER_ALLOC_C */