mempool.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. /*
  2. * Copyright (c) 2017 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <string.h>
  8. #include <sys/math_extras.h>
  9. #include <sys/util.h>
  10. static void *z_heap_aligned_alloc(struct k_heap *heap, size_t align, size_t size)
  11. {
  12. void *mem;
  13. struct k_heap **heap_ref;
  14. size_t __align;
  15. /*
  16. * Adjust the size to make room for our heap reference.
  17. * Merge a rewind bit with align value (see sys_heap_aligned_alloc()).
  18. * This allows for storing the heap pointer right below the aligned
  19. * boundary without wasting any memory.
  20. */
  21. if (size_add_overflow(size, sizeof(heap_ref), &size)) {
  22. return NULL;
  23. }
  24. __align = align | sizeof(heap_ref);
  25. mem = k_heap_aligned_alloc(heap, __align, size, K_NO_WAIT);
  26. if (mem == NULL) {
  27. return NULL;
  28. }
  29. heap_ref = mem;
  30. *heap_ref = heap;
  31. mem = ++heap_ref;
  32. __ASSERT(align == 0 || ((uintptr_t)mem & (align - 1)) == 0,
  33. "misaligned memory at %p (align = %zu)", mem, align);
  34. return mem;
  35. }
  36. void k_free(void *ptr)
  37. {
  38. struct k_heap **heap_ref;
  39. if (ptr != NULL) {
  40. heap_ref = ptr;
  41. ptr = --heap_ref;
  42. SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_free, *heap_ref);
  43. k_heap_free(*heap_ref, ptr);
  44. SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_free, *heap_ref);
  45. }
  46. }
  47. #if (CONFIG_HEAP_MEM_POOL_SIZE > 0)
  48. K_HEAP_DEFINE(_system_heap, CONFIG_HEAP_MEM_POOL_SIZE);
  49. #define _SYSTEM_HEAP (&_system_heap)
  50. void *k_aligned_alloc(size_t align, size_t size)
  51. {
  52. __ASSERT(align / sizeof(void *) >= 1
  53. && (align % sizeof(void *)) == 0,
  54. "align must be a multiple of sizeof(void *)");
  55. __ASSERT((align & (align - 1)) == 0,
  56. "align must be a power of 2");
  57. SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_aligned_alloc, _SYSTEM_HEAP);
  58. void *ret = z_heap_aligned_alloc(_SYSTEM_HEAP, align, size);
  59. SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_aligned_alloc, _SYSTEM_HEAP, ret);
  60. return ret;
  61. }
  62. void *k_malloc(size_t size)
  63. {
  64. SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_malloc, _SYSTEM_HEAP);
  65. void *ret = k_aligned_alloc(sizeof(void *), size);
  66. SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_malloc, _SYSTEM_HEAP, ret);
  67. return ret;
  68. }
  69. void *k_calloc(size_t nmemb, size_t size)
  70. {
  71. void *ret;
  72. size_t bounds;
  73. SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_calloc, _SYSTEM_HEAP);
  74. if (size_mul_overflow(nmemb, size, &bounds)) {
  75. SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_calloc, _SYSTEM_HEAP, NULL);
  76. return NULL;
  77. }
  78. ret = k_malloc(bounds);
  79. if (ret != NULL) {
  80. (void)memset(ret, 0, bounds);
  81. }
  82. SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_calloc, _SYSTEM_HEAP, ret);
  83. return ret;
  84. }
  85. void k_thread_system_pool_assign(struct k_thread *thread)
  86. {
  87. thread->resource_pool = _SYSTEM_HEAP;
  88. }
  89. #else
  90. #define _SYSTEM_HEAP NULL
  91. #endif
  92. void *z_thread_aligned_alloc(size_t align, size_t size)
  93. {
  94. void *ret;
  95. struct k_heap *heap;
  96. if (k_is_in_isr()) {
  97. heap = _SYSTEM_HEAP;
  98. } else {
  99. heap = _current->resource_pool;
  100. }
  101. if (heap != NULL) {
  102. ret = z_heap_aligned_alloc(heap, align, size);
  103. } else {
  104. ret = NULL;
  105. }
  106. return ret;
  107. }