Select one of the symbols to view example projects that use it.
 
Outline
#include "sdkconfig.h"
#include <stdint.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"
#include "freertos/semphr.h"
#include "freertos/stream_buffer.h"
#include "freertos/message_buffer.h"
#include "freertos/idf_additions.h"
#include "esp_heap_caps.h"
#include "esp_log.h"
#include "freertos/portmacro.h"
---------------------------------- Tasks
xTaskCreatePinnedToCoreWithCaps(TaskFunction_t, const char *const, const uint32_t, void *const, UBaseType_t, TaskHandle_t *const, const BaseType_t, UBaseType_t)
prvTaskDeleteWithCaps(TaskHandle_t)
prvTaskDeleteWithCapsTask(void *)
vTaskDeleteWithCaps(TaskHandle_t)
---------------------------------- Queue
xQueueCreateWithCaps(UBaseType_t, UBaseType_t, UBaseType_t)
vQueueDeleteWithCaps(QueueHandle_t)
-------------------------------- Semaphore
xSemaphoreCreateGenericWithCaps(UBaseType_t, UBaseType_t, const uint8_t, UBaseType_t)
vSemaphoreDeleteWithCaps(SemaphoreHandle_t)
------------------------- Stream & Message Buffers
xStreamBufferGenericCreateWithCaps(size_t, size_t, BaseType_t, UBaseType_t)
vStreamBufferGenericDeleteWithCaps(StreamBufferHandle_t, BaseType_t)
Files
loading...
SourceVuESP-IDF Framework and ExamplesFreeRTOSesp_additions/idf_additions.c
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
/* * SPDX-FileCopyrightText: 2023-2024 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 *//* ... */ /* * This file contains the implementation for some the functions in * idf_additions.h *//* ... */ #include "sdkconfig.h" #include <stdint.h> #include "freertos/FreeRTOS.h" #include "freertos/task.h" #include "freertos/queue.h" #include "freertos/semphr.h" #include "freertos/stream_buffer.h" #include "freertos/message_buffer.h" #include "freertos/idf_additions.h" #include "esp_heap_caps.h" #include "esp_log.h" #include "freertos/portmacro.h"12 includes /* -------------------------------------------- Creation With Memory Caps ------------------------------------------- */ /* ---------------------------------- Tasks --------------------------------- */ #if ( configSUPPORT_STATIC_ALLOCATION == 1 ) BaseType_t xTaskCreatePinnedToCoreWithCaps( TaskFunction_t pvTaskCode, const char * const pcName, const configSTACK_DEPTH_TYPE usStackDepth, void * const pvParameters, UBaseType_t uxPriority, TaskHandle_t * const pvCreatedTask, const BaseType_t xCoreID, UBaseType_t uxMemoryCaps ) { TaskHandle_t xHandle; StaticTask_t * pxTaskBuffer; StackType_t * pxStack; /* Allocate memory for the task's TCB. We use pvPortMalloc() here as the * TCB must be in internal memory. *//* ... */ pxTaskBuffer = pvPortMalloc( sizeof( StaticTask_t ) ); /* Allocate memory for the task's stack using the provided memory caps * *//* ... */ pxStack = heap_caps_malloc( usStackDepth, ( uint32_t ) uxMemoryCaps ); if( ( pxTaskBuffer == NULL ) || ( pxStack == NULL ) ) { goto err; }{...} /* Create the task using static creation API*/ xHandle = xTaskCreateStaticPinnedToCore( pvTaskCode, pcName, usStackDepth, pvParameters, uxPriority, pxStack, pxTaskBuffer, xCoreID ); if( xHandle == NULL ) { /* Failed to create task */ goto err; }{...} else if( pvCreatedTask != NULL ) { /* Task created successfully. Return the task handle */ *pvCreatedTask = xHandle; }{...} return pdPASS; err: heap_caps_free( pxStack ); vPortFree( pxTaskBuffer ); return pdFAIL; }{ ... } /* ... */#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */ /*----------------------------------------------------------*/ #if ( configSUPPORT_STATIC_ALLOCATION == 1 ) static void prvTaskDeleteWithCaps( TaskHandle_t xTaskToDelete ) { /* Return value unused if asserts are disabled */ BaseType_t __attribute__( ( unused ) ) xResult; StaticTask_t * pxTaskBuffer; StackType_t * puxStackBuffer; /* The task to be deleted must not be running. * So we suspend the task before deleting it. *//* ... */ vTaskSuspend( xTaskToDelete ); /* Wait for the task to be suspended */ while( eRunning == eTaskGetState( xTaskToDelete ) ) { taskYIELD(); }{...} configASSERT( eRunning != eTaskGetState( xTaskToDelete ) ); xResult = xTaskGetStaticBuffers( xTaskToDelete, &puxStackBuffer, &pxTaskBuffer ); configASSERT( xResult == pdTRUE ); configASSERT( puxStackBuffer != NULL ); configASSERT( pxTaskBuffer != NULL ); /* We can delete the task and free the memory buffers. */ vTaskDelete( xTaskToDelete ); /* Free the memory buffers */ heap_caps_free( puxStackBuffer ); vPortFree( pxTaskBuffer ); }{ ... } static void prvTaskDeleteWithCapsTask( void * pvParameters ) { TaskHandle_t xTaskToDelete = ( TaskHandle_t ) pvParameters; /* Delete the WithCaps task */ prvTaskDeleteWithCaps( xTaskToDelete ); /* Delete the temporary clean up task */ vTaskDelete( NULL ); }{ ... } void vTaskDeleteWithCaps( TaskHandle_t xTaskToDelete ) { /* THIS FUNCTION SHOULD NOT BE CALLED FROM AN INTERRUPT CONTEXT. */ /* TODO: Update it to use portASSERT_IF_IN_ISR() instead. (IDF-10540) */ vPortAssertIfInISR(); TaskHandle_t xCurrentTaskHandle = xTaskGetCurrentTaskHandle(); configASSERT( xCurrentTaskHandle != NULL ); if( ( xTaskToDelete == NULL ) || ( xTaskToDelete == xCurrentTaskHandle ) ) { /* The WithCaps task is deleting itself. While, the task can put itself on the * xTasksWaitingTermination list via the vTaskDelete() call, the idle * task will not free the task TCB and stack memories we created statically * during xTaskCreateWithCaps() or xTaskCreatePinnedToCoreWithCaps(). This * task will never be rescheduled once it is on the xTasksWaitingTermination * list and will not be able to clear the memories. Therefore, it will leak memory. * * To avoid this, we create a new "temporary clean up" task to delete the current task. * This task is created at the priority of the task to be deleted with the same core * affitinty. Its limited purpose is to delete the self-deleting task created WithCaps. * * This approach has the following problems - * 1. Once a WithCaps task deletes itself via vTaskDeleteWithCaps(), it may end up in the * suspended tasks lists for a short time before being deleted. This can give an incorrect * picture about the system state. * * 2. This approach is wasteful and can be error prone. The temporary clean up task will need * system resources to get scheduled and cleanup the WithCaps task. It can be a problem if the system * has several self-deleting WithCaps tasks. * * TODO: A better approach could be either - * * 1. Delegate memory management to the application/user. This way the kernel needn't bother about freeing * the memory (like other static memory task creation APIs like xTaskCreateStatic()) (IDF-10521) * * 2. Have a post deletion hook/callback from the IDLE task to notify higher layers when it is safe to * perform activities such as clearing up the TCB and stack memories. (IDF-10522) *//* ... */ if( xTaskCreatePinnedToCore( ( TaskFunction_t ) prvTaskDeleteWithCapsTask, "prvTaskDeleteWithCapsTask", configMINIMAL_STACK_SIZE, xCurrentTaskHandle, uxTaskPriorityGet( xTaskToDelete ), NULL, xPortGetCoreID() ) != pdFAIL ) { /* Although the current task should get preemted immediately when prvTaskDeleteWithCapsTask is created, * for safety, we suspend the current task and wait for prvTaskDeleteWithCapsTask to delete it. *//* ... */ vTaskSuspend( xTaskToDelete ); /* Should never reach here */ ESP_LOGE( "freertos_additions", "%s: Failed to suspend the task to be deleted", __func__ ); abort(); }{...} else { /* Failed to create the task to delete the current task. */ ESP_LOGE( "freertos_additions", "%s: Failed to create the task to delete the current task", __func__ ); abort(); }{...} }{...} /* Delete the WithCaps task */ prvTaskDeleteWithCaps( xTaskToDelete ); }{ ... } /* ... */#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */ /*----------------------------------------------------------*/ ---------------------------------- Tasks /* ---------------------------------- Queue --------------------------------- */ #if ( configSUPPORT_STATIC_ALLOCATION == 1 ) QueueHandle_t xQueueCreateWithCaps( UBaseType_t uxQueueLength, UBaseType_t uxItemSize, UBaseType_t uxMemoryCaps ) { QueueHandle_t xQueue; StaticQueue_t * pxQueueBuffer; uint8_t * pucQueueStorageBuffer; /* Allocate memory for the queue using the provided memory caps */ pxQueueBuffer = heap_caps_malloc( sizeof( StaticQueue_t ), ( uint32_t ) uxMemoryCaps ); if( uxItemSize == 0 ) { pucQueueStorageBuffer = NULL; }{...} else { pucQueueStorageBuffer = heap_caps_malloc( uxQueueLength * uxItemSize, ( uint32_t ) uxMemoryCaps ); }{...} if( ( pxQueueBuffer == NULL ) || ( ( uxItemSize > 0 ) && ( pucQueueStorageBuffer == NULL ) ) ) { goto err; }{...} /* Create the queue using static creation API */ xQueue = xQueueCreateStatic( uxQueueLength, uxItemSize, pucQueueStorageBuffer, pxQueueBuffer ); if( xQueue == NULL ) { goto err; }{...} return xQueue; err: heap_caps_free( pucQueueStorageBuffer ); heap_caps_free( pxQueueBuffer ); return NULL; }{ ... } /* ... */#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */ /*----------------------------------------------------------*/ #if ( configSUPPORT_STATIC_ALLOCATION == 1 ) void vQueueDeleteWithCaps( QueueHandle_t xQueue ) { /* Return value unused if asserts are disabled */ BaseType_t __attribute__( ( unused ) ) xResult; StaticQueue_t * pxQueueBuffer; uint8_t * pucQueueStorageBuffer; /* Retrieve the buffers used to create the queue before deleting it */ xResult = xQueueGetStaticBuffers( xQueue, &pucQueueStorageBuffer, &pxQueueBuffer ); configASSERT( xResult == pdTRUE ); /* Delete the queue */ vQueueDelete( xQueue ); /* Free the memory buffers */ heap_caps_free( pxQueueBuffer ); heap_caps_free( pucQueueStorageBuffer ); }{ ... } /* ... */#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */ /*----------------------------------------------------------*/ ---------------------------------- Queue /* -------------------------------- Semaphore ------------------------------- */ #if ( configSUPPORT_STATIC_ALLOCATION == 1 ) SemaphoreHandle_t xSemaphoreCreateGenericWithCaps( UBaseType_t uxMaxCount, UBaseType_t uxInitialCount, const uint8_t ucQueueType, UBaseType_t uxMemoryCaps ) { SemaphoreHandle_t xSemaphore; StaticSemaphore_t * pxSemaphoreBuffer; /* Allocate memory for the semaphore using the provided memory caps */ pxSemaphoreBuffer = heap_caps_malloc( sizeof( StaticSemaphore_t ), ( uint32_t ) uxMemoryCaps ); if( pxSemaphoreBuffer == NULL ) { return NULL; }{...} /* Create the semaphore using static creation API */ if( ucQueueType == queueQUEUE_TYPE_MUTEX ) { xSemaphore = xSemaphoreCreateMutexStatic( pxSemaphoreBuffer ); }{...} else if( ucQueueType == queueQUEUE_TYPE_COUNTING_SEMAPHORE ) { xSemaphore = xSemaphoreCreateCountingStatic( uxMaxCount, uxInitialCount, pxSemaphoreBuffer ); }{...} else if( ucQueueType == queueQUEUE_TYPE_BINARY_SEMAPHORE ) { xSemaphore = xSemaphoreCreateBinaryStatic( pxSemaphoreBuffer ); }{...} else /* ucQueueType == queueQUEUE_TYPE_RECURSIVE_MUTEX */ { xSemaphore = xSemaphoreCreateRecursiveMutexStatic( pxSemaphoreBuffer ); }{...} if( xSemaphore == NULL ) { heap_caps_free( pxSemaphoreBuffer ); }{...} return xSemaphore; }{ ... } /* ... */#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */ /*----------------------------------------------------------*/ #if ( configSUPPORT_STATIC_ALLOCATION == 1 ) void vSemaphoreDeleteWithCaps( SemaphoreHandle_t xSemaphore ) { /* Return value unused if asserts are disabled */ BaseType_t __attribute__( ( unused ) ) xResult; StaticSemaphore_t * pxSemaphoreBuffer; /* Retrieve the buffer used to create the semaphore before deleting it * *//* ... */ xResult = xSemaphoreGetStaticBuffer( xSemaphore, &pxSemaphoreBuffer ); configASSERT( xResult == pdTRUE ); /* Delete the semaphore */ vSemaphoreDelete( xSemaphore ); /* Free the memory buffer */ heap_caps_free( pxSemaphoreBuffer ); }{ ... } /* ... */#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */ /*----------------------------------------------------------*/ -------------------------------- Semaphore /* ------------------------- Stream & Message Buffers ----------------------- */ #if ( configSUPPORT_STATIC_ALLOCATION == 1 ) StreamBufferHandle_t xStreamBufferGenericCreateWithCaps( size_t xBufferSizeBytes, size_t xTriggerLevelBytes, BaseType_t xIsMessageBuffer, UBaseType_t uxMemoryCaps ) { StreamBufferHandle_t xStreamBuffer; StaticStreamBuffer_t * pxStaticStreamBuffer; uint8_t * pucStreamBufferStorageArea; /* Allocate memory for the stream or message buffer using the provided * memory caps *//* ... */ pxStaticStreamBuffer = heap_caps_malloc( sizeof( StaticStreamBuffer_t ), ( uint32_t ) uxMemoryCaps ); pucStreamBufferStorageArea = heap_caps_malloc( xBufferSizeBytes, ( uint32_t ) uxMemoryCaps ); if( ( pxStaticStreamBuffer == NULL ) || ( pucStreamBufferStorageArea == NULL ) ) { goto err; }{...} /* Create the stream or message buffer using static creation API */ if( xIsMessageBuffer == pdTRUE ) { xStreamBuffer = ( StreamBufferHandle_t ) xMessageBufferCreateStatic( xBufferSizeBytes, pucStreamBufferStorageArea, pxStaticStreamBuffer ); }{...} else { xStreamBuffer = xStreamBufferCreateStatic( xBufferSizeBytes, xTriggerLevelBytes, pucStreamBufferStorageArea, pxStaticStreamBuffer ); }{...} if( xStreamBuffer == NULL ) { goto err; }{...} return xStreamBuffer; err: heap_caps_free( pucStreamBufferStorageArea ); heap_caps_free( pxStaticStreamBuffer ); return NULL; }{ ... } /* ... */#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */ /*----------------------------------------------------------*/ #if ( configSUPPORT_STATIC_ALLOCATION == 1 ) void vStreamBufferGenericDeleteWithCaps( StreamBufferHandle_t xStreamBuffer, BaseType_t xIsMessageBuffer ) { /* Return value unused if asserts are disabled */ BaseType_t __attribute__( ( unused ) ) xResult; StaticStreamBuffer_t * pxStaticStreamBuffer; uint8_t * pucStreamBufferStorageArea; /* Retrieve the buffers used to create the stream or message buffer * before deleting it *//* ... */ if( xIsMessageBuffer == pdTRUE ) { xResult = xMessageBufferGetStaticBuffers( xStreamBuffer, &pucStreamBufferStorageArea, &pxStaticStreamBuffer ); }{...} else { xResult = xStreamBufferGetStaticBuffers( xStreamBuffer, &pucStreamBufferStorageArea, &pxStaticStreamBuffer ); }{...} configASSERT( xResult == pdTRUE ); /* Delete the stream or message buffer */ if( xIsMessageBuffer == pdTRUE ) { vMessageBufferDelete( xStreamBuffer ); }{...} else { vSemaphoreDelete( xStreamBuffer ); }{...} /* Free the memory buffers */ heap_caps_free( pxStaticStreamBuffer ); heap_caps_free( pucStreamBufferStorageArea ); }{ ... } /* ... */#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */------------------------- Stream & Message Buffers /*----------------------------------------------------------*/
Details
Show:
from
Types: Columns:
This file uses the notable symbols shown below. Click anywhere in the file to view more details.