memory_buffer_alloc.c 20 KB

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