1 /*
2 * Copyright (c) 2018 Intel Corporation
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/kernel.h>
8 #include <string.h>
9 #include "wrapper.h"
10
11 #define TIME_OUT_TICKS 10
12
13 K_MEM_SLAB_DEFINE(cv2_mem_slab, sizeof(struct cv2_mslab),
14 CONFIG_CMSIS_V2_MEM_SLAB_MAX_COUNT, 4);
15
16 static const osMemoryPoolAttr_t init_mslab_attrs = {
17 .name = "ZephyrMemPool",
18 .attr_bits = 0,
19 .cb_mem = NULL,
20 .cb_size = 0,
21 .mp_mem = NULL,
22 .mp_size = 0,
23 };
24
25 /**
26 * @brief Create and Initialize a memory pool.
27 */
osMemoryPoolNew(uint32_t block_count,uint32_t block_size,const osMemoryPoolAttr_t * attr)28 osMemoryPoolId_t osMemoryPoolNew(uint32_t block_count, uint32_t block_size,
29 const osMemoryPoolAttr_t *attr)
30 {
31 struct cv2_mslab *mslab;
32
33 BUILD_ASSERT(K_HEAP_MEM_POOL_SIZE >=
34 CONFIG_CMSIS_V2_MEM_SLAB_MAX_DYNAMIC_SIZE,
35 "heap must be configured to be at least the max dynamic size");
36
37 if (k_is_in_isr()) {
38 return NULL;
39 }
40
41 if ((attr != NULL) && (attr->mp_size < block_count * block_size)) {
42 return NULL;
43 }
44
45 if (attr == NULL) {
46 attr = &init_mslab_attrs;
47 }
48
49 if (k_mem_slab_alloc(&cv2_mem_slab, (void **)&mslab, K_MSEC(100)) == 0) {
50 (void)memset(mslab, 0, sizeof(struct cv2_mslab));
51 } else {
52 return NULL;
53 }
54
55 if (attr->mp_mem == NULL) {
56 __ASSERT((block_count * block_size) <=
57 CONFIG_CMSIS_V2_MEM_SLAB_MAX_DYNAMIC_SIZE,
58 "memory slab/pool size exceeds dynamic maximum");
59
60 mslab->pool = k_calloc(block_count, block_size);
61 if (mslab->pool == NULL) {
62 k_mem_slab_free(&cv2_mem_slab, (void *)mslab);
63 return NULL;
64 }
65 mslab->is_dynamic_allocation = TRUE;
66 } else {
67 mslab->pool = attr->mp_mem;
68 mslab->is_dynamic_allocation = FALSE;
69 }
70
71 int rc = k_mem_slab_init(&mslab->z_mslab, mslab->pool, block_size, block_count);
72
73 if (rc != 0) {
74 k_mem_slab_free(&cv2_mem_slab, (void *)mslab);
75 if (attr->mp_mem == NULL) {
76 k_free(mslab->pool);
77 }
78 return NULL;
79 }
80
81 if (attr->name == NULL) {
82 strncpy(mslab->name, init_mslab_attrs.name,
83 sizeof(mslab->name) - 1);
84 } else {
85 strncpy(mslab->name, attr->name, sizeof(mslab->name) - 1);
86 }
87
88 return (osMemoryPoolId_t)mslab;
89 }
90
91 /**
92 * @brief Allocate a memory block from a memory pool.
93 */
osMemoryPoolAlloc(osMemoryPoolId_t mp_id,uint32_t timeout)94 void *osMemoryPoolAlloc(osMemoryPoolId_t mp_id, uint32_t timeout)
95 {
96 struct cv2_mslab *mslab = (struct cv2_mslab *)mp_id;
97 int retval;
98 void *ptr;
99
100 if (mslab == NULL) {
101 return NULL;
102 }
103
104 /* Can be called from ISRs only if timeout is set to 0 */
105 if (timeout > 0 && k_is_in_isr()) {
106 return NULL;
107 }
108
109 if (timeout == 0U) {
110 retval = k_mem_slab_alloc(
111 (struct k_mem_slab *)(&mslab->z_mslab),
112 (void **)&ptr, K_NO_WAIT);
113 } else if (timeout == osWaitForever) {
114 retval = k_mem_slab_alloc(
115 (struct k_mem_slab *)(&mslab->z_mslab),
116 (void **)&ptr, K_FOREVER);
117 } else {
118 retval = k_mem_slab_alloc(
119 (struct k_mem_slab *)(&mslab->z_mslab),
120 (void **)&ptr, K_TICKS(timeout));
121 }
122
123 if (retval == 0) {
124 return ptr;
125 } else {
126 return NULL;
127 }
128 }
129
130 /**
131 * @brief Return an allocated memory block back to a specific memory pool.
132 */
osMemoryPoolFree(osMemoryPoolId_t mp_id,void * block)133 osStatus_t osMemoryPoolFree(osMemoryPoolId_t mp_id, void *block)
134 {
135 struct cv2_mslab *mslab = (struct cv2_mslab *)mp_id;
136
137 if (mslab == NULL) {
138 return osErrorParameter;
139 }
140
141 /* Note: Below error code is not supported.
142 * osErrorResource: the memory pool specified by parameter mp_id
143 * is in an invalid memory pool state.
144 */
145
146 k_mem_slab_free((struct k_mem_slab *)(&mslab->z_mslab), (void *)block);
147
148 return osOK;
149 }
150
151 /**
152 * @brief Get name of a Memory Pool object.
153 */
osMemoryPoolGetName(osMemoryPoolId_t mp_id)154 const char *osMemoryPoolGetName(osMemoryPoolId_t mp_id)
155 {
156 struct cv2_mslab *mslab = (struct cv2_mslab *)mp_id;
157
158 if (!k_is_in_isr() && (mslab != NULL)) {
159 return mslab->name;
160 } else {
161 return NULL;
162 }
163 }
164
165 /**
166 * @brief Get maximum number of memory blocks in a Memory Pool.
167 */
osMemoryPoolGetCapacity(osMemoryPoolId_t mp_id)168 uint32_t osMemoryPoolGetCapacity(osMemoryPoolId_t mp_id)
169 {
170 struct cv2_mslab *mslab = (struct cv2_mslab *)mp_id;
171
172 if (mslab == NULL) {
173 return 0;
174 } else {
175 return mslab->z_mslab.info.num_blocks;
176 }
177 }
178
179 /**
180 * @brief Get memory block size in a Memory Pool.
181 */
osMemoryPoolGetBlockSize(osMemoryPoolId_t mp_id)182 uint32_t osMemoryPoolGetBlockSize(osMemoryPoolId_t mp_id)
183 {
184 struct cv2_mslab *mslab = (struct cv2_mslab *)mp_id;
185
186 if (mslab == NULL) {
187 return 0;
188 } else {
189 return mslab->z_mslab.info.block_size;
190 }
191 }
192
193 /**
194 * @brief Get number of memory blocks used in a Memory Pool.
195 */
osMemoryPoolGetCount(osMemoryPoolId_t mp_id)196 uint32_t osMemoryPoolGetCount(osMemoryPoolId_t mp_id)
197 {
198 struct cv2_mslab *mslab = (struct cv2_mslab *)mp_id;
199
200 if (mslab == NULL) {
201 return 0;
202 } else {
203 return k_mem_slab_num_used_get(&mslab->z_mslab);
204 }
205 }
206
207 /**
208 * @brief Get number of memory blocks available in a Memory Pool.
209 */
osMemoryPoolGetSpace(osMemoryPoolId_t mp_id)210 uint32_t osMemoryPoolGetSpace(osMemoryPoolId_t mp_id)
211 {
212 struct cv2_mslab *mslab = (struct cv2_mslab *)mp_id;
213
214 if (mslab == NULL) {
215 return 0;
216 } else {
217 return k_mem_slab_num_free_get(&mslab->z_mslab);
218 }
219 }
220
221 /**
222 * @brief Delete a Memory Pool object.
223 */
osMemoryPoolDelete(osMemoryPoolId_t mp_id)224 osStatus_t osMemoryPoolDelete(osMemoryPoolId_t mp_id)
225 {
226 struct cv2_mslab *mslab = (struct cv2_mslab *)mp_id;
227
228 if (mslab == NULL) {
229 return osErrorParameter;
230 }
231
232 if (k_is_in_isr()) {
233 return osErrorISR;
234 }
235
236 /* The status code "osErrorResource" (the memory pool specified by
237 * parameter mp_id is in an invalid memory pool state) is not
238 * supported in Zephyr.
239 */
240
241 if (mslab->is_dynamic_allocation) {
242 k_free(mslab->pool);
243 }
244 k_mem_slab_free(&cv2_mem_slab, (void *)mslab);
245
246 return osOK;
247 }
248