sys_heap.h 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. /*
  2. * Copyright (c) 2019 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_SYS_SYS_HEAP_H_
  7. #define ZEPHYR_INCLUDE_SYS_SYS_HEAP_H_
  8. #include <stddef.h>
  9. #include <stdbool.h>
  10. #include <zephyr/types.h>
  11. /* Simple, fast heap implementation.
  12. *
  13. * A more or less conventional segregated fit allocator with
  14. * power-of-two buckets.
  15. *
  16. * Excellent space efficiency. Chunks can be split arbitrarily in 8
  17. * byte units. Overhead is only four bytes per allocated chunk (eight
  18. * bytes for heaps >256kb or on 64 bit systems), plus a log2-sized
  19. * array of 2-word bucket headers. No coarse alignment restrictions
  20. * on blocks, they can be split and merged (in units of 8 bytes)
  21. * arbitrarily.
  22. *
  23. * Simple API. Initialize at runtime with any blob of memory and not
  24. * a macro-generated, carefully aligned static array. Allocate and
  25. * free by user pointer and not an opaque block handle.
  26. *
  27. * Good fragmentation resistance. Freed blocks are always immediately
  28. * merged with adjacent free blocks. Allocations are attempted from a
  29. * sample of the smallest bucket that might fit, falling back rapidly
  30. * to the smallest block guaranteed to fit. Split memory remaining in
  31. * the chunk is always returned immediately to the heap for other
  32. * allocation.
  33. *
  34. * Excellent performance with firmly bounded runtime. All operations
  35. * are constant time (though there is a search of the smallest bucket
  36. * that has a compile-time-configurable upper bound, setting this to
  37. * extreme values results in an effectively linear search of the
  38. * list), objectively fast (~hundred instructions) and and amenable to
  39. * locked operation.
  40. */
  41. /* Note: the init_mem/bytes fields are for the static initializer to
  42. * have somewhere to put the arguments. The actual heap metadata at
  43. * runtime lives in the heap memory itself and this struct simply
  44. * functions as an opaque pointer. Would be good to clean this up and
  45. * put the two values somewhere else, though it would make
  46. * SYS_HEAP_DEFINE a little hairy to write.
  47. */
  48. struct sys_heap {
  49. struct z_heap *heap;
  50. void *init_mem;
  51. size_t init_bytes;
  52. };
  53. struct z_heap_stress_result {
  54. uint32_t total_allocs;
  55. uint32_t successful_allocs;
  56. uint32_t total_frees;
  57. uint64_t accumulated_in_use_bytes;
  58. };
  59. /** @brief Initialize sys_heap
  60. *
  61. * Initializes a sys_heap struct to manage the specified memory.
  62. *
  63. * @param heap Heap to initialize
  64. * @param mem Untyped pointer to unused memory
  65. * @param bytes Size of region pointed to by @a mem
  66. */
  67. void sys_heap_init(struct sys_heap *heap, void *mem, size_t bytes);
  68. /** @brief Allocate memory from a sys_heap
  69. *
  70. * Returns a pointer to a block of unused memory in the heap. This
  71. * memory will not otherwise be used until it is freed with
  72. * sys_heap_free(). If no memory can be allocated, NULL will be
  73. * returned. The allocated memory is guaranteed to have a starting
  74. * address which is a multiple of sizeof(void *). If a bigger alignment
  75. * is necessary then sys_heap_aligned_alloc() should be used instead.
  76. *
  77. * @note The sys_heap implementation is not internally synchronized.
  78. * No two sys_heap functions should operate on the same heap at the
  79. * same time. All locking must be provided by the user.
  80. *
  81. * @param heap Heap from which to allocate
  82. * @param bytes Number of bytes requested
  83. * @return Pointer to memory the caller can now use
  84. */
  85. void *sys_heap_alloc(struct sys_heap *heap, size_t bytes);
  86. /** @brief Allocate aligned memory from a sys_heap
  87. *
  88. * Behaves in all ways like sys_heap_alloc(), except that the returned
  89. * memory (if available) will have a starting address in memory which
  90. * is a multiple of the specified power-of-two alignment value in
  91. * bytes. With align=0 this behaves exactly like sys_heap_alloc().
  92. * The resulting memory can be returned to the heap using sys_heap_free().
  93. *
  94. * @param heap Heap from which to allocate
  95. * @param align Alignment in bytes, must be a power of two
  96. * @param bytes Number of bytes requested
  97. * @return Pointer to memory the caller can now use
  98. */
  99. void *sys_heap_aligned_alloc(struct sys_heap *heap, size_t align, size_t bytes);
  100. /** @brief Free memory into a sys_heap
  101. *
  102. * De-allocates a pointer to memory previously returned from
  103. * sys_heap_alloc such that it can be used for other purposes. The
  104. * caller must not use the memory region after entry to this function.
  105. *
  106. * @note The sys_heap implementation is not internally synchronized.
  107. * No two sys_heap functions should operate on the same heap at the
  108. * same time. All locking must be provided by the user.
  109. *
  110. * @param heap Heap to which to return the memory
  111. * @param mem A pointer previously returned from sys_heap_alloc()
  112. */
  113. void sys_heap_free(struct sys_heap *heap, void *mem);
  114. /** @brief Expand the size of an existing allocation
  115. *
  116. * Returns a pointer to a new memory region with the same contents,
  117. * but a different allocated size. If the new allocation can be
  118. * expanded in place, the pointer returned will be identical.
  119. * Otherwise the data will be copies to a new block and the old one
  120. * will be freed as per sys_heap_free(). If the specified size is
  121. * smaller than the original, the block will be truncated in place and
  122. * the remaining memory returned to the heap. If the allocation of a
  123. * new block fails, then NULL will be returned and the old block will
  124. * not be freed or modified.
  125. *
  126. * @note The return of a NULL on failure is a different behavior than
  127. * POSIX realloc(), which specifies that the original pointer will be
  128. * returned (i.e. it is not possible to safely detect realloc()
  129. * failure in POSIX, but it is here).
  130. *
  131. * @param heap Heap from which to allocate
  132. * @param ptr Original pointer returned from a previous allocation
  133. * @param align Alignment in bytes, must be a power of two
  134. * @param bytes Number of bytes requested for the new block
  135. * @return Pointer to memory the caller can now use, or NULL
  136. */
  137. void *sys_heap_aligned_realloc(struct sys_heap *heap, void *ptr,
  138. size_t align, size_t bytes);
  139. #define sys_heap_realloc(heap, ptr, bytes) \
  140. sys_heap_aligned_realloc(heap, ptr, 0, bytes)
  141. /** @brief Validate heap integrity
  142. *
  143. * Validates the internal integrity of a sys_heap. Intended for unit
  144. * test and validation code, though potentially useful as a user API
  145. * for applications with complicated runtime reliability requirements.
  146. * Note: this cannot catch every possible error, but if it returns
  147. * true then the heap is in a consistent state and can correctly
  148. * handle any sys_heap_alloc() request and free any live pointer
  149. * returned from a previou allocation.
  150. *
  151. * @param heap Heap to validate
  152. * @return true, if the heap is valid, otherwise false
  153. */
  154. bool sys_heap_validate(struct sys_heap *heap);
  155. /** @brief sys_heap stress test rig
  156. *
  157. * Test rig for heap allocation validation. This will loop for @a
  158. * op_count cycles, in each iteration making a random choice to
  159. * allocate or free a pointer of randomized (power law) size based on
  160. * heuristics designed to keep the heap in a state where it is near @a
  161. * target_percent full. Allocation and free operations are provided
  162. * by the caller as callbacks (i.e. this can in theory test any heap).
  163. * Results, including counts of frees and successful/unsuccessful
  164. * allocations, are returnewd via the @result struct.
  165. *
  166. * @param alloc_fn Callback to perform an allocation. Passes back the @a
  167. * arg parameter as a context handle.
  168. * @param free_fn Callback to perform a free of a pointer returned from
  169. * @a alloc. Passes back the @a arg parameter as a
  170. * context handle.
  171. * @param arg Context handle to pass back to the callbacks
  172. * @param total_bytes Size of the byte array the heap was initialized in
  173. * @param op_count How many iterations to test
  174. * @param scratch_mem A pointer to scratch memory to be used by the
  175. * test. Should be about 1/2 the size of the heap
  176. * for tests that need to stress fragmentation.
  177. * @param scratch_bytes Size of the memory pointed to by @a scratch_mem
  178. * @param target_percent Percentage fill value (1-100) to which the
  179. * random allocation choices will seek. High
  180. * values will result in significant allocation
  181. * failures and a very fragmented heap.
  182. * @param result Struct into which to store test results.
  183. */
  184. void sys_heap_stress(void *(*alloc_fn)(void *arg, size_t bytes),
  185. void (*free_fn)(void *arg, void *p),
  186. void *arg, size_t total_bytes,
  187. uint32_t op_count,
  188. void *scratch_mem, size_t scratch_bytes,
  189. int target_percent,
  190. struct z_heap_stress_result *result);
  191. /** @brief Print heap internal structure information to the console
  192. *
  193. * Print information on the heap structure such as its size, chunk buckets,
  194. * chunk list and some statistics for debugging purpose.
  195. *
  196. * @param heap Heap to print information about
  197. * @param dump_chunks True to print the entire heap chunk list
  198. */
  199. void sys_heap_print_info(struct sys_heap *heap, bool dump_chunks);
  200. /** @brief Dump heap internal structure information to the console
  201. *
  202. * Print information on the heap structure such as its size, chunk buckets,
  203. * chunk list and some statistics for debugging purpose.
  204. *
  205. * @param heap Heap to print information about
  206. */
  207. void sys_heap_dump(struct sys_heap *heap);
  208. #endif /* ZEPHYR_INCLUDE_SYS_SYS_HEAP_H_ */