1 /* Copyright (c) 2021 Intel Corporation
2  * SPDX-License-Identifier: Apache-2.0
3  */
4 
5 #ifndef ZEPHYR_INCLUDE_SYS_MULTI_HEAP_H_
6 #define ZEPHYR_INCLUDE_SYS_MULTI_HEAP_H_
7 
8 #include <zephyr/types.h>
9 
10 #define MAX_MULTI_HEAPS 8
11 
12 /**
13  * @defgroup multi_heap_wrapper Multi-Heap Wrapper
14  * @ingroup heaps
15  * @{
16  */
17 
18 /**
19  * @brief Multi-heap allocator
20  *
21  * A sys_multi_heap represents a single allocator made from multiple,
22  * separately managed pools of memory that must be accessed via a
23  * unified API.  They can be discontiguous, and in many cases will be
24  * expected to have different capabilities (for example: latency,
25  * cacheability, cpu affinity, etc...)
26  *
27  * Allocation from the multiheap provides an opaque "configuration"
28  * value to specify requirements and heuristics to assist the choice
29  * in backend, which is then provided to a user-specified "choice"
30  * function whose job it is to select a heap based on information in
31  * the config specifier and runtime state (heap full state, etc...)
32  */
33 struct sys_multi_heap;
34 
35 /**
36  * @brief Multi-heap choice function
37  *
38  * This is a user-provided functions whose responsibility is selecting
39  * a specific sys_heap backend based on the opaque cfg value, which is
40  * specified by the user as an argument to sys_multi_heap_alloc(), and
41  * performing the allocation on behalf of the caller.  The callback is
42  * free to choose any registered heap backend to perform the
43  * allocation, and may choose to pad the user-provided values as
44  * needed, and to use an aligned allocation where required by the
45  * specified configuration.
46  *
47  * NULL may be returned, which will cause the
48  * allocation to fail and a NULL reported to the calling code.
49  *
50  * @param mheap Multi-heap structure.
51  * @param cfg An opaque user-provided value.  It may be interpreted in
52  *            any way by the application
53  * @param align Alignment of requested memory (or zero for no alignment)
54  * @param size The user-specified allocation size in bytes
55  * @return A pointer to the allocated memory
56  */
57 typedef void *(*sys_multi_heap_fn_t)(struct sys_multi_heap *mheap, void *cfg,
58 				     size_t align, size_t size);
59 
60 
61 struct sys_multi_heap_rec {
62 	struct sys_heap *heap;
63 	void *user_data;
64 };
65 
66 struct sys_multi_heap {
67 	unsigned int nheaps;
68 	sys_multi_heap_fn_t choice;
69 	struct sys_multi_heap_rec heaps[MAX_MULTI_HEAPS];
70 };
71 
72 /**
73  * @brief Initialize multi-heap
74  *
75  * Initialize a sys_multi_heap struct with the specified choice
76  * function.  Note that individual heaps must be added later with
77  * sys_multi_heap_add_heap so that the heap bounds can be tracked by
78  * the multi heap code.
79  *
80  * @note In general a multiheap is likely to be instantiated
81  * semi-statically from system configuration (for example, via
82  * linker-provided bounds on available memory in different regions, or
83  * from devicetree definitions of hardware-provided addressable
84  * memory, etc...).  The general expectation is that a soc- or
85  * board-level platform device will be initialized at system boot from
86  * these upstream configuration sources and not that an application
87  * will assemble a multi-heap on its own.
88  *
89  * @param heap A sys_multi_heap to initialize
90  * @param choice_fn A sys_multi_heap_fn_t callback used to select
91  *                  heaps at allocation time
92  */
93 void sys_multi_heap_init(struct sys_multi_heap *heap,
94 			 sys_multi_heap_fn_t choice_fn);
95 
96 /**
97  * @brief Add sys_heap to multi heap
98  *
99  * This adds a known sys_heap backend to an existing multi heap,
100  * allowing the multi heap internals to track the bounds of the heap
101  * and determine which heap (if any) from which a freed block was
102  * allocated.
103  *
104  * @param mheap A sys_multi_heap to which to add a heap
105  * @param heap The heap to add
106  * @param user_data pointer to any data for the heap
107  */
108 void sys_multi_heap_add_heap(struct sys_multi_heap *mheap, struct sys_heap *heap, void *user_data);
109 
110 /**
111  * @brief Allocate memory from multi heap
112  *
113  * Just as for sys_heap_alloc(), allocates a block of memory of the
114  * specified size in bytes.  Takes an opaque configuration pointer
115  * passed to the multi heap choice function, which is used by
116  * integration code to choose a heap backend.
117  *
118  * @param mheap Multi heap pointer
119  * @param cfg Opaque configuration parameter, as for sys_multi_heap_fn_t
120  * @param bytes Requested size of the allocation, in bytes
121  * @return A valid pointer to heap memory, or NULL if no memory is available
122  */
123 void *sys_multi_heap_alloc(struct sys_multi_heap *mheap, void *cfg, size_t bytes);
124 
125 /**
126  * @brief Allocate aligned memory from multi heap
127  *
128  * Just as for sys_multi_heap_alloc(), allocates a block of memory of
129  * the specified size in bytes.  Takes an additional parameter
130  * specifying a power of two alignment, in bytes.
131  *
132  * @param mheap Multi heap pointer
133  * @param cfg Opaque configuration parameter, as for sys_multi_heap_fn_t
134  * @param align Power of two alignment for the returned pointer, in bytes
135  * @param bytes Requested size of the allocation, in bytes
136  * @return A valid pointer to heap memory, or NULL if no memory is available
137  */
138 void *sys_multi_heap_aligned_alloc(struct sys_multi_heap *mheap,
139 				   void *cfg, size_t align, size_t bytes);
140 
141 /**
142  * @brief Get a specific heap for provided address
143  *
144  * Finds a single system heap (with user_data)
145  * controlling the provided pointer
146  *
147  * @param mheap Multi heap pointer
148  * @param addr address to be found, must be a pointer to a block allocated by sys_multi_heap_alloc
149  * @return 0 multi_heap_rec pointer to a structure to be filled with return data
150  *			 or NULL if the heap has not been found
151  */
152 const struct sys_multi_heap_rec *sys_multi_heap_get_heap(const struct sys_multi_heap *mheap,
153 							 void *addr);
154 
155 /**
156  * @brief Free memory allocated from multi heap
157  *
158  * Returns the specified block, which must be the return value of a
159  * previously successful sys_multi_heap_alloc() or
160  * sys_multi_heap_aligned_alloc() call, to the heap backend from which
161  * it was allocated.
162  *
163  * Accepts NULL as a block parameter, which is specified to have no
164  * effect.
165  *
166  * @param mheap Multi heap pointer
167  * @param block Block to free, must be a pointer to a block allocated by sys_multi_heap_alloc
168  */
169 void sys_multi_heap_free(struct sys_multi_heap *mheap, void *block);
170 
171 /** @brief Expand the size of an existing allocation on the multi heap
172  *
173  * Returns a pointer to a new memory region with the same contents,
174  * but a different allocated size.  If the new allocation can be
175  * expanded in place, the pointer returned will be identical.
176  * Otherwise the data will be copies to a new block and the old one
177  * will be freed as per sys_heap_free().  If the specified size is
178  * smaller than the original, the block will be truncated in place and
179  * the remaining memory returned to the heap.  If the allocation of a
180  * new block fails, then NULL will be returned and the old block will
181  * not be freed or modified. If a new allocation is needed, the choice
182  * for the heap used will be bases on the cfg parameter (same as in sys_multi_heap_aligned_alloc).
183  *
184  * @param mheap Multi heap pointer
185  * @param cfg Opaque configuration parameter, as for sys_multi_heap_fn_t
186  * @param ptr Original pointer returned from a previous allocation
187  * @param align Alignment in bytes, must be a power of two
188  * @param bytes Number of bytes requested for the new block
189  * @return Pointer to memory the caller can now use, or NULL
190  */
191 void *sys_multi_heap_aligned_realloc(struct sys_multi_heap *mheap, void *cfg,
192 				     void *ptr, size_t align, size_t bytes);
193 
194 #define sys_multi_heap_realloc(mheap, cfg, ptr, bytes) \
195 	sys_multi_heap_aligned_realloc(mheap, cfg, ptr, 0, bytes)
196 
197 /**
198  * @}
199  */
200 
201 #endif /* ZEPHYR_INCLUDE_SYS_MULTI_HEAP_H_ */
202