2022-01-29 16:49:56 +08:00
/*
* SPDX - FileCopyrightText : 2020 Amazon . com , Inc . or its affiliates
*
* SPDX - License - Identifier : MIT
*
* SPDX - FileContributor : 2016 - 2022 Espressif Systems ( Shanghai ) CO LTD
*/
2022-11-02 19:49:40 +08:00
2016-08-17 23:08:22 +08:00
/*
2021-08-24 12:06:21 +08:00
* FreeRTOS Kernel V10 .4 .3
2021-08-10 05:13:43 +08:00
* Copyright ( C ) 2020 Amazon . com , Inc . or its affiliates . All Rights Reserved .
2019-11-28 15:27:47 -03:00
*
* Permission is hereby granted , free of charge , to any person obtaining a copy of
* this software and associated documentation files ( the " Software " ) , to deal in
* the Software without restriction , including without limitation the rights to
* use , copy , modify , merge , publish , distribute , sublicense , and / or sell copies of
* the Software , and to permit persons to whom the Software is furnished to do so ,
* subject to the following conditions :
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY , FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER LIABILITY , WHETHER
* IN AN ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM , OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE .
*
2021-08-10 05:13:43 +08:00
* https : //www.FreeRTOS.org
* https : //github.com/FreeRTOS
2019-11-28 15:27:47 -03:00
*
*/
2016-08-17 23:08:22 +08:00
/* Standard includes. */
# include <stdlib.h>
# include <string.h>
/* Defining MPU_WRAPPERS_INCLUDED_FROM_API_FILE prevents task.h from redefining
2021-08-10 05:13:43 +08:00
* all the API functions to use the MPU wrappers . That should only be done when
* task . h is included from an application file . */
2016-08-17 23:08:22 +08:00
# define MPU_WRAPPERS_INCLUDED_FROM_API_FILE
2019-11-28 15:27:47 -03:00
2016-08-17 23:08:22 +08:00
/* FreeRTOS includes. */
# include "FreeRTOS.h"
# include "task.h"
# include "timers.h"
2019-11-28 15:27:47 -03:00
# include "stack_macros.h"
2016-08-17 23:08:22 +08:00
2021-08-30 14:45:31 +08:00
# ifdef ESP_PLATFORM
2022-11-02 19:49:40 +08:00
# undef _REENT_INIT_PTR
# define _REENT_INIT_PTR esp_reent_init
extern void esp_vApplicationIdleHook ( void ) ;
2022-02-17 16:40:52 +08:00
# endif //ESP_PLATFORM
2021-08-30 14:45:31 +08:00
2019-11-28 15:27:47 -03:00
/* Lint e9021, e961 and e750 are suppressed as a MISRA exception justified
2021-08-10 05:13:43 +08:00
* because the MPU ports require MPU_WRAPPERS_INCLUDED_FROM_API_FILE to be defined
* for the header files above , but not in this file , in order to generate the
* correct privileged Vs unprivileged linkage and placement . */
2019-11-28 15:27:47 -03:00
# undef MPU_WRAPPERS_INCLUDED_FROM_API_FILE /*lint !e961 !e750 !e9021. */
2016-08-17 23:08:22 +08:00
/* Set configUSE_STATS_FORMATTING_FUNCTIONS to 2 to include the stats formatting
2021-08-10 05:13:43 +08:00
* functions but without including stdio . h here . */
2016-08-17 23:08:22 +08:00
# if ( configUSE_STATS_FORMATTING_FUNCTIONS == 1 )
2021-08-10 05:13:43 +08:00
/* At the bottom of this file are two optional functions that can be used
* to generate human readable text from the raw data generated by the
* uxTaskGetSystemState ( ) function . Note the formatting functions are provided
* for convenience only , and are NOT considered part of the kernel . */
# include <stdio.h>
2016-08-17 23:08:22 +08:00
# endif /* configUSE_STATS_FORMATTING_FUNCTIONS == 1 ) */
2021-08-10 05:13:43 +08:00
# if ( configUSE_PREEMPTION == 0 )
/* If the cooperative scheduler is being used then a yield should not be
* performed just because a higher priority task has been woken . */
# define taskYIELD_IF_USING_PREEMPTION()
2016-08-17 23:08:22 +08:00
# else
2021-08-10 05:13:43 +08:00
# define taskYIELD_IF_USING_PREEMPTION() portYIELD_WITHIN_API()
2016-08-17 23:08:22 +08:00
# endif
2019-11-28 15:27:47 -03:00
/* Values that can be assigned to the ucNotifyState member of the TCB. */
2021-08-10 05:13:43 +08:00
# define taskNOT_WAITING_NOTIFICATION ( ( uint8_t ) 0 ) /* Must be zero as it is the initialised value. */
# define taskWAITING_NOTIFICATION ( ( uint8_t ) 1 )
# define taskNOTIFICATION_RECEIVED ( ( uint8_t ) 2 )
2016-09-22 12:20:56 +08:00
2019-11-28 15:27:47 -03:00
/*
* The value used to fill the stack of a task when the task is created . This
* is used purely for checking the high water mark for tasks .
*/
2021-08-10 05:13:43 +08:00
# define tskSTACK_FILL_BYTE ( 0xa5U )
2016-09-22 12:20:56 +08:00
2021-08-10 05:13:43 +08:00
/* Bits used to record how a task's stack and TCB were allocated. */
# define tskDYNAMICALLY_ALLOCATED_STACK_AND_TCB ( ( uint8_t ) 0 )
# define tskSTATICALLY_ALLOCATED_STACK_ONLY ( ( uint8_t ) 1 )
# define tskSTATICALLY_ALLOCATED_STACK_AND_TCB ( ( uint8_t ) 2 )
2016-09-22 12:20:56 +08:00
2019-11-28 15:27:47 -03:00
/* If any of the following are set then task stacks are filled with a known
2021-08-10 05:13:43 +08:00
* value so the high water mark can be determined . If none of the following are
* set then don ' t fill the stack so there is no unnecessary dependency on memset . */
# if ( ( configCHECK_FOR_STACK_OVERFLOW > 1 ) || ( configUSE_TRACE_FACILITY == 1 ) || ( INCLUDE_uxTaskGetStackHighWaterMark == 1 ) || ( INCLUDE_uxTaskGetStackHighWaterMark2 == 1 ) )
# define tskSET_NEW_STACKS_TO_KNOWN_VALUE 1
2019-11-28 15:27:47 -03:00
# else
2021-08-10 05:13:43 +08:00
# define tskSET_NEW_STACKS_TO_KNOWN_VALUE 0
2016-08-17 23:08:22 +08:00
# endif
/*
* Macros used by vListTask to indicate which state a task is in .
*/
2021-08-10 05:13:43 +08:00
# define tskRUNNING_CHAR ( 'X' )
# define tskBLOCKED_CHAR ( 'B' )
# define tskREADY_CHAR ( 'R' )
# define tskDELETED_CHAR ( 'D' )
# define tskSUSPENDED_CHAR ( 'S' )
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
/*
2021-08-10 05:13:43 +08:00
* Some kernel aware debuggers require the data the debugger needs access to to
* be global , rather than file scope .
2019-11-28 15:27:47 -03:00
*/
# ifdef portREMOVE_STATIC_QUALIFIER
2021-08-10 05:13:43 +08:00
# define static
2019-11-28 15:27:47 -03:00
# endif
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
/* The name allocated to the Idle task. This can be overridden by defining
2021-08-10 05:13:43 +08:00
* configIDLE_TASK_NAME in FreeRTOSConfig . h . */
2019-11-28 15:27:47 -03:00
# ifndef configIDLE_TASK_NAME
2021-08-10 05:13:43 +08:00
# define configIDLE_TASK_NAME "IDLE"
2019-11-28 15:27:47 -03:00
# endif
2016-08-17 23:08:22 +08:00
# if ( configUSE_PORT_OPTIMISED_TASK_SELECTION == 0 )
2021-08-10 05:13:43 +08:00
/* If configUSE_PORT_OPTIMISED_TASK_SELECTION is 0 then task selection is
* performed in a generic way that is not optimised to any particular
* microcontroller architecture . */
/* uxTopReadyPriority holds the priority of the highest priority ready
* state task . */
# define taskRECORD_READY_PRIORITY( uxPriority ) \
{ \
if ( ( uxPriority ) > uxTopReadyPriority ) \
{ \
uxTopReadyPriority = ( uxPriority ) ; \
} \
} /* taskRECORD_READY_PRIORITY */
/*-----------------------------------------------------------*/
2022-11-23 16:48:34 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
# define taskSELECT_HIGHEST_PRIORITY_TASK() taskSelectHighestPriorityTaskSMP()
2022-11-23 16:48:34 +08:00
# else /* configNUM_CORES > 1 */
2022-11-02 19:49:40 +08:00
# define taskSELECT_HIGHEST_PRIORITY_TASK() \
2021-08-10 05:13:43 +08:00
{ \
UBaseType_t uxTopPriority = uxTopReadyPriority ; \
\
/* Find the highest priority queue that contains ready tasks. */ \
while ( listLIST_IS_EMPTY ( & ( pxReadyTasksLists [ uxTopPriority ] ) ) ) \
{ \
configASSERT ( uxTopPriority ) ; \
- - uxTopPriority ; \
} \
\
/* listGET_OWNER_OF_NEXT_ENTRY indexes through the list, so the tasks of \
2022-11-23 16:48:34 +08:00
* the same priority get an equal share of the processor time . */ \
listGET_OWNER_OF_NEXT_ENTRY ( pxCurrentTCB [ 0 ] , & ( pxReadyTasksLists [ uxTopPriority ] ) ) ; \
uxTopReadyPriority = uxTopPriority ; \
2021-08-10 05:13:43 +08:00
} /* taskSELECT_HIGHEST_PRIORITY_TASK */
2022-11-23 16:48:34 +08:00
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
/*-----------------------------------------------------------*/
/* Define away taskRESET_READY_PRIORITY() and portRESET_READY_PRIORITY() as
* they are only required when a port optimised method of task selection is
* being used . */
# define taskRESET_READY_PRIORITY( uxPriority )
# define portRESET_READY_PRIORITY( uxPriority, uxTopReadyPriority )
2016-08-17 23:08:22 +08:00
# else /* configUSE_PORT_OPTIMISED_TASK_SELECTION */
2021-08-10 05:13:43 +08:00
/* If configUSE_PORT_OPTIMISED_TASK_SELECTION is 1 then task selection is
* performed in a way that is tailored to the particular microcontroller
* architecture being used . */
/* A port optimised version is provided. Call the port defined macros. */
# define taskRECORD_READY_PRIORITY( uxPriority ) portRECORD_READY_PRIORITY( uxPriority, uxTopReadyPriority )
/*-----------------------------------------------------------*/
2022-11-23 16:48:34 +08:00
# define taskSELECT_HIGHEST_PRIORITY_TASK() \
{ \
UBaseType_t uxTopPriority ; \
\
/* Find the highest priority list that contains ready tasks. */ \
portGET_HIGHEST_PRIORITY ( uxTopPriority , uxTopReadyPriority ) ; \
configASSERT ( listCURRENT_LIST_LENGTH ( & ( pxReadyTasksLists [ uxTopPriority ] ) ) > 0 ) ; \
listGET_OWNER_OF_NEXT_ENTRY ( pxCurrentTCB [ 0 ] , & ( pxReadyTasksLists [ uxTopPriority ] ) ) ; \
2021-08-10 05:13:43 +08:00
} /* taskSELECT_HIGHEST_PRIORITY_TASK() */
/*-----------------------------------------------------------*/
/* A port optimised version is provided, call it only if the TCB being reset
* is being referenced from a ready list . If it is referenced from a delayed
* or suspended list then it won ' t be in a ready list . */
# define taskRESET_READY_PRIORITY( uxPriority ) \
{ \
if ( listCURRENT_LIST_LENGTH ( & ( pxReadyTasksLists [ ( uxPriority ) ] ) ) = = ( UBaseType_t ) 0 ) \
{ \
portRESET_READY_PRIORITY ( ( uxPriority ) , ( uxTopReadyPriority ) ) ; \
} \
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_PORT_OPTIMISED_TASK_SELECTION */
/*-----------------------------------------------------------*/
/* pxDelayedTaskList and pxOverflowDelayedTaskList are switched when the tick
2021-08-10 05:13:43 +08:00
* count overflows . */
# define taskSWITCH_DELAYED_LISTS() \
{ \
List_t * pxTemp ; \
\
/* The delayed tasks list should be empty when the lists are switched. */ \
configASSERT ( ( listLIST_IS_EMPTY ( pxDelayedTaskList ) ) ) ; \
\
pxTemp = pxDelayedTaskList ; \
pxDelayedTaskList = pxOverflowDelayedTaskList ; \
pxOverflowDelayedTaskList = pxTemp ; \
xNumOfOverflows + + ; \
prvResetNextTaskUnblockTime ( ) ; \
}
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
/*
* Place the task represented by pxTCB into the appropriate ready list for
* the task . It is inserted at the end of the list .
*/
2021-08-10 05:13:43 +08:00
# define prvAddTaskToReadyList( pxTCB ) \
traceMOVED_TASK_TO_READY_STATE ( pxTCB ) ; \
taskRECORD_READY_PRIORITY ( ( pxTCB ) - > uxPriority ) ; \
vListInsertEnd ( & ( pxReadyTasksLists [ ( pxTCB ) - > uxPriority ] ) , & ( ( pxTCB ) - > xStateListItem ) ) ; \
tracePOST_MOVED_TASK_TO_READY_STATE ( pxTCB )
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
2022-07-29 19:31:24 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
# define prvCheckForYield( pxTCB, xCurCoreID, xYieldEqualPriority ) ( prvCheckForYieldUsingPrioritySMP( ( pxTCB )->uxPriority, ( pxTCB )->xCoreID, xCurCoreID, xYieldEqualPriority ) == pdTRUE )
# define prvCheckForYieldUsingPriority( uxTaskPriority, xTaskCoreID, xCurCoreID, xYieldEqualPriority ) ( prvCheckForYieldUsingPrioritySMP( uxTaskPriority, xTaskCoreID, xCurCoreID, xYieldEqualPriority ) == pdTRUE )
2022-07-29 19:31:24 +08:00
# else
2022-11-02 19:49:40 +08:00
# define prvCheckForYield( pxTargetTCB, xCurCoreID, xYieldEqualPriority ) ( ( ( pxTargetTCB )->uxPriority + ( ( xYieldEqualPriority == pdTRUE ) ? 1 : 0 ) ) > pxCurrentTCB[ 0 ]->uxPriority )
# define prvCheckForYieldUsingPriority( uxTaskPriority, xTaskCoreID, xCurCoreID, xYieldEqualPriority ) ( ( uxTaskPriority + ( ( xYieldEqualPriority == pdTRUE ) ? 1 : 0 ) ) >= pxCurrentTCB[ 0 ]->uxPriority )
2022-07-29 19:31:24 +08:00
# endif /* configNUM_CORES > 1 */
/*-----------------------------------------------------------*/
2022-08-01 17:29:03 +08:00
/*
* Check if a particular task ( using its xCoreID ) can run on a designated core .
* On single core , this macro always evaluates to true .
*/
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
# define taskCAN_RUN_ON_CORE( xCore, xCoreID ) ( ( ( ( xCoreID ) == xCore ) || ( ( xCoreID ) == tskNO_AFFINITY ) ) ? pdTRUE : pdFALSE )
2022-08-01 17:29:03 +08:00
# else
2022-11-02 19:49:40 +08:00
# define taskCAN_RUN_ON_CORE( xCore, xCoreID ) ( pdTRUE )
2022-08-01 17:29:03 +08:00
# endif /* configNUM_CORES > 1 */
2022-08-24 16:25:21 +08:00
/* Check if a task is a currently running task. */
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
# define taskIS_CURRENTLY_RUNNING( pxTCB ) ( ( ( pxTCB ) == pxCurrentTCB[ 0 ] ) || ( ( pxTCB ) == pxCurrentTCB[ 1 ] ) )
# define taskIS_CURRENTLY_RUNNING_ON_CORE( pxTCB, xCoreID ) ( ( pxTCB ) == pxCurrentTCB[ ( xCoreID ) ] )
2022-08-24 16:25:21 +08:00
# else
2022-11-02 19:49:40 +08:00
# define taskIS_CURRENTLY_RUNNING( pxTCB ) ( ( pxTCB ) == pxCurrentTCB[ 0 ] )
# define taskIS_CURRENTLY_RUNNING_ON_CORE( pxTCB, xCoreID ) taskIS_CURRENTLY_RUNNING( ( pxTCB ) )
2022-08-24 16:25:21 +08:00
# endif /* configNUM_CORES > 1 */
2016-08-17 23:08:22 +08:00
/*
2021-08-10 05:13:43 +08:00
* Several functions take a TaskHandle_t parameter that can optionally be NULL ,
2016-08-17 23:08:22 +08:00
* where NULL is used to indicate that the handle of the currently executing
* task should be used in place of the parameter . This macro simply checks to
* see if the parameter is NULL and returns a pointer to the appropriate TCB .
*/
2021-09-30 00:18:19 +08:00
# if configNUM_CORES > 1
2021-04-19 15:59:44 +10:00
/* In SMP, we need to disable interrupts if getting the current task handle outside a critical section. Calling xTaskGetCurrentTaskHandle() ensures this. */
2022-11-02 19:49:40 +08:00
# define prvGetTCBFromHandle( pxHandle ) ( ( ( pxHandle ) == NULL ) ? xTaskGetCurrentTaskHandle() : ( ( TaskHandle_t ) pxHandle ) )
2021-04-19 15:59:44 +10:00
# else
2022-11-02 19:49:40 +08:00
# define prvGetTCBFromHandle( pxHandle ) ( ( ( pxHandle ) == NULL ) ? ( TaskHandle_t ) pxCurrentTCB[ 0 ] : ( ( TaskHandle_t ) pxHandle ) )
2021-04-19 15:59:44 +10:00
# endif
2016-08-17 23:08:22 +08:00
/* The item value of the event list item is normally used to hold the priority
2021-08-10 05:13:43 +08:00
* of the task to which it belongs ( coded to allow it to be held in reverse
* priority order ) . However , it is occasionally borrowed for other purposes . It
* is important its value is not updated due to a task priority change while it is
* being used for another purpose . The following bit definition is used to inform
* the scheduler that the value should not be changed - in which case it is the
* responsibility of whichever module is using the value to ensure it gets set back
* to its original value when it is released . */
# if ( configUSE_16_BIT_TICKS == 1 )
# define taskEVENT_LIST_ITEM_VALUE_IN_USE 0x8000U
2016-08-17 23:08:22 +08:00
# else
2021-08-10 05:13:43 +08:00
# define taskEVENT_LIST_ITEM_VALUE_IN_USE 0x80000000UL
2016-08-17 23:08:22 +08:00
# endif
2019-11-28 15:27:47 -03:00
/*
* Task control block . A task control block ( TCB ) is allocated for each task ,
* and stores task state information , including a pointer to the task ' s context
* ( the task ' s run time environment , including register values )
*/
2021-08-10 05:13:43 +08:00
typedef struct tskTaskControlBlock /* The old naming convention is used to prevent breaking kernel aware debuggers. */
2019-11-28 15:27:47 -03:00
{
2021-08-10 05:13:43 +08:00
volatile StackType_t * pxTopOfStack ; /*< Points to the location of the last item placed on the tasks stack. THIS MUST BE THE FIRST MEMBER OF THE TCB STRUCT. */
# if ( portUSING_MPU_WRAPPERS == 1 )
xMPU_SETTINGS xMPUSettings ; /*< The MPU settings are defined as part of the port layer. THIS MUST BE THE SECOND MEMBER OF THE TCB STRUCT. */
# endif
ListItem_t xStateListItem ; /*< The list that the state list item of a task is reference from denotes the state of that task (Ready, Blocked, Suspended ). */
ListItem_t xEventListItem ; /*< Used to reference a task from an event list. */
UBaseType_t uxPriority ; /*< The priority of the task. 0 is the lowest priority. */
StackType_t * pxStack ; /*< Points to the start of the stack. */
char pcTaskName [ configMAX_TASK_NAME_LEN ] ; /*< Descriptive name given to the task when created. Facilitates debugging only. */ /*lint !e971 Unqualified char types are allowed for strings and single characters only. */
BaseType_t xCoreID ; /*< Core this task is pinned to */
# if ( ( portSTACK_GROWTH > 0 ) || ( configRECORD_STACK_HIGH_ADDRESS == 1 ) )
StackType_t * pxEndOfStack ; /*< Points to the highest valid address for the stack. */
# endif
# if ( portCRITICAL_NESTING_IN_TCB == 1 )
UBaseType_t uxCriticalNesting ; /*< Holds the critical section nesting depth for ports that do not maintain their own count in the port layer. */
# endif
# if ( configUSE_TRACE_FACILITY == 1 )
UBaseType_t uxTCBNumber ; /*< Stores a number that increments each time a TCB is created. It allows debuggers to determine when a task has been deleted and then recreated. */
UBaseType_t uxTaskNumber ; /*< Stores a number specifically for use by third party trace code. */
# endif
# if ( configUSE_MUTEXES == 1 )
UBaseType_t uxBasePriority ; /*< The priority last assigned to the task - used by the priority inheritance mechanism. */
UBaseType_t uxMutexesHeld ;
# endif
# if ( configUSE_APPLICATION_TASK_TAG == 1 )
TaskHookFunction_t pxTaskTag ;
# endif
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS > 0 )
void * pvThreadLocalStoragePointers [ configNUM_THREAD_LOCAL_STORAGE_POINTERS ] ;
2022-11-08 17:48:48 +08:00
# if ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 )
2022-11-02 19:49:40 +08:00
TlsDeleteCallbackFunction_t pvThreadLocalStoragePointersDelCallback [ configNUM_THREAD_LOCAL_STORAGE_POINTERS ] ;
# endif
2021-08-10 05:13:43 +08:00
# endif
# if ( configGENERATE_RUN_TIME_STATS == 1 )
uint32_t ulRunTimeCounter ; /*< Stores the amount of time the task has spent in the Running state. */
# endif
# if ( configUSE_NEWLIB_REENTRANT == 1 )
/* Allocate a Newlib reent structure that is specific to this task.
* Note Newlib support has been included by popular demand , but is not
* used by the FreeRTOS maintainers themselves . FreeRTOS is not
* responsible for resulting newlib operation . User must be familiar with
* newlib and must provide system - wide implementations of the necessary
* stubs . Be warned that ( at the time of writing ) the current newlib design
* implements a system - wide malloc ( ) that must be provided with locks .
*
* See the third party link http : //www.nadler.com/embedded/newlibAndFreeRTOS.html
* for additional information . */
struct _reent xNewLib_reent ;
# endif
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2021-09-06 12:46:14 +08:00
volatile uint32_t ulNotifiedValue [ configTASK_NOTIFICATION_ARRAY_ENTRIES ] ;
volatile uint8_t ucNotifyState [ configTASK_NOTIFICATION_ARRAY_ENTRIES ] ;
2021-08-10 05:13:43 +08:00
# endif
/* See the comments in FreeRTOS.h with the definition of
* tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE . */
# if ( tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE != 0 ) /*lint !e731 !e9029 Macro has been consolidated for readability reasons. */
uint8_t ucStaticallyAllocated ; /*< Set to pdTRUE if the task is a statically allocated to ensure no attempt is made to free the memory. */
# endif
# if ( INCLUDE_xTaskAbortDelay == 1 )
uint8_t ucDelayAborted ;
# endif
# if ( configUSE_POSIX_ERRNO == 1 )
int iTaskErrno ;
# endif
2019-11-28 15:27:47 -03:00
} tskTCB ;
/* The old tskTCB name is maintained above then typedefed to the new TCB_t name
2021-08-10 05:13:43 +08:00
* below to enable the use of older kernel aware debuggers . */
2019-11-28 15:27:47 -03:00
typedef tskTCB TCB_t ;
/*lint -save -e956 A manual analysis and inspection has been used to determine
2021-08-10 05:13:43 +08:00
* which static variables must be declared volatile . */
2022-11-02 19:49:40 +08:00
PRIVILEGED_DATA TCB_t * volatile pxCurrentTCB [ configNUM_CORES ] = { NULL } ;
2019-11-28 15:27:47 -03:00
/* Lists for ready and blocked tasks. --------------------
2021-08-10 05:13:43 +08:00
* xDelayedTaskList1 and xDelayedTaskList2 could be moved to function scope but
* doing so breaks some kernel aware debuggers and debuggers that rely on removing
* the static qualifier . */
PRIVILEGED_DATA static List_t pxReadyTasksLists [ configMAX_PRIORITIES ] ; /*< Prioritised ready tasks. */
PRIVILEGED_DATA static List_t xDelayedTaskList1 ; /*< Delayed tasks. */
PRIVILEGED_DATA static List_t xDelayedTaskList2 ; /*< Delayed tasks (two lists are used - one for delays that have overflowed the current tick count. */
PRIVILEGED_DATA static List_t * volatile pxDelayedTaskList ; /*< Points to the delayed task list currently being used. */
PRIVILEGED_DATA static List_t * volatile pxOverflowDelayedTaskList ; /*< Points to the delayed task list currently being used to hold tasks that have overflowed the current tick count. */
2022-11-02 19:49:40 +08:00
PRIVILEGED_DATA static List_t xPendingReadyList [ configNUM_CORES ] ; /*< Tasks that have been readied while the scheduler was suspended. They will be moved to the ready list when the scheduler is resumed. */
2021-08-30 14:45:31 +08:00
2022-08-12 15:52:14 +08:00
/* Spinlock required for SMP critical sections. This lock protects all of the
* kernel ' s data structures such as various tasks lists , flags , and tick counts . */
2022-11-14 15:27:00 +08:00
PRIVILEGED_DATA static portMUX_TYPE xKernelLock = portMUX_INITIALIZER_UNLOCKED ;
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
# if ( INCLUDE_vTaskDelete == 1 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
PRIVILEGED_DATA static List_t xTasksWaitingTermination ; /*< Tasks that have been deleted - but their memory not yet freed. */
PRIVILEGED_DATA static volatile UBaseType_t uxDeletedTasksWaitingCleanUp = ( UBaseType_t ) 0U ;
2019-11-28 15:27:47 -03:00
# endif
# if ( INCLUDE_vTaskSuspend == 1 )
2021-08-10 05:13:43 +08:00
PRIVILEGED_DATA static List_t xSuspendedTaskList ; /*< Tasks that are currently suspended. */
2019-11-28 15:27:47 -03:00
# endif
/* Global POSIX errno. Its value is changed upon context switching to match
2021-08-10 05:13:43 +08:00
* the errno of the currently running task . */
2019-11-28 15:27:47 -03:00
# if ( configUSE_POSIX_ERRNO == 1 )
2021-08-10 05:13:43 +08:00
int FreeRTOS_errno = 0 ;
2019-11-28 15:27:47 -03:00
# endif
/* Other file private variables. --------------------------------*/
2021-08-10 05:13:43 +08:00
PRIVILEGED_DATA static volatile UBaseType_t uxCurrentNumberOfTasks = ( UBaseType_t ) 0U ;
PRIVILEGED_DATA static volatile TickType_t xTickCount = ( TickType_t ) configINITIAL_TICK_COUNT ;
PRIVILEGED_DATA static volatile UBaseType_t uxTopReadyPriority = tskIDLE_PRIORITY ;
PRIVILEGED_DATA static volatile BaseType_t xSchedulerRunning = pdFALSE ;
PRIVILEGED_DATA static volatile TickType_t xPendedTicks = ( TickType_t ) 0U ;
2022-11-02 19:49:40 +08:00
PRIVILEGED_DATA static volatile BaseType_t xYieldPending [ configNUM_CORES ] = { pdFALSE } ;
2021-08-10 05:13:43 +08:00
PRIVILEGED_DATA static volatile BaseType_t xNumOfOverflows = ( BaseType_t ) 0 ;
PRIVILEGED_DATA static UBaseType_t uxTaskNumber = ( UBaseType_t ) 0U ;
PRIVILEGED_DATA static volatile TickType_t xNextTaskUnblockTime = ( TickType_t ) 0U ; /* Initialised to portMAX_DELAY before the scheduler starts. */
2022-11-02 19:49:40 +08:00
PRIVILEGED_DATA static TaskHandle_t xIdleTaskHandle [ configNUM_CORES ] = { NULL } ; /*< Holds the handle of the idle task. The idle task is created automatically when the scheduler is started. */
2019-11-28 15:27:47 -03:00
/* Context switches are held pending while the scheduler is suspended. Also,
2021-08-10 05:13:43 +08:00
* interrupts must not manipulate the xStateListItem of a TCB , or any of the
* lists the xStateListItem can be referenced from , if the scheduler is suspended .
* If an interrupt needs to unblock a task while the scheduler is suspended then it
* moves the task ' s event list item into the xPendingReadyList , ready for the
* kernel to move the task from the pending ready list into the real ready list
* when the scheduler is unsuspended . The pending ready list itself can only be
* accessed from a critical section . */
2022-11-02 19:49:40 +08:00
PRIVILEGED_DATA static volatile UBaseType_t uxSchedulerSuspended [ configNUM_CORES ] = { ( UBaseType_t ) pdFALSE } ;
2019-11-28 15:27:47 -03:00
# if ( configGENERATE_RUN_TIME_STATS == 1 )
2021-09-09 11:54:47 +08:00
/* Do not move these variables to function scope as doing so prevents the
* code working with debuggers that need to remove the static qualifier . */
2022-11-02 19:49:40 +08:00
PRIVILEGED_DATA static uint32_t ulTaskSwitchedInTime [ configNUM_CORES ] = { 0U } ; /*< Holds the value of a timer/counter the last time a task was switched in. */
PRIVILEGED_DATA static uint32_t ulTotalRunTime = 0UL ; /*< Holds the total amount of execution time as defined by the run time counter clock. */
2019-11-28 15:27:47 -03:00
# endif
2022-11-02 19:49:40 +08:00
/* per-CPU flags indicating that we are doing context switch, it is used by apptrace and sysview modules */
/* in order to avoid calls of vPortYield from traceTASK_SWITCHED_IN/OUT when waiting */
/* for locks to be free or for host to read full trace buffer */
PRIVILEGED_DATA static volatile BaseType_t xSwitchingContext [ configNUM_CORES ] = { pdFALSE } ;
2019-11-28 15:27:47 -03:00
/*lint -restore */
/*-----------------------------------------------------------*/
2016-08-17 23:08:22 +08:00
/* Callback function prototypes. --------------------------*/
2022-11-02 19:49:40 +08:00
# if ( configCHECK_FOR_STACK_OVERFLOW > 0 )
2019-11-28 15:27:47 -03:00
2022-11-02 19:49:40 +08:00
extern void vApplicationStackOverflowHook ( TaskHandle_t xTask ,
char * pcTaskName ) ;
2016-08-17 23:08:22 +08:00
# endif
2022-11-02 19:49:40 +08:00
# if ( configUSE_TICK_HOOK > 0 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
extern void vApplicationTickHook ( void ) ; /*lint !e526 Symbol not defined as it is an application callback. */
2019-11-28 15:27:47 -03:00
2016-08-24 12:23:58 +08:00
# endif
2022-11-02 19:49:40 +08:00
# if ( configSUPPORT_STATIC_ALLOCATION == 1 )
2019-11-28 15:27:47 -03:00
2022-11-02 19:49:40 +08:00
extern void vApplicationGetIdleTaskMemory ( StaticTask_t * * ppxIdleTaskTCBBuffer ,
StackType_t * * ppxIdleTaskStackBuffer ,
uint32_t * pulIdleTaskStackSize ) ; /*lint !e526 Symbol not defined as it is an application callback. */
2019-11-28 15:27:47 -03:00
# endif
2016-08-24 12:23:58 +08:00
2016-08-17 23:08:22 +08:00
/* File private functions. --------------------------------*/
/**
* Utility task that simply returns pdTRUE if the task referenced by xTask is
* currently in the Suspended state , or pdFALSE if the task referenced by xTask
* is in any other state .
*/
# if ( INCLUDE_vTaskSuspend == 1 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
static BaseType_t prvTaskIsTaskSuspended ( const TaskHandle_t xTask ) PRIVILEGED_FUNCTION ;
2019-11-28 15:27:47 -03:00
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_vTaskSuspend */
/*
* Utility to ready all the lists used by the scheduler . This is called
* automatically upon the creation of the first task .
*/
static void prvInitialiseTaskLists ( void ) PRIVILEGED_FUNCTION ;
/*
* The idle task , which as all tasks is implemented as a never ending loop .
* The idle task is automatically created and added to the ready lists upon
* creation of the first user task .
*
* The portTASK_FUNCTION_PROTO ( ) macro is used to allow port / compiler specific
* language extensions . The equivalent prototype for this function is :
*
* void prvIdleTask ( void * pvParameters ) ;
*
*/
2021-09-09 11:54:47 +08:00
static portTASK_FUNCTION_PROTO ( prvIdleTask , pvParameters ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
/*
* Utility to free all memory allocated by the scheduler to hold a TCB ,
* including the stack pointed to by the TCB .
*
* This does not free memory allocated by the task itself ( i . e . memory
* allocated by calls to pvPortMalloc from within the tasks application code ) .
*/
# if ( INCLUDE_vTaskDelete == 1 )
2021-08-10 05:13:43 +08:00
static void prvDeleteTCB ( TCB_t * pxTCB ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
# endif
2019-11-28 15:27:47 -03:00
/* Function to call the Thread Local Storage Pointer Deletion Callbacks. Will be
* called during task deletion before prvDeleteTCB is called .
2020-11-10 18:40:01 +11:00
*/
2022-11-08 17:48:48 +08:00
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS > 0 ) && ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 )
2022-11-02 19:49:40 +08:00
static void prvDeleteTLS ( TCB_t * pxTCB ) ;
2017-11-23 22:35:54 +08:00
# endif
2016-08-17 23:08:22 +08:00
/*
* Used only by the idle task . This checks to see if anything has been placed
* in the list of tasks waiting to be deleted . If so the task is cleaned up
* and its TCB deleted .
*/
static void prvCheckTasksWaitingTermination ( void ) PRIVILEGED_FUNCTION ;
/*
* The currently executing task is entering the Blocked state . Add the task to
* either the current or the overflow delayed task list .
*/
2022-08-01 19:06:44 +08:00
static void prvAddCurrentTaskToDelayedList ( TickType_t xTicksToWait ,
const BaseType_t xCanBlockIndefinitely ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
/*
* Fills an TaskStatus_t structure with information on each task that is
* referenced from the pxList list ( which may be a ready list , a delayed list ,
* a suspended list , etc . ) .
*
* THIS FUNCTION IS INTENDED FOR DEBUGGING ONLY , AND SHOULD NOT BE CALLED FROM
* NORMAL APPLICATION CODE .
*/
# if ( configUSE_TRACE_FACILITY == 1 )
2021-08-10 05:13:43 +08:00
static UBaseType_t prvListTasksWithinSingleList ( TaskStatus_t * pxTaskStatusArray ,
List_t * pxList ,
eTaskState eState ) PRIVILEGED_FUNCTION ;
2019-11-28 15:27:47 -03:00
# endif
/*
* Searches pxList for a task with name pcNameToQuery - returning a handle to
* the task if it is found , or NULL if the task is not found .
*/
# if ( INCLUDE_xTaskGetHandle == 1 )
2021-08-10 05:13:43 +08:00
static TCB_t * prvSearchForNameWithinSingleList ( List_t * pxList ,
const char pcNameToQuery [ ] ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
# endif
/*
* When a task is created , the stack of the task is filled with a known value .
* This function determines the ' high water mark ' of the task stack by
* determining how much of the stack remains at the original preset value .
*/
2019-11-28 15:27:47 -03:00
# if ( ( configUSE_TRACE_FACILITY == 1 ) || ( INCLUDE_uxTaskGetStackHighWaterMark == 1 ) || ( INCLUDE_uxTaskGetStackHighWaterMark2 == 1 ) )
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
static configSTACK_DEPTH_TYPE prvTaskCheckFreeStackSpace ( const uint8_t * pucStackByte ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
# endif
/*
* Return the amount of time , in ticks , that will pass before the kernel will
* next move a task from the Blocked state to the Running state .
*
* This conditional compilation should use inequality to 0 , not equality to 1.
* This is to ensure portSUPPRESS_TICKS_AND_SLEEP ( ) can be called when user
* defined low power mode implementations require configUSE_TICKLESS_IDLE to be
* set to a value other than 1.
*/
# if ( configUSE_TICKLESS_IDLE != 0 )
2021-08-10 05:13:43 +08:00
static TickType_t prvGetExpectedIdleTime ( void ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
# endif
/*
* Set xNextTaskUnblockTime to the time at which the next Blocked state task
* will exit the Blocked state .
*/
2021-09-02 15:19:49 +08:00
static void prvResetNextTaskUnblockTime ( void ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
# if ( ( configUSE_TRACE_FACILITY == 1 ) && ( configUSE_STATS_FORMATTING_FUNCTIONS > 0 ) )
2021-08-10 05:13:43 +08:00
/*
* Helper function used to pad task names with spaces when printing out
* human readable tables of task information .
*/
static char * prvWriteNameToBuffer ( char * pcBuffer ,
const char * pcTaskName ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
# endif
2016-09-22 12:20:56 +08:00
/*
* Called after a Task_t structure has been allocated either statically or
* dynamically to fill in the structure ' s members .
*/
2021-08-10 05:13:43 +08:00
static void prvInitialiseNewTask ( TaskFunction_t pxTaskCode ,
const char * const pcName , /*lint !e971 Unqualified char types are allowed for strings and single characters only. */
const uint32_t ulStackDepth ,
void * const pvParameters ,
UBaseType_t uxPriority ,
TaskHandle_t * const pxCreatedTask ,
TCB_t * pxNewTCB ,
const MemoryRegion_t * const xRegions ,
BaseType_t xCoreID ) PRIVILEGED_FUNCTION ;
2016-09-22 12:20:56 +08:00
/*
* Called after a new task has been created and initialised to place the task
* under the control of the scheduler .
*/
2022-08-01 17:29:03 +08:00
static void prvAddNewTaskToReadyList ( TCB_t * pxNewTCB ) PRIVILEGED_FUNCTION ;
2022-07-29 19:31:24 +08:00
# if ( configNUM_CORES > 1 )
/*
* Check whether a yield ( on either core ) is required after unblocking ( or
* changing the priority of ) a particular task .
*
* - This function is the SMP replacement for checking if an unblocked task has
* a higher ( or equal ) priority than the current task .
* - It should be called before calling taskYIELD_IF_USING_PREEMPTION ( ) or
* before setting xYieldRequired
* - If it is the other core that requires a yield , this function will
* internally trigger the other core to yield
*
* Note : In some special instances , a yield is triggered if the unblocked task
* has an equal priority ( such as in xTaskResumeAll ) . Thus the
* xYieldEqualPriority parameter specifies whether to yield if the current
* task has equal priority .
*
* Scheduling Algorithm :
* This function will bias towards yielding the current core .
* - If the unblocked task has a higher ( or equal ) priority than then current
* core , the current core is yielded regardless of the current priority of the
* other core .
* - A core ( current or other ) will only yield if their schedulers are not
* suspended .
*
* Todo : This can be optimized ( IDF - 5772 )
*
* Entry :
* - This function must be called in a critical section
* - A task must just have been unblocked , or its priority raised
* Exit :
* - Returns pdTRUE if the current core requires yielding
* - The other core will be triggered to yield if required
*/
2022-11-02 19:49:40 +08:00
static BaseType_t prvCheckForYieldUsingPrioritySMP ( UBaseType_t uxTaskPriority ,
BaseType_t xTaskCoreID ,
BaseType_t xCurCoreID ,
BaseType_t xYieldEqualPriority ) PRIVILEGED_FUNCTION ;
2022-07-29 19:31:24 +08:00
# endif /* configNUM_CORES > 1 */
2019-11-28 15:27:47 -03:00
/*
* freertos_tasks_c_additions_init ( ) should only be called if the user definable
* macro FREERTOS_TASKS_C_ADDITIONS_INIT ( ) is defined , as that is the only macro
* called by the function .
*/
# ifdef FREERTOS_TASKS_C_ADDITIONS_INIT
2016-09-22 12:20:56 +08:00
2021-08-10 05:13:43 +08:00
static void freertos_tasks_c_additions_init ( void ) PRIVILEGED_FUNCTION ;
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
# endif
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
# if ( configSUPPORT_STATIC_ALLOCATION == 1 )
2016-09-22 12:20:56 +08:00
2022-09-05 17:31:22 +08:00
TaskHandle_t xTaskCreateStaticPinnedToCore ( TaskFunction_t pxTaskCode ,
2021-08-10 05:13:43 +08:00
const char * const pcName ,
const uint32_t ulStackDepth ,
void * const pvParameters ,
UBaseType_t uxPriority ,
2022-09-05 17:31:22 +08:00
StackType_t * const puxStackBuffer ,
2021-08-10 05:13:43 +08:00
StaticTask_t * const pxTaskBuffer ,
const BaseType_t xCoreID )
{
2022-11-02 19:49:40 +08:00
TCB_t * pxNewTCB ;
TaskHandle_t xReturn ;
2021-08-10 05:13:43 +08:00
2022-09-05 17:31:22 +08:00
configASSERT ( portVALID_STACK_MEM ( puxStackBuffer ) ) ;
configASSERT ( portVALID_TCB_MEM ( pxTaskBuffer ) ) ;
configASSERT ( ( ( xCoreID > = 0 ) & & ( xCoreID < configNUM_CORES ) ) | | ( xCoreID = = tskNO_AFFINITY ) ) ;
2021-08-10 05:13:43 +08:00
2021-09-02 15:19:49 +08:00
# if ( configASSERT_DEFINED == 1 )
{
/* Sanity check that the size of the structure used to declare a
* variable of type StaticTask_t equals the size of the real task
* structure . */
volatile size_t xSize = sizeof ( StaticTask_t ) ;
configASSERT ( xSize = = sizeof ( TCB_t ) ) ;
( void ) xSize ; /* Prevent lint warning when configASSERT() is not used. */
}
2021-08-10 05:13:43 +08:00
# endif /* configASSERT_DEFINED */
2022-09-05 17:31:22 +08:00
if ( ( pxTaskBuffer ! = NULL ) & & ( puxStackBuffer ! = NULL ) )
2021-08-10 05:13:43 +08:00
{
/* The memory used for the task's TCB and stack are passed into this
2021-08-26 13:45:51 +08:00
* function - use them . */
2021-08-10 05:13:43 +08:00
pxNewTCB = ( TCB_t * ) pxTaskBuffer ; /*lint !e740 !e9087 Unusual cast is ok as the structures are designed to have the same alignment, and the size is checked by an assert. */
2022-09-05 17:31:22 +08:00
pxNewTCB - > pxStack = ( StackType_t * ) puxStackBuffer ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
# if ( tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE != 0 ) /*lint !e731 !e9029 Macro has been consolidated for readability reasons. */
{
/* Tasks can be created statically or dynamically, so note this
* task was created statically in case the task is later deleted . */
pxNewTCB - > ucStaticallyAllocated = tskSTATICALLY_ALLOCATED_STACK_AND_TCB ;
}
2021-08-10 05:13:43 +08:00
# endif /* tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE */
2022-09-05 17:31:22 +08:00
prvInitialiseNewTask ( pxTaskCode , pcName , ulStackDepth , pvParameters , uxPriority , & xReturn , pxNewTCB , NULL , xCoreID ) ;
2022-08-01 17:29:03 +08:00
prvAddNewTaskToReadyList ( pxNewTCB ) ;
2021-08-10 05:13:43 +08:00
}
else
{
xReturn = NULL ;
}
return xReturn ;
}
2016-09-22 12:20:56 +08:00
# endif /* SUPPORT_STATIC_ALLOCATION */
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
# if ( ( portUSING_MPU_WRAPPERS == 1 ) && ( configSUPPORT_STATIC_ALLOCATION == 1 ) )
2019-11-28 15:27:47 -03:00
2021-08-26 13:45:51 +08:00
BaseType_t xTaskCreateRestrictedStatic ( const TaskParameters_t * const pxTaskDefinition ,
TaskHandle_t * pxCreatedTask )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * pxNewTCB ;
BaseType_t xReturn = errCOULD_NOT_ALLOCATE_REQUIRED_MEMORY ;
2021-08-10 05:13:43 +08:00
configASSERT ( pxTaskDefinition - > puxStackBuffer ! = NULL ) ;
configASSERT ( pxTaskDefinition - > pxTaskBuffer ! = NULL ) ;
if ( ( pxTaskDefinition - > puxStackBuffer ! = NULL ) & & ( pxTaskDefinition - > pxTaskBuffer ! = NULL ) )
{
/* Allocate space for the TCB. Where the memory comes from depends
2021-08-26 13:45:51 +08:00
* on the implementation of the port malloc function and whether or
* not static allocation is being used . */
2021-08-10 05:13:43 +08:00
pxNewTCB = ( TCB_t * ) pxTaskDefinition - > pxTaskBuffer ;
/* Store the stack location in the TCB. */
pxNewTCB - > pxStack = pxTaskDefinition - > puxStackBuffer ;
2021-08-26 13:45:51 +08:00
# if ( tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE != 0 )
{
/* Tasks can be created statically or dynamically, so note this
* task was created statically in case the task is later deleted . */
pxNewTCB - > ucStaticallyAllocated = tskSTATICALLY_ALLOCATED_STACK_AND_TCB ;
}
2021-08-10 05:13:43 +08:00
# endif /* tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE */
2021-08-26 13:45:51 +08:00
prvInitialiseNewTask ( pxTaskDefinition - > pvTaskCode ,
pxTaskDefinition - > pcName ,
( uint32_t ) pxTaskDefinition - > usStackDepth ,
pxTaskDefinition - > pvParameters ,
pxTaskDefinition - > uxPriority ,
pxCreatedTask , pxNewTCB ,
pxTaskDefinition - > xRegions ,
tskNO_AFFINITY ) ;
2021-08-10 05:13:43 +08:00
2022-08-01 17:29:03 +08:00
prvAddNewTaskToReadyList ( pxNewTCB ) ;
2021-08-10 05:13:43 +08:00
xReturn = pdPASS ;
}
return xReturn ;
}
2019-11-28 15:27:47 -03:00
# endif /* ( portUSING_MPU_WRAPPERS == 1 ) && ( configSUPPORT_STATIC_ALLOCATION == 1 ) */
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
# if ( ( portUSING_MPU_WRAPPERS == 1 ) && ( configSUPPORT_DYNAMIC_ALLOCATION == 1 ) )
2016-09-22 12:20:56 +08:00
2021-09-02 15:19:49 +08:00
BaseType_t xTaskCreateRestricted ( const TaskParameters_t * const pxTaskDefinition ,
TaskHandle_t * pxCreatedTask )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * pxNewTCB ;
BaseType_t xReturn = errCOULD_NOT_ALLOCATE_REQUIRED_MEMORY ;
2021-08-10 05:13:43 +08:00
configASSERT ( pxTaskDefinition - > puxStackBuffer ) ;
if ( pxTaskDefinition - > puxStackBuffer ! = NULL )
{
/* Allocate space for the TCB. Where the memory comes from depends
2021-08-26 13:45:51 +08:00
* on the implementation of the port malloc function and whether or
* not static allocation is being used . */
2021-08-10 05:13:43 +08:00
pxNewTCB = ( TCB_t * ) pvPortMallocTcbMem ( sizeof ( TCB_t ) ) ;
if ( pxNewTCB ! = NULL )
{
/* Store the stack location in the TCB. */
pxNewTCB - > pxStack = pxTaskDefinition - > puxStackBuffer ;
2021-08-26 13:45:51 +08:00
# if ( tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE != 0 )
{
/* Tasks can be created statically or dynamically, so note
* this task had a statically allocated stack in case it is
* later deleted . The TCB was allocated dynamically . */
pxNewTCB - > ucStaticallyAllocated = tskSTATICALLY_ALLOCATED_STACK_ONLY ;
}
2021-08-10 05:13:43 +08:00
# endif /* tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE */
2021-08-26 13:45:51 +08:00
prvInitialiseNewTask ( pxTaskDefinition - > pvTaskCode ,
pxTaskDefinition - > pcName ,
( uint32_t ) pxTaskDefinition - > usStackDepth ,
pxTaskDefinition - > pvParameters ,
pxTaskDefinition - > uxPriority ,
pxCreatedTask , pxNewTCB ,
pxTaskDefinition - > xRegions ,
tskNO_AFFINITY ) ;
2021-08-10 05:13:43 +08:00
2022-08-01 17:29:03 +08:00
prvAddNewTaskToReadyList ( pxNewTCB ) ;
2021-08-10 05:13:43 +08:00
xReturn = pdPASS ;
}
}
return xReturn ;
}
2016-09-22 12:20:56 +08:00
# endif /* portUSING_MPU_WRAPPERS */
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
# if ( configSUPPORT_DYNAMIC_ALLOCATION == 1 )
2016-09-22 12:20:56 +08:00
2022-09-05 17:31:22 +08:00
BaseType_t xTaskCreatePinnedToCore ( TaskFunction_t pxTaskCode ,
const char * const pcName , /*lint !e971 Unqualified char types are allowed for strings and single characters only. */
const configSTACK_DEPTH_TYPE usStackDepth ,
void * const pvParameters ,
UBaseType_t uxPriority ,
TaskHandle_t * const pxCreatedTask ,
2022-11-02 19:49:40 +08:00
const BaseType_t xCoreID )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * pxNewTCB ;
BaseType_t xReturn ;
2021-08-10 05:13:43 +08:00
/* If the stack grows down then allocate the stack then the TCB so the stack
2021-08-26 13:45:51 +08:00
* does not grow into the TCB . Likewise if the stack grows up then allocate
* the TCB then the stack . */
# if ( portSTACK_GROWTH > 0 )
{
/* Allocate space for the TCB. Where the memory comes from depends on
* the implementation of the port malloc function and whether or not static
* allocation is being used . */
2022-11-02 19:49:40 +08:00
pxNewTCB = ( TCB_t * ) pvPortMallocTcbMem ( sizeof ( TCB_t ) ) ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
if ( pxNewTCB ! = NULL )
{
/* Allocate space for the stack used by the task being created.
* The base of the stack memory stored in the TCB so the task can
* be deleted later if required . */
2022-11-02 19:49:40 +08:00
pxNewTCB - > pxStack = ( StackType_t * ) pvPortMallocStackMem ( ( ( ( size_t ) usStackDepth ) * sizeof ( StackType_t ) ) ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
if ( pxNewTCB - > pxStack = = NULL )
{
/* Could not allocate the stack. Delete the allocated TCB. */
vPortFree ( pxNewTCB ) ;
pxNewTCB = NULL ;
}
2021-08-10 05:13:43 +08:00
}
}
# else /* portSTACK_GROWTH */
2021-08-26 13:45:51 +08:00
{
StackType_t * pxStack ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
/* Allocate space for the stack used by the task being created. */
2022-11-02 19:49:40 +08:00
pxStack = pvPortMallocStackMem ( ( ( ( size_t ) usStackDepth ) * sizeof ( StackType_t ) ) ) ; /*lint !e9079 All values returned by pvPortMalloc() have at least the alignment required by the MCU's stack and this allocation is the stack. */
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
if ( pxStack ! = NULL )
{
/* Allocate space for the TCB. */
2022-11-02 19:49:40 +08:00
pxNewTCB = ( TCB_t * ) pvPortMallocTcbMem ( sizeof ( TCB_t ) ) ; /*lint !e9087 !e9079 All values returned by pvPortMalloc() have at least the alignment required by the MCU's stack, and the first member of TCB_t is always a pointer to the task's stack. */
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
if ( pxNewTCB ! = NULL )
{
/* Store the stack location in the TCB. */
pxNewTCB - > pxStack = pxStack ;
}
else
{
/* The stack cannot be used as the TCB was not created. Free
* it again . */
vPortFree ( pxStack ) ;
}
2021-08-10 05:13:43 +08:00
}
else
{
2021-08-26 13:45:51 +08:00
pxNewTCB = NULL ;
2021-08-10 05:13:43 +08:00
}
}
# endif /* portSTACK_GROWTH */
if ( pxNewTCB ! = NULL )
{
2021-08-26 13:45:51 +08:00
# if ( tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE != 0 ) /*lint !e9029 !e731 Macro has been consolidated for readability reasons. */
{
/* Tasks can be created statically or dynamically, so note this
* task was created dynamically in case it is later deleted . */
pxNewTCB - > ucStaticallyAllocated = tskDYNAMICALLY_ALLOCATED_STACK_AND_TCB ;
}
2021-08-10 05:13:43 +08:00
# endif /* tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE */
2022-09-05 17:31:22 +08:00
prvInitialiseNewTask ( pxTaskCode , pcName , ( uint32_t ) usStackDepth , pvParameters , uxPriority , pxCreatedTask , pxNewTCB , NULL , xCoreID ) ;
2022-08-01 17:29:03 +08:00
prvAddNewTaskToReadyList ( pxNewTCB ) ;
2021-08-10 05:13:43 +08:00
xReturn = pdPASS ;
}
else
{
xReturn = errCOULD_NOT_ALLOCATE_REQUIRED_MEMORY ;
}
return xReturn ;
}
2016-08-17 23:08:22 +08:00
2016-09-22 12:20:56 +08:00
# endif /* configSUPPORT_DYNAMIC_ALLOCATION */
/*-----------------------------------------------------------*/
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
static void prvInitialiseNewTask ( TaskFunction_t pxTaskCode ,
const char * const pcName , /*lint !e971 Unqualified char types are allowed for strings and single characters only. */
const uint32_t ulStackDepth ,
void * const pvParameters ,
UBaseType_t uxPriority ,
TaskHandle_t * const pxCreatedTask ,
2021-08-26 13:45:51 +08:00
TCB_t * pxNewTCB ,
2021-08-10 05:13:43 +08:00
const MemoryRegion_t * const xRegions ,
BaseType_t xCoreID )
2016-09-22 12:20:56 +08:00
{
2021-08-26 13:45:51 +08:00
StackType_t * pxTopOfStack ;
UBaseType_t x ;
2016-08-17 23:08:22 +08:00
2022-09-05 17:31:22 +08:00
# if ( configNUM_CORES == 1 )
2022-11-02 19:49:40 +08:00
{
xCoreID = 0 ;
}
2021-08-10 05:13:43 +08:00
# endif
2021-08-26 13:45:51 +08:00
# if ( portUSING_MPU_WRAPPERS == 1 )
2021-08-10 05:13:43 +08:00
/* Should the task be created in privileged mode? */
BaseType_t xRunPrivileged ;
2021-08-26 13:45:51 +08:00
2021-08-10 05:13:43 +08:00
if ( ( uxPriority & portPRIVILEGE_BIT ) ! = 0U )
{
xRunPrivileged = pdTRUE ;
}
else
{
xRunPrivileged = pdFALSE ;
}
uxPriority & = ~ portPRIVILEGE_BIT ;
# endif /* portUSING_MPU_WRAPPERS == 1 */
/* Avoid dependency on memset() if it is not required. */
2021-08-26 13:45:51 +08:00
# if ( tskSET_NEW_STACKS_TO_KNOWN_VALUE == 1 )
{
/* Fill the stack with a known value to assist debugging. */
( void ) memset ( pxNewTCB - > pxStack , ( int ) tskSTACK_FILL_BYTE , ( size_t ) ulStackDepth * sizeof ( StackType_t ) ) ;
}
2021-08-10 05:13:43 +08:00
# endif /* tskSET_NEW_STACKS_TO_KNOWN_VALUE */
2022-11-02 19:49:40 +08:00
# if ( configUSE_TRACE_FACILITY == 1 )
{
/* Zero the uxTaskNumber TCB member to avoid random value from dynamically allocated TCBs */
pxNewTCB - > uxTaskNumber = 0 ;
}
# endif /* ( configUSE_TRACE_FACILITY == 1 ) */
2021-08-10 05:13:43 +08:00
/* Calculate the top of stack address. This depends on whether the stack
2021-08-26 13:45:51 +08:00
* grows from high memory to low ( as per the 80 x86 ) or vice versa .
* portSTACK_GROWTH is used to make the result positive or negative as required
* by the port . */
# if ( portSTACK_GROWTH < 0 )
{
pxTopOfStack = & ( pxNewTCB - > pxStack [ ulStackDepth - ( uint32_t ) 1 ] ) ;
pxTopOfStack = ( StackType_t * ) ( ( ( portPOINTER_SIZE_TYPE ) pxTopOfStack ) & ( ~ ( ( portPOINTER_SIZE_TYPE ) portBYTE_ALIGNMENT_MASK ) ) ) ; /*lint !e923 !e9033 !e9078 MISRA exception. Avoiding casts between pointers and integers is not practical. Size differences accounted for using portPOINTER_SIZE_TYPE type. Checked by assert(). */
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
/* Check the alignment of the calculated top of stack is correct. */
configASSERT ( ( ( ( portPOINTER_SIZE_TYPE ) pxTopOfStack & ( portPOINTER_SIZE_TYPE ) portBYTE_ALIGNMENT_MASK ) = = 0UL ) ) ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
# if ( configRECORD_STACK_HIGH_ADDRESS == 1 )
{
/* Also record the stack's high address, which may assist
* debugging . */
pxNewTCB - > pxEndOfStack = pxTopOfStack ;
}
# endif /* configRECORD_STACK_HIGH_ADDRESS */
2021-08-10 05:13:43 +08:00
}
# else /* portSTACK_GROWTH */
2021-08-26 13:45:51 +08:00
{
pxTopOfStack = pxNewTCB - > pxStack ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
/* Check the alignment of the stack buffer is correct. */
configASSERT ( ( ( ( portPOINTER_SIZE_TYPE ) pxNewTCB - > pxStack & ( portPOINTER_SIZE_TYPE ) portBYTE_ALIGNMENT_MASK ) = = 0UL ) ) ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
/* The other extreme of the stack space is required if stack checking is
* performed . */
pxNewTCB - > pxEndOfStack = pxNewTCB - > pxStack + ( ulStackDepth - ( uint32_t ) 1 ) ;
}
2021-08-10 05:13:43 +08:00
# endif /* portSTACK_GROWTH */
/* Store the task name in the TCB. */
if ( pcName ! = NULL )
{
for ( x = ( UBaseType_t ) 0 ; x < ( UBaseType_t ) configMAX_TASK_NAME_LEN ; x + + )
{
pxNewTCB - > pcTaskName [ x ] = pcName [ x ] ;
/* Don't copy all configMAX_TASK_NAME_LEN if the string is shorter than
2021-08-26 13:45:51 +08:00
* configMAX_TASK_NAME_LEN characters just in case the memory after the
* string is not accessible ( extremely unlikely ) . */
2021-08-10 05:13:43 +08:00
if ( pcName [ x ] = = ( char ) 0x00 )
{
break ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
/* Ensure the name string is terminated in the case that the string length
2021-08-26 13:45:51 +08:00
* was greater or equal to configMAX_TASK_NAME_LEN . */
2021-08-10 05:13:43 +08:00
pxNewTCB - > pcTaskName [ configMAX_TASK_NAME_LEN - 1 ] = ' \0 ' ;
}
else
{
/* The task has not been given a name, so just ensure there is a NULL
2021-08-26 13:45:51 +08:00
* terminator when it is read out . */
2021-08-10 05:13:43 +08:00
pxNewTCB - > pcTaskName [ 0 ] = 0x00 ;
}
/* This is used as an array index so must ensure it's not too large. First
2021-08-26 13:45:51 +08:00
* remove the privilege bit if one is present . */
2021-08-10 05:13:43 +08:00
if ( uxPriority > = ( UBaseType_t ) configMAX_PRIORITIES )
{
uxPriority = ( UBaseType_t ) configMAX_PRIORITIES - ( UBaseType_t ) 1U ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
pxNewTCB - > uxPriority = uxPriority ;
pxNewTCB - > xCoreID = xCoreID ;
# if ( configUSE_MUTEXES == 1 )
2021-08-26 13:45:51 +08:00
{
pxNewTCB - > uxBasePriority = uxPriority ;
pxNewTCB - > uxMutexesHeld = 0 ;
}
2021-08-10 05:13:43 +08:00
# endif /* configUSE_MUTEXES */
vListInitialiseItem ( & ( pxNewTCB - > xStateListItem ) ) ;
vListInitialiseItem ( & ( pxNewTCB - > xEventListItem ) ) ;
/* Set the pxNewTCB as a link back from the ListItem_t. This is so we can get
2021-08-26 13:45:51 +08:00
* back to the containing TCB from a generic item in a list . */
2021-08-10 05:13:43 +08:00
listSET_LIST_ITEM_OWNER ( & ( pxNewTCB - > xStateListItem ) , pxNewTCB ) ;
/* Event lists are always in priority order. */
listSET_LIST_ITEM_VALUE ( & ( pxNewTCB - > xEventListItem ) , ( TickType_t ) configMAX_PRIORITIES - ( TickType_t ) uxPriority ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
listSET_LIST_ITEM_OWNER ( & ( pxNewTCB - > xEventListItem ) , pxNewTCB ) ;
# if ( portCRITICAL_NESTING_IN_TCB == 1 )
2021-08-26 13:45:51 +08:00
{
pxNewTCB - > uxCriticalNesting = ( UBaseType_t ) 0U ;
}
2021-08-10 05:13:43 +08:00
# endif /* portCRITICAL_NESTING_IN_TCB */
# if ( configUSE_APPLICATION_TASK_TAG == 1 )
{
pxNewTCB - > pxTaskTag = NULL ;
}
# endif /* configUSE_APPLICATION_TASK_TAG */
# if ( configGENERATE_RUN_TIME_STATS == 1 )
{
pxNewTCB - > ulRunTimeCounter = 0UL ;
}
# endif /* configGENERATE_RUN_TIME_STATS */
# if ( portUSING_MPU_WRAPPERS == 1 )
{
vPortStoreTaskMPUSettings ( & ( pxNewTCB - > xMPUSettings ) , xRegions , pxNewTCB - > pxStack , ulStackDepth ) ;
}
# else
{
/* Avoid compiler warning about unreferenced parameter. */
( void ) xRegions ;
}
# endif
2021-08-26 13:45:51 +08:00
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS != 0 )
2021-08-10 05:13:43 +08:00
{
for ( x = 0 ; x < ( UBaseType_t ) configNUM_THREAD_LOCAL_STORAGE_POINTERS ; x + + )
{
pxNewTCB - > pvThreadLocalStoragePointers [ x ] = NULL ;
2022-11-02 19:49:40 +08:00
# if ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 )
pxNewTCB - > pvThreadLocalStoragePointersDelCallback [ x ] = NULL ;
2021-08-10 05:13:43 +08:00
# endif
}
}
# endif
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
{
2021-09-06 12:46:14 +08:00
memset ( ( void * ) & ( pxNewTCB - > ulNotifiedValue [ 0 ] ) , 0x00 , sizeof ( pxNewTCB - > ulNotifiedValue ) ) ;
memset ( ( void * ) & ( pxNewTCB - > ucNotifyState [ 0 ] ) , 0x00 , sizeof ( pxNewTCB - > ucNotifyState ) ) ;
2021-08-10 05:13:43 +08:00
}
# endif
# if ( configUSE_NEWLIB_REENTRANT == 1 )
{
/* Initialise this task's Newlib reent structure. */
2021-10-19 12:07:56 +08:00
_REENT_INIT_PTR ( ( & ( pxNewTCB - > xNewLib_reent ) ) ) ;
2021-08-10 05:13:43 +08:00
}
# endif
# if ( INCLUDE_xTaskAbortDelay == 1 )
{
pxNewTCB - > ucDelayAborted = pdFALSE ;
}
# endif
/* Initialize the TCB stack to look as if the task was already running,
* but had been interrupted by the scheduler . The return address is set
* to the start of the task function . Once the stack has been initialised
* the top of stack variable is updated . */
# if ( portUSING_MPU_WRAPPERS == 1 )
{
/* If the port has capability to detect stack overflow,
* pass the stack end address to the stack initialization
* function as well . */
# if ( portHAS_STACK_OVERFLOW_CHECKING == 1 )
{
# if ( portSTACK_GROWTH < 0 )
{
pxNewTCB - > pxTopOfStack = pxPortInitialiseStack ( pxTopOfStack , pxNewTCB - > pxStack , pxTaskCode , pvParameters , xRunPrivileged ) ;
}
# else /* portSTACK_GROWTH */
{
pxNewTCB - > pxTopOfStack = pxPortInitialiseStack ( pxTopOfStack , pxNewTCB - > pxEndOfStack , pxTaskCode , pvParameters , xRunPrivileged ) ;
}
# endif /* portSTACK_GROWTH */
}
# else /* portHAS_STACK_OVERFLOW_CHECKING */
{
pxNewTCB - > pxTopOfStack = pxPortInitialiseStack ( pxTopOfStack , pxTaskCode , pvParameters , xRunPrivileged ) ;
}
# endif /* portHAS_STACK_OVERFLOW_CHECKING */
}
# else /* portUSING_MPU_WRAPPERS */
{
/* If the port has capability to detect stack overflow,
* pass the stack end address to the stack initialization
* function as well . */
# if ( portHAS_STACK_OVERFLOW_CHECKING == 1 )
{
# if ( portSTACK_GROWTH < 0 )
{
pxNewTCB - > pxTopOfStack = pxPortInitialiseStack ( pxTopOfStack , pxNewTCB - > pxStack , pxTaskCode , pvParameters ) ;
}
# else /* portSTACK_GROWTH */
{
pxNewTCB - > pxTopOfStack = pxPortInitialiseStack ( pxTopOfStack , pxNewTCB - > pxEndOfStack , pxTaskCode , pvParameters ) ;
}
# endif /* portSTACK_GROWTH */
}
# else /* portHAS_STACK_OVERFLOW_CHECKING */
{
pxNewTCB - > pxTopOfStack = pxPortInitialiseStack ( pxTopOfStack , pxTaskCode , pvParameters ) ;
}
# endif /* portHAS_STACK_OVERFLOW_CHECKING */
}
# endif /* portUSING_MPU_WRAPPERS */
if ( pxCreatedTask ! = NULL )
{
/* Pass the handle out in an anonymous way. The handle can be used to
* change the created task ' s priority , delete the created task , etc . */
* pxCreatedTask = ( TaskHandle_t ) pxNewTCB ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2016-09-22 12:20:56 +08:00
}
/*-----------------------------------------------------------*/
2022-08-01 17:29:03 +08:00
static void prvAddNewTaskToReadyList ( TCB_t * pxNewTCB )
2016-09-22 12:20:56 +08:00
{
2021-08-10 05:13:43 +08:00
/* Ensure interrupts don't access the task lists while the lists are being
* updated . */
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
uxCurrentNumberOfTasks + + ;
2022-08-01 17:29:03 +08:00
if ( uxCurrentNumberOfTasks = = ( UBaseType_t ) 1 )
2021-08-10 05:13:43 +08:00
{
2022-08-01 17:29:03 +08:00
/* This is the first task to be created so do the preliminary
* initialisation required . We will not recover if this call
* fails , but we will report the failure . */
prvInitialiseTaskLists ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
2021-08-10 05:13:43 +08:00
}
2022-08-01 17:29:03 +08:00
if ( ( pxCurrentTCB [ 0 ] = = NULL ) & & ( taskCAN_RUN_ON_CORE ( 0 , pxNewTCB - > xCoreID ) = = pdTRUE ) )
2021-08-10 05:13:43 +08:00
{
2022-08-01 17:29:03 +08:00
/* On core 0, there are no other tasks, or all the other tasks
* are in the suspended state - make this the current task . */
pxCurrentTCB [ 0 ] = pxNewTCB ;
}
2021-08-10 05:13:43 +08:00
2022-08-01 17:29:03 +08:00
# if ( configNUM_CORES > 1 )
else if ( ( pxCurrentTCB [ 1 ] = = NULL ) & & ( taskCAN_RUN_ON_CORE ( 1 , pxNewTCB - > xCoreID ) = = pdTRUE ) )
2021-08-10 05:13:43 +08:00
{
2022-08-01 17:29:03 +08:00
/* On core 1, there are no other tasks, or all the other tasks
* are in the suspended state - make this the current task . */
pxCurrentTCB [ 1 ] = pxNewTCB ;
2021-08-10 05:13:43 +08:00
}
2022-08-01 17:29:03 +08:00
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
else
{
/* If the scheduler is not already running, make this task the
* current task if it is the highest priority task to be created
* so far . */
if ( xSchedulerRunning = = pdFALSE )
{
2022-11-02 19:49:40 +08:00
if ( ( pxCurrentTCB [ 0 ] ! = NULL ) & &
( taskCAN_RUN_ON_CORE ( 0 , pxNewTCB - > xCoreID ) = = pdTRUE ) & &
( pxCurrentTCB [ 0 ] - > uxPriority < = pxNewTCB - > uxPriority ) )
2021-08-10 05:13:43 +08:00
{
2022-08-01 17:29:03 +08:00
pxCurrentTCB [ 0 ] = pxNewTCB ;
2021-08-10 05:13:43 +08:00
}
2022-08-01 17:29:03 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
else if ( ( pxCurrentTCB [ 1 ] ! = NULL ) & &
( taskCAN_RUN_ON_CORE ( 1 , pxNewTCB - > xCoreID ) = = pdTRUE ) & &
( pxCurrentTCB [ 1 ] - > uxPriority < = pxNewTCB - > uxPriority ) )
2022-08-01 17:29:03 +08:00
{
pxCurrentTCB [ 1 ] = pxNewTCB ;
}
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
uxTaskNumber + + ;
# if ( configUSE_TRACE_FACILITY == 1 )
2021-08-26 13:45:51 +08:00
{
/* Add a counter into the TCB for tracing only. */
pxNewTCB - > uxTCBNumber = uxTaskNumber ;
}
2021-08-10 05:13:43 +08:00
# endif /* configUSE_TRACE_FACILITY */
traceTASK_CREATE ( pxNewTCB ) ;
prvAddTaskToReadyList ( pxNewTCB ) ;
portSETUP_TCB ( pxNewTCB ) ;
2022-08-01 17:29:03 +08:00
if ( xSchedulerRunning ! = pdFALSE )
2021-08-10 05:13:43 +08:00
{
2022-08-01 17:29:03 +08:00
/* If the created task is of a higher priority than the current task
* then it should run now . */
if ( prvCheckForYield ( pxNewTCB , xPortGetCoreID ( ) , pdTRUE ) )
{
taskYIELD_IF_USING_PREEMPTION ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-01 17:29:03 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2022-07-29 19:31:24 +08:00
# if ( configNUM_CORES > 1 )
static BaseType_t prvCheckForYieldUsingPrioritySMP ( UBaseType_t uxTaskPriority ,
BaseType_t xTaskCoreID ,
BaseType_t xCurCoreID ,
BaseType_t xYieldEqualPriority )
{
if ( xYieldEqualPriority = = pdTRUE )
{
/* Increment the task priority to achieve the same affect as if( uxTaskPriority >= pxCurrentTCB->uxPriority ) */
uxTaskPriority + + ;
}
/* Indicate whether the current core needs to yield */
BaseType_t xYieldRequiredCurrentCore ;
/* If the target task can run on the current core, and has a higher priority than the current core, then yield the current core */
if ( ( ( xTaskCoreID = = xCurCoreID ) | | ( xTaskCoreID = = tskNO_AFFINITY ) ) & & ( uxTaskPriority > pxCurrentTCB [ xCurCoreID ] - > uxPriority ) )
{
/* Return true for the caller to yield the current core */
xYieldRequiredCurrentCore = pdTRUE ;
}
/* If the target task can run on the other core, and has a higher priority then the other core, and the other core has not suspended scheduling, the yield the other core */
2022-11-02 19:49:40 +08:00
else if ( ( ( xTaskCoreID = = ! xCurCoreID ) | | ( xTaskCoreID = = tskNO_AFFINITY ) ) & &
( uxTaskPriority > pxCurrentTCB [ ! xCurCoreID ] - > uxPriority ) & &
( uxSchedulerSuspended [ ! xCurCoreID ] = = ( UBaseType_t ) pdFALSE ) )
2022-07-29 19:31:24 +08:00
{
/* Signal the other core to yield */
vPortYieldOtherCore ( ! xCurCoreID ) ;
xYieldRequiredCurrentCore = pdFALSE ;
}
else
{
xYieldRequiredCurrentCore = pdFALSE ;
}
return xYieldRequiredCurrentCore ;
}
# endif /* configNUM_CORES > 1 */
/*-----------------------------------------------------------*/
2016-08-17 23:08:22 +08:00
# if ( INCLUDE_vTaskDelete == 1 )
2017-11-23 22:35:54 +08:00
2021-08-10 05:13:43 +08:00
void vTaskDelete ( TaskHandle_t xTaskToDelete )
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
2022-08-24 16:25:21 +08:00
BaseType_t xFreeNow ;
2021-08-10 05:13:43 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
2022-08-24 16:25:21 +08:00
BaseType_t xCurCoreID ;
# if ( configNUM_CORES > 1 )
xCurCoreID = xPortGetCoreID ( ) ;
# else
xCurCoreID = 0 ;
( void ) xCurCoreID ;
# endif
2021-08-10 05:13:43 +08:00
/* If null is passed in here then it is the calling task that is
2021-08-26 13:45:51 +08:00
* being deleted . */
2021-08-10 05:13:43 +08:00
pxTCB = prvGetTCBFromHandle ( xTaskToDelete ) ;
/* Remove task from the ready/delayed list. */
if ( uxListRemove ( & ( pxTCB - > xStateListItem ) ) = = ( UBaseType_t ) 0 )
{
taskRESET_READY_PRIORITY ( pxTCB - > uxPriority ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* Is the task waiting on an event also? */
if ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) ! = NULL )
{
( void ) uxListRemove ( & ( pxTCB - > xEventListItem ) ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* Increment the uxTaskNumber also so kernel aware debuggers can
2021-08-26 13:45:51 +08:00
* detect that the task lists need re - generating . This is done before
* portPRE_TASK_DELETE_HOOK ( ) as in the Windows port that macro will
* not return . */
2021-08-10 05:13:43 +08:00
uxTaskNumber + + ;
2022-12-14 21:07:48 +08:00
/* We cannot free the task immediately if it is currently running (on either core) */
xFreeNow = ( taskIS_CURRENTLY_RUNNING ( pxTCB ) ) ? pdFALSE : pdTRUE ;
2022-08-24 16:25:21 +08:00
if ( xFreeNow = = pdFALSE )
2021-08-10 05:13:43 +08:00
{
/* A task is deleting itself. This cannot complete within the
* task itself , as a context switch to another task is required .
* Place the task in the termination list . The idle task will
* check the termination list and free up any memory allocated by
* the scheduler for the TCB and stack of the deleted task . */
vListInsertEnd ( & xTasksWaitingTermination , & ( pxTCB - > xStateListItem ) ) ;
/* Increment the ucTasksDeleted variable so the idle task knows
* there is a task that has been deleted and that it should therefore
* check the xTasksWaitingTermination list . */
+ + uxDeletedTasksWaitingCleanUp ;
2022-08-24 16:25:21 +08:00
/* Call the delete hook before portPRE_TASK_DELETE_HOOK() as
* portPRE_TASK_DELETE_HOOK ( ) does not return in the Win32 port . */
traceTASK_DELETE ( pxTCB ) ;
2021-08-10 05:13:43 +08:00
/* The pre-delete hook is primarily for the Windows simulator,
* in which Windows specific clean up operations are performed ,
* after which it is not possible to yield away from this task -
* hence xYieldPending is used to latch that a context switch is
* required . */
2022-08-24 16:25:21 +08:00
portPRE_TASK_DELETE_HOOK ( pxTCB , & xYieldPending [ xCurCoreID ] ) ;
2021-08-10 05:13:43 +08:00
2022-08-24 16:25:21 +08:00
# if ( configNUM_CORES > 1 )
if ( taskIS_CURRENTLY_RUNNING_ON_CORE ( pxTCB , ! xCurCoreID ) )
{
2022-12-14 21:07:48 +08:00
/* SMP case of deleting a task currently running on a different core. Same issue
2022-11-02 19:49:40 +08:00
* as a task deleting itself , but we need to send a yield to this task now
* before we release xKernelLock .
*
* Specifically there is a case where the other core may already be spinning on
* xKernelLock waiting to go into a blocked state . A check is added in
* prvAddCurrentTaskToDelayedList ( ) to prevent it from removing itself from
* xTasksWaitingTermination list in this case ( instead it will immediately
* release xKernelLock again and be yielded before the FreeRTOS function
* returns . ) */
2022-08-24 16:25:21 +08:00
vPortYieldOtherCore ( ! xCurCoreID ) ;
}
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
}
else
{
- - uxCurrentNumberOfTasks ;
2022-08-24 16:25:21 +08:00
traceTASK_DELETE ( pxTCB ) ;
2021-08-10 05:13:43 +08:00
/* Reset the next expected unblock time in case it referred to
2021-08-26 13:45:51 +08:00
* the task that has just been deleted . */
2021-08-10 05:13:43 +08:00
prvResetNextTaskUnblockTime ( ) ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
2022-11-02 19:49:40 +08:00
if ( xFreeNow = = pdTRUE )
{
2022-11-08 17:48:48 +08:00
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS > 0 ) && ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 )
2021-08-10 05:13:43 +08:00
prvDeleteTLS ( pxTCB ) ;
# endif
prvDeleteTCB ( pxTCB ) ;
}
/* Force a reschedule if it is the currently running task that has just
* been deleted . */
if ( xSchedulerRunning ! = pdFALSE )
{
2022-08-24 16:25:21 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2022-11-02 19:49:40 +08:00
2022-08-24 16:25:21 +08:00
if ( taskIS_CURRENTLY_RUNNING_ON_CORE ( pxTCB , xPortGetCoreID ( ) ) )
2021-08-10 05:13:43 +08:00
{
configASSERT ( xTaskGetSchedulerState ( ) ! = taskSCHEDULER_SUSPENDED ) ;
portYIELD_WITHIN_API ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-11-02 19:49:40 +08:00
2022-08-24 16:25:21 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
}
}
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_vTaskDelete */
/*-----------------------------------------------------------*/
2021-09-02 18:01:08 +08:00
# if ( INCLUDE_xTaskDelayUntil == 1 )
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM
/* backward binary compatibility - remove later */
# undef vTaskDelayUntil
void vTaskDelayUntil ( TickType_t * const pxPreviousWakeTime ,
const TickType_t xTimeIncrement )
{
xTaskDelayUntil ( pxPreviousWakeTime , xTimeIncrement ) ;
}
# endif // ESP_PLATFORM
2021-09-02 18:01:08 +08:00
BaseType_t xTaskDelayUntil ( TickType_t * const pxPreviousWakeTime ,
const TickType_t xTimeIncrement )
2021-08-10 05:13:43 +08:00
{
TickType_t xTimeToWake ;
2021-09-02 18:01:08 +08:00
BaseType_t xAlreadyYielded , xShouldDelay = pdFALSE ;
2021-08-10 05:13:43 +08:00
configASSERT ( pxPreviousWakeTime ) ;
configASSERT ( ( xTimeIncrement > 0U ) ) ;
2021-11-24 14:34:40 +05:30
configASSERT ( xTaskGetSchedulerState ( ) ! = taskSCHEDULER_SUSPENDED ) ;
2021-08-10 05:13:43 +08:00
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_SUSPEND_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* Minor optimisation. The tick count cannot change in this
* block . */
const TickType_t xConstTickCount = xTickCount ;
/* Generate the tick time at which the task wants to wake. */
xTimeToWake = * pxPreviousWakeTime + xTimeIncrement ;
if ( xConstTickCount < * pxPreviousWakeTime )
{
/* The tick count has overflowed since this function was
* lasted called . In this case the only time we should ever
* actually delay is if the wake time has also overflowed ,
* and the wake time is greater than the tick time . When this
* is the case it is as if neither time had overflowed . */
if ( ( xTimeToWake < * pxPreviousWakeTime ) & & ( xTimeToWake > xConstTickCount ) )
{
xShouldDelay = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
/* The tick time has not overflowed. In this case we will
* delay if either the wake time has overflowed , and / or the
* tick time is less than the wake time . */
if ( ( xTimeToWake < * pxPreviousWakeTime ) | | ( xTimeToWake > xConstTickCount ) )
{
xShouldDelay = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
/* Update the wake time ready for the next call. */
* pxPreviousWakeTime = xTimeToWake ;
if ( xShouldDelay ! = pdFALSE )
{
2022-08-01 17:32:40 +08:00
traceTASK_DELAY_UNTIL ( xTimeToWake ) ;
2021-08-10 05:13:43 +08:00
/* prvAddCurrentTaskToDelayedList() needs the block time, not
* the time to wake , so subtract the current tick count . */
2022-08-01 19:06:44 +08:00
prvAddCurrentTaskToDelayedList ( xTimeToWake - xConstTickCount , pdFALSE ) ;
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-11-14 15:27:00 +08:00
xAlreadyYielded = prvEXIT_CRITICAL_OR_RESUME_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
2021-09-02 18:01:08 +08:00
/* Force a reschedule if xTaskResumeAll has not already done so, we may
* have put ourselves to sleep . */
if ( xAlreadyYielded = = pdFALSE )
{
portYIELD_WITHIN_API ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-09-05 17:31:22 +08:00
2021-09-02 18:01:08 +08:00
return xShouldDelay ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
2021-09-02 18:01:08 +08:00
# endif /* INCLUDE_xTaskDelayUntil */
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
# if ( INCLUDE_vTaskDelay == 1 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
void vTaskDelay ( const TickType_t xTicksToDelay )
{
2022-09-05 17:31:22 +08:00
BaseType_t xAlreadyYielded = pdFALSE ;
2021-08-10 05:13:43 +08:00
/* A delay time of zero just forces a reschedule. */
if ( xTicksToDelay > ( TickType_t ) 0U )
{
2021-11-24 14:34:40 +05:30
configASSERT ( xTaskGetSchedulerState ( ) ! = taskSCHEDULER_SUSPENDED ) ;
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_SUSPEND_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
traceTASK_DELAY ( ) ;
/* A task that is removed from the event list while the
* scheduler is suspended will not get placed in the ready
* list or removed from the blocked list until the scheduler
* is resumed .
*
* This task cannot be in an event list as it is the currently
* executing task . */
2022-08-01 19:06:44 +08:00
prvAddCurrentTaskToDelayedList ( xTicksToDelay , pdFALSE ) ;
2021-08-10 05:13:43 +08:00
}
2022-11-14 15:27:00 +08:00
xAlreadyYielded = prvEXIT_CRITICAL_OR_RESUME_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-09-05 17:31:22 +08:00
/* Force a reschedule if xTaskResumeAll has not already done so, we may
* have put ourselves to sleep . */
if ( xAlreadyYielded = = pdFALSE )
{
portYIELD_WITHIN_API ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-10 05:13:43 +08:00
}
2021-02-09 08:57:35 +08:00
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_vTaskDelay */
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
# if ( ( INCLUDE_eTaskGetState == 1 ) || ( configUSE_TRACE_FACILITY == 1 ) || ( INCLUDE_xTaskAbortDelay == 1 ) )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
eTaskState eTaskGetState ( TaskHandle_t xTask )
{
2021-08-26 13:45:51 +08:00
eTaskState eReturn ;
List_t const * pxStateList , * pxDelayedList , * pxOverflowedDelayedList ;
const TCB_t * const pxTCB = xTask ;
2021-08-10 05:13:43 +08:00
configASSERT ( pxTCB ) ;
2022-11-02 19:49:40 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ; /*Need critical section incase either core context switches in between */
2022-08-24 16:25:21 +08:00
if ( taskIS_CURRENTLY_RUNNING ( pxTCB ) )
2021-08-10 05:13:43 +08:00
{
/* The task calling this function is querying its own state. */
eReturn = eRunning ;
}
else
{
pxStateList = listLIST_ITEM_CONTAINER ( & ( pxTCB - > xStateListItem ) ) ;
pxDelayedList = pxDelayedTaskList ;
pxOverflowedDelayedList = pxOverflowDelayedTaskList ;
if ( ( pxStateList = = pxDelayedList ) | | ( pxStateList = = pxOverflowedDelayedList ) )
{
/* The task being queried is referenced from one of the Blocked
2021-08-26 13:45:51 +08:00
* lists . */
2021-08-10 05:13:43 +08:00
eReturn = eBlocked ;
}
# if ( INCLUDE_vTaskSuspend == 1 )
else if ( pxStateList = = & xSuspendedTaskList )
{
/* The task being queried is referenced from the suspended
* list . Is it genuinely suspended or is it blocked
* indefinitely ? */
if ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) = = NULL )
{
2021-09-06 12:46:14 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
BaseType_t x ;
2021-08-10 05:13:43 +08:00
/* The task does not appear on the event list item of
* and of the RTOS objects , but could still be in the
* blocked state if it is waiting on its notification
2021-09-06 12:46:14 +08:00
* rather than waiting on an object . If not , is
* suspended . */
eReturn = eSuspended ;
for ( x = 0 ; x < configTASK_NOTIFICATION_ARRAY_ENTRIES ; x + + )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
if ( pxTCB - > ucNotifyState [ x ] = = taskWAITING_NOTIFICATION )
{
eReturn = eBlocked ;
break ;
}
2021-08-10 05:13:43 +08:00
}
}
# else /* if ( configUSE_TASK_NOTIFICATIONS == 1 ) */
{
eReturn = eSuspended ;
}
# endif /* if ( configUSE_TASK_NOTIFICATIONS == 1 ) */
}
else
{
eReturn = eBlocked ;
}
}
# endif /* if ( INCLUDE_vTaskSuspend == 1 ) */
# if ( INCLUDE_vTaskDelete == 1 )
else if ( ( pxStateList = = & xTasksWaitingTermination ) | | ( pxStateList = = NULL ) )
{
/* The task being queried is referenced from the deleted
* tasks list , or it is not referenced from any lists at
* all . */
eReturn = eDeleted ;
}
# endif
else /*lint !e525 Negative indentation is intended to make use of pre-processor clearer. */
{
/* If the task is not in any other state, it must be in the
* Ready ( including pending ready ) state . */
eReturn = eReady ;
}
}
2022-11-02 19:49:40 +08:00
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return eReturn ;
} /*lint !e818 xTask cannot be a pointer to const because it is a typedef. */
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_eTaskGetState */
/*-----------------------------------------------------------*/
# if ( INCLUDE_uxTaskPriorityGet == 1 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
UBaseType_t uxTaskPriorityGet ( const TaskHandle_t xTask )
{
TCB_t const * pxTCB ;
UBaseType_t uxReturn ;
2016-08-17 23:08:22 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* If null is passed in here then it is the priority of the task
* that called uxTaskPriorityGet ( ) that is being queried . */
pxTCB = prvGetTCBFromHandle ( xTask ) ;
uxReturn = pxTCB - > uxPriority ;
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
return uxReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_uxTaskPriorityGet */
/*-----------------------------------------------------------*/
# if ( INCLUDE_uxTaskPriorityGet == 1 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
UBaseType_t uxTaskPriorityGetFromISR ( const TaskHandle_t xTask )
{
2021-08-26 13:45:51 +08:00
TCB_t const * pxTCB ;
2021-08-10 05:13:43 +08:00
UBaseType_t uxReturn ;
/* RTOS ports that support interrupt nesting have the concept of a
2021-08-26 13:45:51 +08:00
* maximum system call ( or maximum API call ) interrupt priority .
2021-08-10 05:13:43 +08:00
* Interrupts that are above the maximum system call priority are keep
* permanently enabled , even when the RTOS kernel is in a critical section ,
* but cannot make any calls to FreeRTOS API functions . If configASSERT ( )
* is defined in FreeRTOSConfig . h then
* portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) will result in an assertion
* failure if a FreeRTOS API function is called from an interrupt that has
* been assigned a priority above the configured maximum system call
* priority . Only FreeRTOS functions that end in FromISR can be called
* from interrupts that have been assigned a priority at or ( logically )
* below the maximum system call interrupt priority . FreeRTOS maintains a
* separate interrupt safe API to ensure interrupt entry is as fast and as
* simple as possible . More information ( albeit Cortex - M specific ) is
* provided on the following link :
2021-08-26 13:45:51 +08:00
* https : //www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */
2021-08-10 05:13:43 +08:00
portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) ;
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL_ISR ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* If null is passed in here then it is the priority of the calling
* task that is being queried . */
pxTCB = prvGetTCBFromHandle ( xTask ) ;
uxReturn = pxTCB - > uxPriority ;
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return uxReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_uxTaskPriorityGet */
/*-----------------------------------------------------------*/
# if ( INCLUDE_vTaskPrioritySet == 1 )
2021-08-10 05:13:43 +08:00
void vTaskPrioritySet ( TaskHandle_t xTask ,
UBaseType_t uxNewPriority )
{
TCB_t * pxTCB ;
UBaseType_t uxCurrentBasePriority , uxPriorityUsedOnEntry ;
BaseType_t xYieldRequired = pdFALSE ;
configASSERT ( ( uxNewPriority < configMAX_PRIORITIES ) ) ;
/* Ensure the new priority is valid. */
if ( uxNewPriority > = ( UBaseType_t ) configMAX_PRIORITIES )
{
uxNewPriority = ( UBaseType_t ) configMAX_PRIORITIES - ( UBaseType_t ) 1U ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* If null is passed in here then it is the priority of the calling
* task that is being changed . */
pxTCB = prvGetTCBFromHandle ( xTask ) ;
traceTASK_PRIORITY_SET ( pxTCB , uxNewPriority ) ;
# if ( configUSE_MUTEXES == 1 )
{
uxCurrentBasePriority = pxTCB - > uxBasePriority ;
}
# else
{
uxCurrentBasePriority = pxTCB - > uxPriority ;
}
# endif
if ( uxCurrentBasePriority ! = uxNewPriority )
{
/* The priority change may have readied a task of higher
* priority than the calling task . */
if ( uxNewPriority > uxCurrentBasePriority )
{
2022-08-24 16:25:21 +08:00
if ( ! taskIS_CURRENTLY_RUNNING ( pxTCB ) )
2021-08-10 05:13:43 +08:00
{
/* The priority of a task other than the currently
* running task is being raised . Is the priority being
* raised above that of the running task ? */
2022-11-02 19:49:40 +08:00
if ( prvCheckForYieldUsingPriority ( uxNewPriority , pxTCB - > xCoreID , xPortGetCoreID ( ) , pdTRUE ) )
2021-08-10 05:13:43 +08:00
{
xYieldRequired = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
/* The priority of the running task is being raised,
* but the running task must already be the highest
* priority task able to run so no yield is required . */
}
}
2022-08-24 16:25:21 +08:00
else if ( taskIS_CURRENTLY_RUNNING_ON_CORE ( pxTCB , 0 ) )
2021-08-10 05:13:43 +08:00
{
/* Setting the priority of the running task down means
* there may now be another task of higher priority that
* is ready to execute . */
xYieldRequired = pdTRUE ;
}
2022-11-02 19:49:40 +08:00
2022-08-24 16:25:21 +08:00
# if ( configNUM_CORES > 1 )
else if ( taskIS_CURRENTLY_RUNNING_ON_CORE ( pxTCB , 1 ) )
{
/* Setting the priority of the running task on the other
* core down means there may now be another task of
* higher priority that is ready to execute . */
vPortYieldOtherCore ( 1 ) ;
}
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
else
{
/* Setting the priority of any other task down does not
* require a yield as the running task must be above the
* new priority of the task being modified . */
}
/* Remember the ready list the task might be referenced from
* before its uxPriority member is changed so the
* taskRESET_READY_PRIORITY ( ) macro can function correctly . */
uxPriorityUsedOnEntry = pxTCB - > uxPriority ;
# if ( configUSE_MUTEXES == 1 )
{
2021-08-26 13:45:51 +08:00
/* Only change the priority being used if the task is not
* currently using an inherited priority . */
if ( pxTCB - > uxBasePriority = = pxTCB - > uxPriority )
{
pxTCB - > uxPriority = uxNewPriority ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* The base priority gets set whatever. */
pxTCB - > uxBasePriority = uxNewPriority ;
2021-08-10 05:13:43 +08:00
}
2021-08-26 13:45:51 +08:00
# else /* if ( configUSE_MUTEXES == 1 ) */
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
pxTCB - > uxPriority = uxNewPriority ;
2021-08-10 05:13:43 +08:00
}
2021-08-26 13:45:51 +08:00
# endif /* if ( configUSE_MUTEXES == 1 ) */
2021-08-10 05:13:43 +08:00
/* Only reset the event list item value if the value is not
* being used for anything else . */
if ( ( listGET_LIST_ITEM_VALUE ( & ( pxTCB - > xEventListItem ) ) & taskEVENT_LIST_ITEM_VALUE_IN_USE ) = = 0UL )
{
listSET_LIST_ITEM_VALUE ( & ( pxTCB - > xEventListItem ) , ( ( TickType_t ) configMAX_PRIORITIES - ( TickType_t ) uxNewPriority ) ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* If the task is in the blocked or suspended list we need do
* nothing more than change its priority variable . However , if
* the task is in a ready list it needs to be removed and placed
* in the list appropriate to its new priority . */
if ( listIS_CONTAINED_WITHIN ( & ( pxReadyTasksLists [ uxPriorityUsedOnEntry ] ) , & ( pxTCB - > xStateListItem ) ) ! = pdFALSE )
{
/* The task is currently in its ready list - remove before
2021-09-09 11:54:47 +08:00
* adding it to its new ready list . As we are in a critical
2021-08-10 05:13:43 +08:00
* section we can do this even if the scheduler is suspended . */
if ( uxListRemove ( & ( pxTCB - > xStateListItem ) ) = = ( UBaseType_t ) 0 )
{
/* It is known that the task is in its ready list so
* there is no need to check again and the port level
* reset macro can be called directly . */
portRESET_READY_PRIORITY ( uxPriorityUsedOnEntry , uxTopReadyPriority ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-26 13:45:51 +08:00
2021-08-10 05:13:43 +08:00
prvAddTaskToReadyList ( pxTCB ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
if ( xYieldRequired ! = pdFALSE )
{
taskYIELD_IF_USING_PREEMPTION ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* Remove compiler warning about unused variables when the port
* optimised task selection is not being used . */
( void ) uxPriorityUsedOnEntry ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_vTaskPrioritySet */
/*-----------------------------------------------------------*/
# if ( INCLUDE_vTaskSuspend == 1 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
void vTaskSuspend ( TaskHandle_t xTaskToSuspend )
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
2021-08-10 05:13:43 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* If null is passed in here then it is the running task that is
* being suspended . */
pxTCB = prvGetTCBFromHandle ( xTaskToSuspend ) ;
traceTASK_SUSPEND ( pxTCB ) ;
/* Remove task from the ready/delayed list and place in the
* suspended list . */
if ( uxListRemove ( & ( pxTCB - > xStateListItem ) ) = = ( UBaseType_t ) 0 )
{
taskRESET_READY_PRIORITY ( pxTCB - > uxPriority ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* Is the task waiting on an event also? */
if ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) ! = NULL )
{
( void ) uxListRemove ( & ( pxTCB - > xEventListItem ) ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
vListInsertEnd ( & xSuspendedTaskList , & ( pxTCB - > xStateListItem ) ) ;
2021-09-06 12:46:14 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
BaseType_t x ;
for ( x = 0 ; x < configTASK_NOTIFICATION_ARRAY_ENTRIES ; x + + )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
if ( pxTCB - > ucNotifyState [ x ] = = taskWAITING_NOTIFICATION )
{
/* The task was blocked to wait for a notification, but is
* now suspended , so no notification was received . */
pxTCB - > ucNotifyState [ x ] = taskNOT_WAITING_NOTIFICATION ;
}
2021-08-10 05:13:43 +08:00
}
}
2021-09-06 12:46:14 +08:00
# endif /* if ( configUSE_TASK_NOTIFICATIONS == 1 ) */
2021-08-10 05:13:43 +08:00
2022-08-24 16:25:21 +08:00
if ( xSchedulerRunning ! = pdFALSE )
2021-08-10 05:13:43 +08:00
{
2022-08-24 16:25:21 +08:00
/* Reset the next expected unblock time in case it referred to the
* task that is now in the Suspended state . */
2021-08-10 05:13:43 +08:00
prvResetNextTaskUnblockTime ( ) ;
}
2022-08-24 16:25:21 +08:00
else
2021-08-10 05:13:43 +08:00
{
2022-08-24 16:25:21 +08:00
mtCOVERAGE_TEST_MARKER ( ) ;
2022-09-12 19:53:09 +08:00
}
2022-08-24 16:25:21 +08:00
if ( taskIS_CURRENTLY_RUNNING_ON_CORE ( pxTCB , xPortGetCoreID ( ) ) )
2021-08-10 05:13:43 +08:00
{
2022-08-24 16:25:21 +08:00
if ( xSchedulerRunning ! = pdFALSE )
2021-08-10 05:13:43 +08:00
{
2022-08-24 16:25:21 +08:00
/* The current task has just been suspended. */
configASSERT ( uxSchedulerSuspended [ xPortGetCoreID ( ) ] = = 0 ) ;
portYIELD_WITHIN_API ( ) ;
2021-08-10 05:13:43 +08:00
}
else
{
2022-08-24 16:25:21 +08:00
/* The scheduler is not running, but the task that was pointed
* to by pxCurrentTCB has just been suspended and pxCurrentTCB
* must be adjusted to point to a different task . */
if ( listCURRENT_LIST_LENGTH ( & xSuspendedTaskList ) = = uxCurrentNumberOfTasks ) /*lint !e931 Right has no side effect, just volatile. */
{
/* No other tasks are ready, so set pxCurrentTCB back to
* NULL so when the next task is created pxCurrentTCB will
* be set to point to it no matter what its relative priority
* is . */
pxCurrentTCB [ xPortGetCoreID ( ) ] = NULL ;
}
else
{
vTaskSwitchContext ( ) ;
}
2021-08-10 05:13:43 +08:00
}
}
2022-11-02 19:49:40 +08:00
2022-08-24 16:25:21 +08:00
# if ( configNUM_CORES > 1 )
else if ( taskIS_CURRENTLY_RUNNING_ON_CORE ( pxTCB , ! xPortGetCoreID ( ) ) )
2021-08-10 05:13:43 +08:00
{
2022-08-24 16:25:21 +08:00
/* The other core's current task has just been suspended */
if ( xSchedulerRunning ! = pdFALSE )
{
vPortYieldOtherCore ( ! xPortGetCoreID ( ) ) ;
}
else
{
/* The scheduler is not running, but the task that was pointed
* to by pxCurrentTCB [ otherCore ] has just been suspended .
* We simply set the pxCurrentTCB [ otherCore ] to NULL for now .
* Todo : Update vTaskSwitchContext ( ) to be runnable on
* behalf of the other core . */
pxCurrentTCB [ ! xPortGetCoreID ( ) ] = NULL ;
}
2021-08-10 05:13:43 +08:00
}
2022-08-24 16:25:21 +08:00
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-24 16:25:21 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_vTaskSuspend */
/*-----------------------------------------------------------*/
# if ( INCLUDE_vTaskSuspend == 1 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
static BaseType_t prvTaskIsTaskSuspended ( const TaskHandle_t xTask )
{
BaseType_t xReturn = pdFALSE ;
const TCB_t * const pxTCB = xTask ;
/* Accesses xPendingReadyList so must be called from a critical
* section . */
/* It does not make sense to check if the calling task is suspended. */
configASSERT ( xTask ) ;
/* Is the task being resumed actually in the suspended list? */
if ( listIS_CONTAINED_WITHIN ( & xSuspendedTaskList , & ( pxTCB - > xStateListItem ) ) ! = pdFALSE )
{
/* Has the task already been resumed from within an ISR? */
2022-09-05 17:31:22 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
if ( ( listIS_CONTAINED_WITHIN ( & xPendingReadyList [ 0 ] , & ( pxTCB - > xEventListItem ) ) = = pdFALSE ) & &
( listIS_CONTAINED_WITHIN ( & xPendingReadyList [ 1 ] , & ( pxTCB - > xEventListItem ) ) = = pdFALSE ) )
2022-09-05 17:31:22 +08:00
# else
if ( listIS_CONTAINED_WITHIN ( & xPendingReadyList [ 0 ] , & ( pxTCB - > xEventListItem ) ) = = pdFALSE )
# endif
2021-08-10 05:13:43 +08:00
{
/* Is it in the suspended list because it is in the Suspended
* state , or because is is blocked with no timeout ? */
if ( listIS_CONTAINED_WITHIN ( NULL , & ( pxTCB - > xEventListItem ) ) ! = pdFALSE ) /*lint !e961. The cast is only redundant when NULL is used. */
{
xReturn = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
return xReturn ;
} /*lint !e818 xTask cannot be a pointer to const because it is a typedef. */
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_vTaskSuspend */
/*-----------------------------------------------------------*/
# if ( INCLUDE_vTaskSuspend == 1 )
2021-08-10 05:13:43 +08:00
void vTaskResume ( TaskHandle_t xTaskToResume )
{
TCB_t * const pxTCB = xTaskToResume ;
/* It does not make sense to resume the calling task. */
configASSERT ( xTaskToResume ) ;
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
2022-08-01 15:57:51 +08:00
/* The parameter cannot be NULL as it is impossible to resume the
* currently executing task . */
2022-08-24 16:25:21 +08:00
if ( ! taskIS_CURRENTLY_RUNNING ( pxTCB ) & & ( pxTCB ! = NULL ) )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
if ( prvTaskIsTaskSuspended ( pxTCB ) ! = pdFALSE )
{
traceTASK_RESUME ( pxTCB ) ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
/* The ready list can be accessed even if the scheduler is
* suspended because this is inside a critical section . */
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
prvAddTaskToReadyList ( pxTCB ) ;
2021-08-10 05:13:43 +08:00
2022-08-01 15:57:51 +08:00
/* A higher priority task may have just been resumed. */
if ( prvCheckForYield ( pxTCB , xPortGetCoreID ( ) , pdTRUE ) )
2021-08-26 13:45:51 +08:00
{
/* This yield may not cause the task just resumed to run,
* but will leave the lists in the correct state for the
* next yield . */
taskYIELD_IF_USING_PREEMPTION ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-01 15:57:51 +08:00
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-10 05:13:43 +08:00
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_vTaskSuspend */
/*-----------------------------------------------------------*/
# if ( ( INCLUDE_xTaskResumeFromISR == 1 ) && ( INCLUDE_vTaskSuspend == 1 ) )
2021-08-10 05:13:43 +08:00
BaseType_t xTaskResumeFromISR ( TaskHandle_t xTaskToResume )
{
BaseType_t xYieldRequired = pdFALSE ;
TCB_t * const pxTCB = xTaskToResume ;
configASSERT ( xTaskToResume ) ;
/* RTOS ports that support interrupt nesting have the concept of a
2021-08-26 13:45:51 +08:00
* maximum system call ( or maximum API call ) interrupt priority .
2021-08-10 05:13:43 +08:00
* Interrupts that are above the maximum system call priority are keep
* permanently enabled , even when the RTOS kernel is in a critical section ,
* but cannot make any calls to FreeRTOS API functions . If configASSERT ( )
* is defined in FreeRTOSConfig . h then
* portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) will result in an assertion
* failure if a FreeRTOS API function is called from an interrupt that has
* been assigned a priority above the configured maximum system call
* priority . Only FreeRTOS functions that end in FromISR can be called
* from interrupts that have been assigned a priority at or ( logically )
* below the maximum system call interrupt priority . FreeRTOS maintains a
* separate interrupt safe API to ensure interrupt entry is as fast and as
* simple as possible . More information ( albeit Cortex - M specific ) is
* provided on the following link :
2021-08-26 13:45:51 +08:00
* https : //www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */
2022-08-01 15:57:51 +08:00
portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) ;
2021-08-10 05:13:43 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL_ISR ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
if ( prvTaskIsTaskSuspended ( pxTCB ) ! = pdFALSE )
{
traceTASK_RESUME_FROM_ISR ( pxTCB ) ;
/* Check the ready lists can be accessed. */
2022-09-05 17:31:22 +08:00
/* Known issue IDF-5856. We also need to check if the other core is suspended */
2022-08-01 15:57:51 +08:00
if ( uxSchedulerSuspended [ xPortGetCoreID ( ) ] = = ( UBaseType_t ) pdFALSE )
2021-08-10 05:13:43 +08:00
{
2022-08-01 15:57:51 +08:00
/* Ready lists can be accessed so move the task from the
* suspended list to the ready list directly . */
if ( prvCheckForYield ( pxTCB , xPortGetCoreID ( ) , pdTRUE ) )
2021-08-10 05:13:43 +08:00
{
xYieldRequired = pdTRUE ;
2022-08-01 15:57:51 +08:00
/* Mark that a yield is pending in case the user is not
* using the return value to initiate a context switch
* from the ISR using portYIELD_FROM_ISR . */
xYieldPending [ xPortGetCoreID ( ) ] = pdTRUE ;
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-08-01 15:57:51 +08:00
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
prvAddTaskToReadyList ( pxTCB ) ;
2021-08-10 05:13:43 +08:00
}
else
{
/* The delayed or ready lists cannot be accessed so the task
* is held in the pending ready list until the scheduler is
* unsuspended . */
2022-08-01 15:57:51 +08:00
vListInsertEnd ( & ( xPendingReadyList [ xPortGetCoreID ( ) ] ) , & ( pxTCB - > xEventListItem ) ) ;
2021-08-10 05:13:43 +08:00
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return xYieldRequired ;
}
2016-08-17 23:08:22 +08:00
# endif /* ( ( INCLUDE_xTaskResumeFromISR == 1 ) && ( INCLUDE_vTaskSuspend == 1 ) ) */
/*-----------------------------------------------------------*/
void vTaskStartScheduler ( void )
{
2021-08-10 05:13:43 +08:00
BaseType_t xReturn ;
2016-08-17 23:08:22 +08:00
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM
/* Create an IDLE task for each core */
for ( BaseType_t xCoreID = 0 ; xCoreID < configNUM_CORES ; xCoreID + + )
# endif //ESP_PLATFORM
2021-10-16 01:08:46 +08:00
/* Add the idle task at the lowest priority. */
# if ( configSUPPORT_STATIC_ALLOCATION == 1 )
2021-08-10 05:13:43 +08:00
{
2021-10-16 01:08:46 +08:00
StaticTask_t * pxIdleTaskTCBBuffer = NULL ;
StackType_t * pxIdleTaskStackBuffer = NULL ;
uint32_t ulIdleTaskStackSize ;
2021-08-10 05:13:43 +08:00
/* The Idle task is created using user provided RAM - obtain the
2022-08-01 15:57:51 +08:00
* address of the RAM then create the idle task . */
2021-10-16 01:08:46 +08:00
vApplicationGetIdleTaskMemory ( & pxIdleTaskTCBBuffer , & pxIdleTaskStackBuffer , & ulIdleTaskStackSize ) ;
xIdleTaskHandle [ xCoreID ] = xTaskCreateStaticPinnedToCore ( prvIdleTask ,
2022-11-02 19:49:40 +08:00
configIDLE_TASK_NAME ,
ulIdleTaskStackSize ,
( void * ) NULL , /*lint !e961. The cast is not redundant for all compilers. */
portPRIVILEGE_BIT , /* In effect ( tskIDLE_PRIORITY | portPRIVILEGE_BIT ), but tskIDLE_PRIORITY is zero. */
pxIdleTaskStackBuffer ,
pxIdleTaskTCBBuffer ,
xCoreID ) ; /*lint !e961 MISRA exception, justified as it is not a redundant explicit cast to all supported compilers. */
2021-08-10 05:13:43 +08:00
2021-10-16 01:08:46 +08:00
if ( xIdleTaskHandle [ xCoreID ] ! = NULL )
2021-08-10 05:13:43 +08:00
{
xReturn = pdPASS ;
}
else
{
xReturn = pdFAIL ;
}
}
2021-08-26 13:45:51 +08:00
# else /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */
2021-08-10 05:13:43 +08:00
{
/* The Idle task is being created using dynamically allocated RAM. */
2022-11-02 19:49:40 +08:00
xReturn = xTaskCreatePinnedToCore ( prvIdleTask ,
configIDLE_TASK_NAME ,
2022-11-03 13:56:19 +01:00
configMINIMAL_STACK_SIZE ,
2022-11-02 19:49:40 +08:00
( void * ) NULL ,
portPRIVILEGE_BIT , /* In effect ( tskIDLE_PRIORITY | portPRIVILEGE_BIT ), but tskIDLE_PRIORITY is zero. */
& xIdleTaskHandle [ xCoreID ] ,
xCoreID ) ; /*lint !e961 MISRA exception, justified as it is not a redundant explicit cast to all supported compilers. */
2021-08-10 05:13:43 +08:00
2021-10-16 01:08:46 +08:00
if ( xIdleTaskHandle [ xCoreID ] ! = NULL )
2021-08-10 05:13:43 +08:00
{
xReturn = pdPASS ;
}
else
{
xReturn = pdFAIL ;
}
}
2021-08-26 13:45:51 +08:00
# endif /* configSUPPORT_STATIC_ALLOCATION */
2021-08-10 05:13:43 +08:00
# if ( configUSE_TIMERS == 1 )
{
2021-08-26 13:45:51 +08:00
if ( xReturn = = pdPASS )
{
xReturn = xTimerCreateTimerTask ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-10 05:13:43 +08:00
}
# endif /* configUSE_TIMERS */
if ( xReturn = = pdPASS )
{
/* freertos_tasks_c_additions_init() should only be called if the user
* definable macro FREERTOS_TASKS_C_ADDITIONS_INIT ( ) is defined , as that is
* the only macro called by the function . */
# ifdef FREERTOS_TASKS_C_ADDITIONS_INIT
{
freertos_tasks_c_additions_init ( ) ;
}
# endif
/* Interrupts are turned off here, to ensure a tick does not occur
* before or during the call to xPortStartScheduler ( ) . The stacks of
* the created tasks contain a status word with interrupts switched on
* so interrupts will automatically get re - enabled when the first task
* starts to run . */
portDISABLE_INTERRUPTS ( ) ;
# if ( configUSE_NEWLIB_REENTRANT == 1 )
2021-09-09 14:13:07 +08:00
{
/* Switch Newlib's _impure_ptr variable to point to the _reent
* structure specific to the task that will run first .
* See the third party link http : //www.nadler.com/embedded/newlibAndFreeRTOS.html
* for additional information . */
2022-11-02 19:49:40 +08:00
/* _impure_ptr = &( pxCurrentTCB[xPortGetCoreID()]->xNewLib_reent ); */
2021-09-09 14:13:07 +08:00
}
2021-08-10 05:13:43 +08:00
# endif /* configUSE_NEWLIB_REENTRANT */
xNextTaskUnblockTime = portMAX_DELAY ;
xSchedulerRunning = pdTRUE ;
xTickCount = ( TickType_t ) configINITIAL_TICK_COUNT ;
/* If configGENERATE_RUN_TIME_STATS is defined then the following
* macro must be defined to configure the timer / counter used to generate
* the run time counter time base . NOTE : If configGENERATE_RUN_TIME_STATS
* is set to 0 and the following line fails to build then ensure you do not
* have portCONFIGURE_TIMER_FOR_RUN_TIME_STATS ( ) defined in your
* FreeRTOSConfig . h file . */
portCONFIGURE_TIMER_FOR_RUN_TIME_STATS ( ) ;
traceTASK_SWITCHED_IN ( ) ;
/* Setting up the timer tick is hardware specific and thus in the
* portable interface . */
if ( xPortStartScheduler ( ) ! = pdFALSE )
{
/* Should not reach here as if the scheduler is running the
* function will not return . */
}
else
{
/* Should only reach here if a task calls xTaskEndScheduler(). */
}
}
else
{
/* This line will only be reached if the kernel could not be started,
* because there was not enough FreeRTOS heap to create the idle task
* or the timer task . */
configASSERT ( xReturn ! = errCOULD_NOT_ALLOCATE_REQUIRED_MEMORY ) ;
}
/* Prevent compiler warnings if INCLUDE_xTaskGetIdleTaskHandle is set to 0,
* meaning xIdleTaskHandle is not used anywhere else . */
2022-11-02 19:49:40 +08:00
( void ) xIdleTaskHandle [ 0 ] ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
void vTaskEndScheduler ( void )
{
2021-08-10 05:13:43 +08:00
/* Stop the scheduler interrupts and call the portable scheduler end
* routine so the original ISRs can be restored if necessary . The port
* layer must ensure interrupts enable bit is left in the correct state . */
portDISABLE_INTERRUPTS ( ) ;
xSchedulerRunning = pdFALSE ;
vPortEndScheduler ( ) ;
2016-08-17 23:08:22 +08:00
}
/*----------------------------------------------------------*/
void vTaskSuspendAll ( void )
{
2021-08-10 05:13:43 +08:00
/* A critical section is not required as the variable is of type
* BaseType_t . Please read Richard Barry ' s reply in the following link to a
* post in the FreeRTOS support forum before reporting this as a bug ! -
* https : //goo.gl/wu4acr */
2022-09-05 17:31:22 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
2022-09-05 17:31:22 +08:00
/* For SMP, although each core has their own uxSchedulerSuspended, we still
* need enter a critical section when accessing . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif
2022-07-11 17:00:47 +08:00
/* portSOFRWARE_BARRIER() is only implemented for emulated/simulated ports that
* do not otherwise exhibit real time behaviour . */
portSOFTWARE_BARRIER ( ) ;
/* The scheduler is suspended if uxSchedulerSuspended is non-zero. An increment
* is used to allow calls to vTaskSuspendAll ( ) to nest . */
2021-08-10 05:13:43 +08:00
+ + uxSchedulerSuspended [ xPortGetCoreID ( ) ] ;
2022-07-11 17:00:47 +08:00
/* Enforces ordering for ports and optimised compilers that may otherwise place
* the above increment elsewhere . */
portMEMORY_BARRIER ( ) ;
2022-09-05 17:31:22 +08:00
# if ( configNUM_CORES > 1 )
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif
2016-08-17 23:08:22 +08:00
}
/*----------------------------------------------------------*/
# if ( configUSE_TICKLESS_IDLE != 0 )
2022-08-24 16:25:21 +08:00
static TickType_t prvGetExpectedIdleTime ( void )
2022-09-12 19:53:09 +08:00
{
2022-08-24 16:25:21 +08:00
TickType_t xReturn ;
UBaseType_t uxHigherPriorityReadyTasks = pdFALSE ;
/* We need a critical section here as we are about to access kernel data structures */
taskENTER_CRITICAL ( & xKernelLock ) ;
/* uxHigherPriorityReadyTasks takes care of the case where
* configUSE_PREEMPTION is 0 , so there may be tasks above the idle priority
* task that are in the Ready state , even though the idle task is
* running . */
# if ( configUSE_PORT_OPTIMISED_TASK_SELECTION == 0 )
2021-08-10 05:13:43 +08:00
{
2022-08-24 16:25:21 +08:00
if ( uxTopReadyPriority > tskIDLE_PRIORITY )
{
uxHigherPriorityReadyTasks = pdTRUE ;
}
2021-08-10 05:13:43 +08:00
}
2022-08-24 16:25:21 +08:00
# else
2021-08-10 05:13:43 +08:00
{
2022-08-24 16:25:21 +08:00
const UBaseType_t uxLeastSignificantBit = ( UBaseType_t ) 0x01 ;
/* When port optimised task selection is used the uxTopReadyPriority
* variable is used as a bit map . If bits other than the least
* significant bit are set then there are tasks that have a priority
* above the idle priority that are in the Ready state . This takes
* care of the case where the co - operative scheduler is in use . */
if ( uxTopReadyPriority > uxLeastSignificantBit )
{
uxHigherPriorityReadyTasks = pdTRUE ;
}
2021-08-10 05:13:43 +08:00
}
2022-08-24 16:25:21 +08:00
# endif /* if ( configUSE_PORT_OPTIMISED_TASK_SELECTION == 0 ) */
2022-09-12 19:53:09 +08:00
2021-08-10 05:13:43 +08:00
if ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxPriority > tskIDLE_PRIORITY )
{
xReturn = 0 ;
}
2021-09-30 00:18:19 +08:00
else if ( listCURRENT_LIST_LENGTH ( & ( pxReadyTasksLists [ tskIDLE_PRIORITY ] ) ) > configNUM_CORES )
2021-08-10 05:13:43 +08:00
{
/* There are other idle priority tasks in the ready state. If
* time slicing is used then the very next tick interrupt must be
* processed . */
xReturn = 0 ;
}
2022-08-24 16:25:21 +08:00
else if ( uxHigherPriorityReadyTasks ! = pdFALSE )
{
/* There are tasks in the Ready state that have a priority above the
* idle priority . This path can only be reached if
* configUSE_PREEMPTION is 0. */
xReturn = 0 ;
}
2021-08-10 05:13:43 +08:00
else
{
xReturn = xNextTaskUnblockTime - xTickCount ;
}
2022-08-24 16:25:21 +08:00
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2020-11-10 18:40:01 +11:00
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TICKLESS_IDLE */
/*----------------------------------------------------------*/
BaseType_t xTaskResumeAll ( void )
{
2022-07-11 17:00:47 +08:00
TCB_t * pxTCB = NULL ;
2021-08-10 05:13:43 +08:00
BaseType_t xAlreadyYielded = pdFALSE ;
2022-07-11 17:00:47 +08:00
/* If uxSchedulerSuspended is zero then this function does not match a
* previous call to vTaskSuspendAll ( ) . */
configASSERT ( uxSchedulerSuspended [ xPortGetCoreID ( ) ] ) ;
2021-08-10 05:13:43 +08:00
/* It is possible that an ISR caused a task to be removed from an event
* list while the scheduler was suspended . If this was the case then the
* removed task will have been added to the xPendingReadyList . Once the
* scheduler has been resumed it is safe to move all the pending ready
* tasks from this list into their appropriate ready list . */
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
2022-07-11 17:00:47 +08:00
/* Minor optimization. Core ID can't change while inside a critical section */
BaseType_t xCoreID = xPortGetCoreID ( ) ;
2022-09-05 17:31:22 +08:00
2022-07-11 17:00:47 +08:00
- - uxSchedulerSuspended [ xCoreID ] ;
2021-08-10 05:13:43 +08:00
2022-07-11 17:00:47 +08:00
if ( uxSchedulerSuspended [ xCoreID ] = = ( UBaseType_t ) pdFALSE )
2021-08-10 05:13:43 +08:00
{
if ( uxCurrentNumberOfTasks > ( UBaseType_t ) 0U )
{
/* Move any readied tasks from the pending list into the
* appropriate ready list . */
2022-07-11 17:00:47 +08:00
while ( listLIST_IS_EMPTY ( & xPendingReadyList [ xCoreID ] ) = = pdFALSE )
2021-08-10 05:13:43 +08:00
{
2022-07-11 17:00:47 +08:00
pxTCB = listGET_OWNER_OF_HEAD_ENTRY ( ( & xPendingReadyList [ xCoreID ] ) ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
2021-08-10 05:13:43 +08:00
( void ) uxListRemove ( & ( pxTCB - > xEventListItem ) ) ;
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
prvAddTaskToReadyList ( pxTCB ) ;
/* If the moved task has a priority higher than the current
* task then a yield must be performed . */
2022-08-01 15:57:51 +08:00
if ( prvCheckForYield ( pxTCB , xPortGetCoreID ( ) , pdTRUE ) )
2021-08-10 05:13:43 +08:00
{
2022-07-11 17:00:47 +08:00
xYieldPending [ xCoreID ] = pdTRUE ;
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
if ( pxTCB ! = NULL )
{
/* A task was unblocked while the scheduler was suspended,
* which may have prevented the next unblock time from being
* re - calculated , in which case re - calculate it now . Mainly
* important for low power tickless implementations , where
* this can prevent an unnecessary exit from low power
* state . */
prvResetNextTaskUnblockTime ( ) ;
}
/* If any ticks occurred while the scheduler was suspended then
* they should be processed now . This ensures the tick count does
* not slip , and that any delayed tasks are resumed at the correct
* time . */
2022-11-02 19:49:40 +08:00
2022-07-11 17:00:47 +08:00
/* Core 0 is solely responsible for managing tick count, thus it
* must be the only core to unwind the pended ticks */
2022-11-02 19:49:40 +08:00
if ( xCoreID = = 0 )
2021-08-10 05:13:43 +08:00
{
2022-07-11 17:00:47 +08:00
TickType_t xPendedCounts = xPendedTicks ; /* Non-volatile copy. */
2021-08-10 05:13:43 +08:00
2022-07-11 17:00:47 +08:00
if ( xPendedCounts > ( TickType_t ) 0U )
2021-08-10 05:13:43 +08:00
{
2022-07-11 17:00:47 +08:00
do
{
if ( xTaskIncrementTick ( ) ! = pdFALSE )
{
xYieldPending [ xCoreID ] = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
- - xPendedCounts ;
} while ( xPendedCounts > ( TickType_t ) 0U ) ;
xPendedTicks = 0 ;
2021-08-10 05:13:43 +08:00
}
2022-07-11 17:00:47 +08:00
else
2021-08-10 05:13:43 +08:00
{
2022-07-11 17:00:47 +08:00
mtCOVERAGE_TEST_MARKER ( ) ;
2021-08-10 05:13:43 +08:00
}
}
2022-07-11 17:00:47 +08:00
if ( xYieldPending [ xCoreID ] ! = pdFALSE )
2021-08-10 05:13:43 +08:00
{
2021-09-07 10:30:30 +08:00
# if ( configUSE_PREEMPTION != 0 )
{
xAlreadyYielded = pdTRUE ;
}
2021-08-10 05:13:43 +08:00
# endif
taskYIELD_IF_USING_PREEMPTION ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return xAlreadyYielded ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
TickType_t xTaskGetTickCount ( void )
{
2021-08-10 05:13:43 +08:00
TickType_t xTicks ;
2019-11-28 15:27:47 -03:00
2022-09-05 17:31:22 +08:00
/* Critical section required if running on a 16 bit processor. */
portTICK_TYPE_ENTER_CRITICAL ( ) ;
{
xTicks = xTickCount ;
}
portTICK_TYPE_EXIT_CRITICAL ( ) ;
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
return xTicks ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
TickType_t xTaskGetTickCountFromISR ( void )
{
2021-08-10 05:13:43 +08:00
TickType_t xReturn ;
/* RTOS ports that support interrupt nesting have the concept of a maximum
* system call ( or maximum API call ) interrupt priority . Interrupts that are
* above the maximum system call priority are kept permanently enabled , even
* when the RTOS kernel is in a critical section , but cannot make any calls to
* FreeRTOS API functions . If configASSERT ( ) is defined in FreeRTOSConfig . h
* then portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) will result in an assertion
* failure if a FreeRTOS API function is called from an interrupt that has been
* assigned a priority above the configured maximum system call priority .
* Only FreeRTOS functions that end in FromISR can be called from interrupts
* that have been assigned a priority at or ( logically ) below the maximum
* system call interrupt priority . FreeRTOS maintains a separate interrupt
* safe API to ensure interrupt entry is as fast and as simple as possible .
* More information ( albeit Cortex - M specific ) is provided on the following
2021-08-26 13:45:51 +08:00
* link : https : //www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */
2021-08-10 05:13:43 +08:00
portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) ;
2022-09-05 17:31:22 +08:00
# if ( configNUM_CORES > 1 )
/* We need a critical section here as we are about to access kernel data structures */
taskENTER_CRITICAL_ISR ( & xKernelLock ) ;
# else
UBaseType_t uxSavedInterruptStatus ;
uxSavedInterruptStatus = portTICK_TYPE_SET_INTERRUPT_MASK_FROM_ISR ( ) ;
# endif
2021-08-10 05:13:43 +08:00
{
xReturn = xTickCount ;
}
2022-09-05 17:31:22 +08:00
# if ( configNUM_CORES > 1 )
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
# else
portTICK_TYPE_CLEAR_INTERRUPT_MASK_FROM_ISR ( uxSavedInterruptStatus ) ;
# endif
2021-08-10 05:13:43 +08:00
return xReturn ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
UBaseType_t uxTaskGetNumberOfTasks ( void )
{
2021-08-10 05:13:43 +08:00
/* A critical section is not required because the variables are of type
* BaseType_t . */
return uxCurrentNumberOfTasks ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2021-08-10 05:13:43 +08:00
char * pcTaskGetName ( TaskHandle_t xTaskToQuery ) /*lint !e971 Unqualified char types are allowed for strings and single characters only. */
2019-11-28 15:27:47 -03:00
{
2021-08-10 05:13:43 +08:00
TCB_t * pxTCB ;
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
/* If null is passed in here then the name of the calling task is being
* queried . */
pxTCB = prvGetTCBFromHandle ( xTaskToQuery ) ;
configASSERT ( pxTCB ) ;
return & ( pxTCB - > pcTaskName [ 0 ] ) ;
2019-11-28 15:27:47 -03:00
}
/*-----------------------------------------------------------*/
# if ( INCLUDE_xTaskGetHandle == 1 )
2021-08-10 05:13:43 +08:00
static TCB_t * prvSearchForNameWithinSingleList ( List_t * pxList ,
const char pcNameToQuery [ ] )
{
TCB_t * pxNextTCB , * pxFirstTCB , * pxReturn = NULL ;
UBaseType_t x ;
char cNextChar ;
BaseType_t xBreakLoop ;
/* This function is called with the scheduler suspended. */
if ( listCURRENT_LIST_LENGTH ( pxList ) > ( UBaseType_t ) 0 )
{
listGET_OWNER_OF_NEXT_ENTRY ( pxFirstTCB , pxList ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
do
{
listGET_OWNER_OF_NEXT_ENTRY ( pxNextTCB , pxList ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
/* Check each character in the name looking for a match or
* mismatch . */
xBreakLoop = pdFALSE ;
for ( x = ( UBaseType_t ) 0 ; x < ( UBaseType_t ) configMAX_TASK_NAME_LEN ; x + + )
{
cNextChar = pxNextTCB - > pcTaskName [ x ] ;
if ( cNextChar ! = pcNameToQuery [ x ] )
{
/* Characters didn't match. */
xBreakLoop = pdTRUE ;
}
else if ( cNextChar = = ( char ) 0x00 )
{
/* Both strings terminated, a match must have been
* found . */
pxReturn = pxNextTCB ;
xBreakLoop = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
if ( xBreakLoop ! = pdFALSE )
{
break ;
}
}
if ( pxReturn ! = NULL )
{
/* The handle has been found. */
break ;
}
} while ( pxNextTCB ! = pxFirstTCB ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
return pxReturn ;
}
2019-11-28 15:27:47 -03:00
# endif /* INCLUDE_xTaskGetHandle */
/*-----------------------------------------------------------*/
# if ( INCLUDE_xTaskGetHandle == 1 )
2021-08-10 05:13:43 +08:00
TaskHandle_t xTaskGetHandle ( const char * pcNameToQuery ) /*lint !e971 Unqualified char types are allowed for strings and single characters only. */
{
UBaseType_t uxQueue = configMAX_PRIORITIES ;
TCB_t * pxTCB ;
/* Task names will be truncated to configMAX_TASK_NAME_LEN - 1 bytes. */
configASSERT ( strlen ( pcNameToQuery ) < configMAX_TASK_NAME_LEN ) ;
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_SUSPEND_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* Search the ready lists. */
do
{
uxQueue - - ;
pxTCB = prvSearchForNameWithinSingleList ( ( List_t * ) & ( pxReadyTasksLists [ uxQueue ] ) , pcNameToQuery ) ;
if ( pxTCB ! = NULL )
{
/* Found the handle. */
break ;
}
} while ( uxQueue > ( UBaseType_t ) tskIDLE_PRIORITY ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
/* Search the delayed lists. */
if ( pxTCB = = NULL )
{
pxTCB = prvSearchForNameWithinSingleList ( ( List_t * ) pxDelayedTaskList , pcNameToQuery ) ;
}
if ( pxTCB = = NULL )
{
pxTCB = prvSearchForNameWithinSingleList ( ( List_t * ) pxOverflowDelayedTaskList , pcNameToQuery ) ;
}
# if ( INCLUDE_vTaskSuspend == 1 )
{
if ( pxTCB = = NULL )
{
/* Search the suspended list. */
pxTCB = prvSearchForNameWithinSingleList ( & xSuspendedTaskList , pcNameToQuery ) ;
}
}
# endif
# if ( INCLUDE_vTaskDelete == 1 )
{
if ( pxTCB = = NULL )
{
/* Search the deleted list. */
pxTCB = prvSearchForNameWithinSingleList ( & xTasksWaitingTermination , pcNameToQuery ) ;
}
}
# endif
}
2022-11-14 15:27:00 +08:00
( void ) prvEXIT_CRITICAL_OR_RESUME_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return pxTCB ;
}
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
# endif /* INCLUDE_xTaskGetHandle */
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
# if ( configUSE_TRACE_FACILITY == 1 )
2021-08-10 05:13:43 +08:00
UBaseType_t uxTaskGetSystemState ( TaskStatus_t * const pxTaskStatusArray ,
const UBaseType_t uxArraySize ,
uint32_t * const pulTotalRunTime )
{
UBaseType_t uxTask = 0 , uxQueue = configMAX_PRIORITIES ;
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_SUSPEND_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* Is there a space in the array for each task in the system? */
if ( uxArraySize > = uxCurrentNumberOfTasks )
{
/* Fill in an TaskStatus_t structure with information on each
* task in the Ready state . */
do
{
uxQueue - - ;
uxTask + = prvListTasksWithinSingleList ( & ( pxTaskStatusArray [ uxTask ] ) , & ( pxReadyTasksLists [ uxQueue ] ) , eReady ) ;
} while ( uxQueue > ( UBaseType_t ) tskIDLE_PRIORITY ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
/* Fill in an TaskStatus_t structure with information on each
* task in the Blocked state . */
uxTask + = prvListTasksWithinSingleList ( & ( pxTaskStatusArray [ uxTask ] ) , ( List_t * ) pxDelayedTaskList , eBlocked ) ;
uxTask + = prvListTasksWithinSingleList ( & ( pxTaskStatusArray [ uxTask ] ) , ( List_t * ) pxOverflowDelayedTaskList , eBlocked ) ;
# if ( INCLUDE_vTaskDelete == 1 )
{
/* Fill in an TaskStatus_t structure with information on
* each task that has been deleted but not yet cleaned up . */
uxTask + = prvListTasksWithinSingleList ( & ( pxTaskStatusArray [ uxTask ] ) , & xTasksWaitingTermination , eDeleted ) ;
}
# endif
# if ( INCLUDE_vTaskSuspend == 1 )
{
/* Fill in an TaskStatus_t structure with information on
* each task in the Suspended state . */
uxTask + = prvListTasksWithinSingleList ( & ( pxTaskStatusArray [ uxTask ] ) , & xSuspendedTaskList , eSuspended ) ;
}
# endif
# if ( configGENERATE_RUN_TIME_STATS == 1 )
{
if ( pulTotalRunTime ! = NULL )
{
# ifdef portALT_GET_RUN_TIME_COUNTER_VALUE
portALT_GET_RUN_TIME_COUNTER_VALUE ( ( * pulTotalRunTime ) ) ;
# else
* pulTotalRunTime = portGET_RUN_TIME_COUNTER_VALUE ( ) ;
# endif
}
}
# else /* if ( configGENERATE_RUN_TIME_STATS == 1 ) */
{
if ( pulTotalRunTime ! = NULL )
{
* pulTotalRunTime = 0 ;
}
}
# endif /* if ( configGENERATE_RUN_TIME_STATS == 1 ) */
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-11-14 15:27:00 +08:00
( void ) prvEXIT_CRITICAL_OR_RESUME_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return uxTask ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TRACE_FACILITY */
/*----------------------------------------------------------*/
# if ( INCLUDE_xTaskGetIdleTaskHandle == 1 )
2021-08-10 05:13:43 +08:00
TaskHandle_t xTaskGetIdleTaskHandle ( void )
{
/* If xTaskGetIdleTaskHandle() is called before the scheduler has been
* started , then xIdleTaskHandle will be NULL . */
2022-11-02 19:49:40 +08:00
configASSERT ( ( xIdleTaskHandle [ xPortGetCoreID ( ) ] ! = NULL ) ) ;
return xIdleTaskHandle [ xPortGetCoreID ( ) ] ;
2021-08-10 05:13:43 +08:00
}
TaskHandle_t xTaskGetIdleTaskHandleForCPU ( UBaseType_t cpuid )
{
2021-09-30 00:18:19 +08:00
configASSERT ( cpuid < configNUM_CORES ) ;
2022-11-02 19:49:40 +08:00
configASSERT ( ( xIdleTaskHandle [ cpuid ] ! = NULL ) ) ;
return xIdleTaskHandle [ cpuid ] ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_xTaskGetIdleTaskHandle */
/*----------------------------------------------------------*/
/* This conditional compilation should use inequality to 0, not equality to 1.
2021-08-10 05:13:43 +08:00
* This is to ensure vTaskStepTick ( ) is available when user defined low power mode
* implementations require configUSE_TICKLESS_IDLE to be set to a value other than
* 1. */
2016-08-17 23:08:22 +08:00
# if ( configUSE_TICKLESS_IDLE != 0 )
2021-08-10 05:13:43 +08:00
void vTaskStepTick ( const TickType_t xTicksToJump )
{
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Although this is called with the scheduler suspended. For SMP, we
* still need to take the kernel lock to access xTickCount . */
2022-11-02 19:49:40 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2022-11-14 15:27:00 +08:00
# endif /* configNUM_CORES > 1 */
2022-11-02 19:49:40 +08:00
2021-08-10 05:13:43 +08:00
/* Correct the tick count value after a period during which the tick
* was suppressed . Note this does * not * call the tick hook function for
* each stepped tick . */
configASSERT ( ( xTickCount + xTicksToJump ) < = xNextTaskUnblockTime ) ;
xTickCount + = xTicksToJump ;
traceINCREASE_TICK_COUNT ( xTicksToJump ) ;
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
2022-11-02 19:49:40 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2022-11-14 15:27:00 +08:00
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TICKLESS_IDLE */
/*----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
BaseType_t xTaskCatchUpTicks ( TickType_t xTicksToCatchUp )
{
2021-09-02 15:19:49 +08:00
BaseType_t xYieldOccurred ;
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
/* Must not be called with the scheduler suspended as the implementation
* relies on xPendedTicks being wound down to 0 in xTaskResumeAll ( ) . */
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM
configASSERT ( xTaskGetSchedulerState ( ) ! = taskSCHEDULER_SUSPENDED ) ;
# else
configASSERT ( uxSchedulerSuspended = = 0 ) ;
# endif // ESP_PLATFORM
2019-11-28 15:27:47 -03:00
2022-07-11 18:13:50 +08:00
/* Use xPendedTicks to mimic xTicksToCatchUp number of ticks occurring when
2021-08-10 05:13:43 +08:00
* the scheduler is suspended so the ticks are executed in xTaskResumeAll ( ) . */
2021-09-02 15:19:49 +08:00
vTaskSuspendAll ( ) ;
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
2022-11-14 15:27:00 +08:00
/* Although the scheduler is suspended. For SMP, we still need to take
* the kernel lock to access xPendedTicks . */
2022-11-02 19:49:40 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2022-11-14 15:27:00 +08:00
# endif /* configNUM_CORES > 1 */
xPendedTicks + = xTicksToCatchUp ;
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
2022-11-02 19:49:40 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2022-11-14 15:27:00 +08:00
# endif /* configNUM_CORES > 1 */
2021-09-02 15:19:49 +08:00
xYieldOccurred = xTaskResumeAll ( ) ;
return xYieldOccurred ;
2019-11-28 15:27:47 -03:00
}
/*----------------------------------------------------------*/
# if ( INCLUDE_xTaskAbortDelay == 1 )
2021-08-10 05:13:43 +08:00
BaseType_t xTaskAbortDelay ( TaskHandle_t xTask )
{
TCB_t * pxTCB = xTask ;
BaseType_t xReturn ;
configASSERT ( pxTCB ) ;
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_SUSPEND_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* A task can only be prematurely removed from the Blocked state if
* it is actually in the Blocked state . */
if ( eTaskGetState ( xTask ) = = eBlocked )
{
xReturn = pdPASS ;
/* Remove the reference to the task from the blocked list. An
* interrupt won ' t touch the xStateListItem because the
* scheduler is suspended . */
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
/* Is the task waiting on an event also? If so remove it from
* the event list too . Interrupts can touch the event list item ,
* even though the scheduler is suspended , so a critical section
* is used . */
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
if ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) ! = NULL )
{
( void ) uxListRemove ( & ( pxTCB - > xEventListItem ) ) ;
2021-09-07 10:30:30 +08:00
/* This lets the task know it was forcibly removed from the
* blocked state so it should not re - evaluate its block time and
* then block again . */
2021-08-10 05:13:43 +08:00
pxTCB - > ucDelayAborted = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
/* Place the unblocked task into the appropriate ready list. */
prvAddTaskToReadyList ( pxTCB ) ;
/* A task being unblocked cannot cause an immediate context
* switch if preemption is turned off . */
# if ( configUSE_PREEMPTION == 1 )
{
/* Preemption is on, but a context switch should only be
2021-08-26 13:45:51 +08:00
* performed if the unblocked task has a priority that is
* equal to or higher than the currently executing task . */
2022-08-01 15:57:51 +08:00
if ( prvCheckForYield ( pxTCB , xPortGetCoreID ( ) , pdFALSE ) )
2021-08-10 05:13:43 +08:00
{
/* Pend the yield to be performed when the scheduler
* is unsuspended . */
2022-08-01 15:57:51 +08:00
xYieldPending [ xPortGetCoreID ( ) ] = pdTRUE ;
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
# endif /* configUSE_PREEMPTION */
}
else
{
xReturn = pdFAIL ;
}
}
2022-11-14 15:27:00 +08:00
( void ) prvEXIT_CRITICAL_OR_RESUME_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2019-11-28 15:27:47 -03:00
# endif /* INCLUDE_xTaskAbortDelay */
/*----------------------------------------------------------*/
2016-08-17 23:08:22 +08:00
BaseType_t xTaskIncrementTick ( void )
{
2022-11-14 15:28:58 +08:00
# if ( configNUM_CORES > 1 )
/* Only Core 0 should ever call this function. */
configASSERT ( xPortGetCoreID ( ) = = 0 ) ;
# endif /* ( configNUM_CORES > 1 ) */
2021-08-10 05:13:43 +08:00
TCB_t * pxTCB ;
TickType_t xItemValue ;
BaseType_t xSwitchRequired = pdFALSE ;
/* Called by the portable layer each time a tick interrupt occurs.
2021-08-26 13:45:51 +08:00
* Increments the tick then checks to see if the new tick value will cause any
* tasks to be unblocked . */
2021-08-10 05:13:43 +08:00
traceTASK_INCREMENT_TICK ( xTickCount ) ;
2021-09-09 14:13:07 +08:00
2022-11-14 15:28:58 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
2022-11-14 15:28:58 +08:00
/* For SMP, we need to take the kernel lock here as we are about to
* access kernel data structures ( unlike single core which calls this
* function with interrupts disabled ) . */
2022-11-02 19:49:40 +08:00
taskENTER_CRITICAL_ISR ( & xKernelLock ) ;
2022-11-14 15:28:58 +08:00
# endif /* ( configNUM_CORES > 1 ) */
2022-07-09 18:47:14 +08:00
if ( uxSchedulerSuspended [ 0 ] = = ( UBaseType_t ) pdFALSE )
2021-08-10 05:13:43 +08:00
{
/* Minor optimisation. The tick count cannot change in this
2021-08-26 13:45:51 +08:00
* block . */
2021-08-10 05:13:43 +08:00
const TickType_t xConstTickCount = xTickCount + ( TickType_t ) 1 ;
/* Increment the RTOS tick, switching the delayed and overflowed
2021-08-26 13:45:51 +08:00
* delayed lists if it wraps to 0. */
2021-08-10 05:13:43 +08:00
xTickCount = xConstTickCount ;
if ( xConstTickCount = = ( TickType_t ) 0U ) /*lint !e774 'if' does not always evaluate to false as it is looking for an overflow. */
{
taskSWITCH_DELAYED_LISTS ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* See if this tick has made a timeout expire. Tasks are stored in
* the queue in the order of their wake time - meaning once one task
* has been found whose block time has not expired there is no need to
* look any further down the list . */
if ( xConstTickCount > = xNextTaskUnblockTime )
{
for ( ; ; )
{
if ( listLIST_IS_EMPTY ( pxDelayedTaskList ) ! = pdFALSE )
{
/* The delayed list is empty. Set xNextTaskUnblockTime
* to the maximum possible value so it is extremely
* unlikely that the
* if ( xTickCount > = xNextTaskUnblockTime ) test will pass
* next time through . */
xNextTaskUnblockTime = portMAX_DELAY ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
break ;
}
else
{
/* The delayed list is not empty, get the value of the
* item at the head of the delayed list . This is the time
* at which the task at the head of the delayed list must
* be removed from the Blocked state . */
pxTCB = listGET_OWNER_OF_HEAD_ENTRY ( pxDelayedTaskList ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
xItemValue = listGET_LIST_ITEM_VALUE ( & ( pxTCB - > xStateListItem ) ) ;
if ( xConstTickCount < xItemValue )
{
/* It is not time to unblock this item yet, but the
* item value is the time at which the task at the head
* of the blocked list must be removed from the Blocked
* state - so record the item value in
* xNextTaskUnblockTime . */
xNextTaskUnblockTime = xItemValue ;
2022-07-09 18:47:14 +08:00
break ; /*lint !e9011 Code structure here is deedmed easier to understand with multiple breaks. */
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* It is time to remove the item from the Blocked state. */
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
/* Is the task waiting on an event also? If so remove
* it from the event list . */
if ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) ! = NULL )
{
( void ) uxListRemove ( & ( pxTCB - > xEventListItem ) ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* Place the unblocked task into the appropriate ready
* list . */
prvAddTaskToReadyList ( pxTCB ) ;
/* A task being unblocked cannot cause an immediate
* context switch if preemption is turned off . */
# if ( configUSE_PREEMPTION == 1 )
{
/* Preemption is on, but a context switch should
* only be performed if the unblocked task has a
* priority that is equal to or higher than the
2022-11-14 15:28:58 +08:00
* currently executing task .
*
* For SMP , since this function is only run on core
* 0 , only need to switch contexts if the unblocked
* task can run on core 0. */
if ( ( taskCAN_RUN_ON_CORE ( 0 , pxTCB - > xCoreID ) = = pdTRUE ) & & ( pxTCB - > uxPriority > = pxCurrentTCB [ 0 ] - > uxPriority ) )
2021-08-10 05:13:43 +08:00
{
xSwitchRequired = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
# endif /* configUSE_PREEMPTION */
}
}
}
/* Tasks of equal priority to the currently running task will share
* processing time ( time slice ) if preemption is on , and the application
* writer has not explicitly turned time slicing off . */
# if ( ( configUSE_PREEMPTION == 1 ) && ( configUSE_TIME_SLICING == 1 ) )
{
2022-07-09 18:47:14 +08:00
if ( listCURRENT_LIST_LENGTH ( & ( pxReadyTasksLists [ pxCurrentTCB [ 0 ] - > uxPriority ] ) ) > ( UBaseType_t ) 1 )
2021-08-10 05:13:43 +08:00
{
xSwitchRequired = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
# endif /* ( ( configUSE_PREEMPTION == 1 ) && ( configUSE_TIME_SLICING == 1 ) ) */
2022-07-09 18:47:14 +08:00
2022-11-14 15:28:58 +08:00
# if ( configUSE_TICK_HOOK == 1 )
TickType_t xPendedTicksTemp = xPendedTicks ; /* Non-volatile copy. */
# endif /* configUSE_TICK_HOOK */
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock as we have finished
* accessing the kernel data structures . */
2022-11-02 19:49:40 +08:00
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
2022-11-14 15:28:58 +08:00
# endif /* ( configNUM_CORES > 1 ) */
2022-07-09 18:47:14 +08:00
# if ( configUSE_TICK_HOOK == 1 )
{
/* Guard against the tick hook being called when the pended tick
* count is being unwound ( when the scheduler is being unlocked ) . */
2022-11-14 15:28:58 +08:00
if ( xPendedTicksTemp = = ( TickType_t ) 0 )
2022-07-09 18:47:14 +08:00
{
vApplicationTickHook ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
# endif /* configUSE_TICK_HOOK */
# if ( configUSE_PREEMPTION == 1 )
{
if ( xYieldPending [ 0 ] ! = pdFALSE )
{
xSwitchRequired = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
# endif /* configUSE_PREEMPTION */
2021-08-10 05:13:43 +08:00
}
else
{
+ + xPendedTicks ;
2022-11-14 15:28:58 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock as we have finished
* accessing the kernel data structures . */
2022-11-02 19:49:40 +08:00
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
2022-11-14 15:28:58 +08:00
# endif /* ( configNUM_CORES > 1 ) */
2022-07-09 18:47:14 +08:00
/* The tick hook gets called at regular intervals, even if the
* scheduler is locked . */
# if ( configUSE_TICK_HOOK == 1 )
{
vApplicationTickHook ( ) ;
}
# endif
2021-08-10 05:13:43 +08:00
}
2022-07-09 18:47:14 +08:00
return xSwitchRequired ;
}
2022-11-08 17:35:06 +08:00
# if ( configNUM_CORES > 1 )
BaseType_t xTaskIncrementTickOtherCores ( void )
{
/* Minor optimization. This function can never switch cores mid
* execution */
BaseType_t xCoreID = xPortGetCoreID ( ) ;
BaseType_t xSwitchRequired = pdFALSE ;
2022-07-09 18:47:14 +08:00
2022-11-08 17:35:06 +08:00
/* This function should never be called by Core 0. */
configASSERT ( xCoreID ! = 0 ) ;
2022-07-09 18:47:14 +08:00
2022-11-08 17:35:06 +08:00
/* Called by the portable layer each time a tick interrupt occurs.
* Increments the tick then checks to see if the new tick value will cause any
* tasks to be unblocked . */
traceTASK_INCREMENT_TICK ( xTickCount ) ;
2022-07-09 18:47:14 +08:00
2022-11-08 17:35:06 +08:00
if ( uxSchedulerSuspended [ xCoreID ] = = ( UBaseType_t ) pdFALSE )
{
2022-11-14 15:28:58 +08:00
/* We need take the kernel lock here as we are about to access
* kernel data structures . */
2022-11-08 17:35:06 +08:00
taskENTER_CRITICAL_ISR ( & xKernelLock ) ;
/* A task being unblocked cannot cause an immediate context switch
* if preemption is turned off . */
# if ( configUSE_PREEMPTION == 1 )
{
/* Check if core 0 calling xTaskIncrementTick() has
* unblocked a task that can be run . */
if ( uxTopReadyPriority > pxCurrentTCB [ xCoreID ] - > uxPriority )
2022-07-09 18:47:14 +08:00
{
2022-11-08 17:35:06 +08:00
xSwitchRequired = pdTRUE ;
2022-07-09 18:47:14 +08:00
}
2022-11-08 17:35:06 +08:00
else
2022-07-09 18:47:14 +08:00
{
2022-11-08 17:35:06 +08:00
mtCOVERAGE_TEST_MARKER ( ) ;
2022-07-09 18:47:14 +08:00
}
2022-11-08 17:35:06 +08:00
}
# endif /* if ( configUSE_PREEMPTION == 1 ) */
2022-07-09 18:47:14 +08:00
2022-11-08 17:35:06 +08:00
/* Tasks of equal priority to the currently running task will share
* processing time ( time slice ) if preemption is on , and the application
* writer has not explicitly turned time slicing off . */
# if ( ( configUSE_PREEMPTION == 1 ) && ( configUSE_TIME_SLICING == 1 ) )
{
if ( listCURRENT_LIST_LENGTH ( & ( pxReadyTasksLists [ pxCurrentTCB [ xCoreID ] - > uxPriority ] ) ) > ( UBaseType_t ) 1 )
2022-07-09 18:47:14 +08:00
{
2022-11-08 17:35:06 +08:00
xSwitchRequired = pdTRUE ;
2022-07-09 18:47:14 +08:00
}
2022-11-08 17:35:06 +08:00
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
# endif /* ( ( configUSE_PREEMPTION == 1 ) && ( configUSE_TIME_SLICING == 1 ) ) */
2022-07-09 18:47:14 +08:00
2022-11-14 15:28:58 +08:00
/* Release the previously taken kernel lock as we have finished
* accessing the kernel data structures . */
2022-11-08 17:35:06 +08:00
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
# if ( configUSE_PREEMPTION == 1 )
2022-11-02 19:49:40 +08:00
{
2022-11-08 17:35:06 +08:00
if ( xYieldPending [ xCoreID ] ! = pdFALSE )
{
xSwitchRequired = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-11-02 19:49:40 +08:00
}
2022-11-08 17:35:06 +08:00
# endif /* configUSE_PREEMPTION */
2022-11-02 19:49:40 +08:00
}
2022-11-08 17:35:06 +08:00
# if ( configUSE_TICK_HOOK == 1 )
{
vApplicationTickHook ( ) ;
}
# endif
return xSwitchRequired ;
}
# endif /* ( configNUM_CORES > 1 ) */
2021-08-10 05:13:43 +08:00
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
# if ( configUSE_APPLICATION_TASK_TAG == 1 )
2022-11-02 19:49:40 +08:00
void vTaskSetApplicationTaskTag ( TaskHandle_t xTask ,
TaskHookFunction_t pxHookFunction )
2021-08-10 05:13:43 +08:00
{
2021-09-07 10:30:30 +08:00
TCB_t * xTCB ;
2021-08-10 05:13:43 +08:00
/* If xTask is NULL then it is the task hook of the calling task that is
2021-08-26 13:45:51 +08:00
* getting set . */
2021-08-10 05:13:43 +08:00
if ( xTask = = NULL )
{
2022-11-02 19:49:40 +08:00
xTCB = ( TCB_t * ) pxCurrentTCB [ xPortGetCoreID ( ) ] ;
2021-08-10 05:13:43 +08:00
}
else
{
xTCB = xTask ;
}
/* Save the hook function in the TCB. A critical section is required as
2021-08-26 13:45:51 +08:00
* the value can be accessed from an interrupt . */
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
xTCB - > pxTaskTag = pxHookFunction ;
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_APPLICATION_TASK_TAG */
/*-----------------------------------------------------------*/
# if ( configUSE_APPLICATION_TASK_TAG == 1 )
2021-08-10 05:13:43 +08:00
TaskHookFunction_t xTaskGetApplicationTaskTag ( TaskHandle_t xTask )
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
TaskHookFunction_t xReturn ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* If xTask is NULL then set the calling task's hook. */
pxTCB = prvGetTCBFromHandle ( xTask ) ;
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
/* Save the hook function in the TCB. A critical section is required as
2021-08-26 13:45:51 +08:00
* the value can be accessed from an interrupt . */
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
xReturn = pxTCB - > pxTaskTag ;
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2019-11-28 15:27:47 -03:00
# endif /* configUSE_APPLICATION_TASK_TAG */
/*-----------------------------------------------------------*/
# if ( configUSE_APPLICATION_TASK_TAG == 1 )
2021-08-10 05:13:43 +08:00
TaskHookFunction_t xTaskGetApplicationTaskTagFromISR ( TaskHandle_t xTask )
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
TaskHookFunction_t xReturn ;
2022-11-14 15:27:00 +08:00
UBaseType_t uxSavedInterruptStatus ;
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
/* If xTask is NULL then set the calling task's hook. */
pxTCB = prvGetTCBFromHandle ( xTask ) ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* Save the hook function in the TCB. A critical section is required as
2021-08-26 13:45:51 +08:00
* the value can be accessed from an interrupt . */
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_MASK_ISR ( & xKernelLock , uxSavedInterruptStatus ) ;
2021-08-10 05:13:43 +08:00
{
xReturn = pxTCB - > pxTaskTag ;
}
2022-11-14 15:27:00 +08:00
prvEXIT_CRITICAL_OR_UNMASK_ISR ( & xKernelLock , uxSavedInterruptStatus ) ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_APPLICATION_TASK_TAG */
/*-----------------------------------------------------------*/
# if ( configUSE_APPLICATION_TASK_TAG == 1 )
2021-09-09 14:13:07 +08:00
BaseType_t xTaskCallApplicationTaskHook ( TaskHandle_t xTask ,
void * pvParameter )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * xTCB ;
BaseType_t xReturn ;
2021-08-10 05:13:43 +08:00
/* If xTask is NULL then we are calling our own task hook. */
if ( xTask = = NULL )
{
xTCB = xTaskGetCurrentTaskHandle ( ) ;
}
else
{
xTCB = xTask ;
}
if ( xTCB - > pxTaskTag ! = NULL )
{
xReturn = xTCB - > pxTaskTag ( pvParameter ) ;
}
else
{
xReturn = pdFAIL ;
}
return xReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_APPLICATION_TASK_TAG */
/*-----------------------------------------------------------*/
2022-11-23 16:48:34 +08:00
# if ( configNUM_CORES > 1 )
static void taskSelectHighestPriorityTaskSMP ( void )
{
/* This function is called from a critical section. So some optimizations are made */
BaseType_t uxCurPriority ;
BaseType_t xTaskScheduled = pdFALSE ;
BaseType_t xNewTopPrioritySet = pdFALSE ;
BaseType_t xCoreID = xPortGetCoreID ( ) ; /* Optimization: Read once */
2021-10-14 01:31:05 +08:00
2022-11-23 16:48:34 +08:00
/* Search for tasks, starting form the highest ready priority. If nothing is
* found , we eventually default to the IDLE tasks at priority 0 */
2021-10-14 01:31:05 +08:00
2022-11-23 16:48:34 +08:00
for ( uxCurPriority = uxTopReadyPriority ; uxCurPriority > = 0 & & xTaskScheduled = = pdFALSE ; uxCurPriority - - )
{
/* Check if current priority has one or more ready tasks. Skip if none */
if ( listLIST_IS_EMPTY ( & ( pxReadyTasksLists [ uxCurPriority ] ) ) )
2021-10-14 01:31:05 +08:00
{
2022-11-23 16:48:34 +08:00
continue ;
}
2022-11-02 19:49:40 +08:00
2022-11-23 16:48:34 +08:00
/* Save a copy of highest priority that has a ready state task */
if ( xNewTopPrioritySet = = pdFALSE )
{
xNewTopPrioritySet = pdTRUE ;
uxTopReadyPriority = uxCurPriority ;
}
2021-10-14 01:31:05 +08:00
2022-11-23 16:48:34 +08:00
/* We now search this priority's ready task list for a runnable task.
* We always start searching from the head of the list , so we reset
* pxIndex to point to the tail so that we start walking the list from
* the first item */
pxReadyTasksLists [ uxCurPriority ] . pxIndex = ( ListItem_t * ) & ( pxReadyTasksLists [ uxCurPriority ] . xListEnd ) ;
2021-10-14 01:31:05 +08:00
2022-11-23 16:48:34 +08:00
/* Get the first item on the list */
TCB_t * pxTCBCur ;
TCB_t * pxTCBFirst ;
listGET_OWNER_OF_NEXT_ENTRY ( pxTCBCur , & ( pxReadyTasksLists [ uxCurPriority ] ) ) ;
pxTCBFirst = pxTCBCur ;
2022-11-02 19:49:40 +08:00
2022-11-23 16:48:34 +08:00
do
{
/* Check if the current task is currently being executed. However, if
* it ' s being executed by the current core , we can still schedule it .
* Todo : Each task can store a xTaskRunState , instead of needing to
* check each core */
UBaseType_t ux ;
2021-10-14 01:31:05 +08:00
2022-11-23 16:48:34 +08:00
for ( ux = 0 ; ux < ( UBaseType_t ) configNUM_CORES ; ux + + )
{
if ( ux = = xCoreID )
2022-11-02 19:49:40 +08:00
{
2022-11-23 16:48:34 +08:00
continue ;
2022-11-02 19:49:40 +08:00
}
2022-11-23 16:48:34 +08:00
else if ( pxCurrentTCB [ ux ] = = pxTCBCur )
2022-11-02 19:49:40 +08:00
{
2022-11-23 16:48:34 +08:00
/* Current task is already being executed. Get the next task */
2022-11-02 19:49:40 +08:00
goto get_next_task ;
}
2022-11-23 16:48:34 +08:00
}
2022-11-02 19:49:40 +08:00
2022-11-23 16:48:34 +08:00
/* Check if the current task has a compatible affinity */
if ( ( pxTCBCur - > xCoreID ! = xCoreID ) & & ( pxTCBCur - > xCoreID ! = tskNO_AFFINITY ) )
{
goto get_next_task ;
}
2022-11-02 19:49:40 +08:00
2022-11-23 16:48:34 +08:00
/* The current task is runnable. Schedule it */
pxCurrentTCB [ xCoreID ] = pxTCBCur ;
xTaskScheduled = pdTRUE ;
2021-10-14 01:31:05 +08:00
2022-11-23 16:48:34 +08:00
/* Move the current tasks list item to the back of the list in order
* to implement best effort round robin . To do this , we need to reset
* the pxIndex to point to the tail again . */
pxReadyTasksLists [ uxCurPriority ] . pxIndex = ( ListItem_t * ) & ( pxReadyTasksLists [ uxCurPriority ] . xListEnd ) ;
uxListRemove ( & ( pxTCBCur - > xStateListItem ) ) ;
vListInsertEnd ( & ( pxReadyTasksLists [ uxCurPriority ] ) , & ( pxTCBCur - > xStateListItem ) ) ;
break ;
2021-10-14 01:31:05 +08:00
2022-11-23 16:48:34 +08:00
get_next_task :
/* The current task cannot be scheduled. Get the next task in the list */
listGET_OWNER_OF_NEXT_ENTRY ( pxTCBCur , & ( pxReadyTasksLists [ uxCurPriority ] ) ) ;
} while ( pxTCBCur ! = pxTCBFirst ) ; /* Check to see if we've walked the entire list */
2022-11-02 19:49:40 +08:00
}
2022-11-23 16:48:34 +08:00
assert ( xTaskScheduled = = pdTRUE ) ; /* At this point, a task MUST have been scheduled */
}
# endif /* configNUM_CORES > 1 */
2021-10-14 01:31:05 +08:00
void vTaskSwitchContext ( void )
{
2022-11-14 15:28:58 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
2022-11-14 15:28:58 +08:00
/* For SMP, we need to take the kernel lock here as we are about to
* access kernel data structures ( unlike single core which calls this
* function with either interrupts disabled or when the scheduler hasn ' t
* started yet ) . */
2022-11-02 19:49:40 +08:00
taskENTER_CRITICAL_ISR ( & xKernelLock ) ;
2022-11-14 15:28:58 +08:00
# endif /* ( configNUM_CORES > 1 ) */
2022-11-02 19:49:40 +08:00
2021-08-10 05:13:43 +08:00
if ( uxSchedulerSuspended [ xPortGetCoreID ( ) ] ! = ( UBaseType_t ) pdFALSE )
{
/* The scheduler is currently suspended - do not allow a context
2021-08-26 13:45:51 +08:00
* switch . */
2021-08-10 05:13:43 +08:00
xYieldPending [ xPortGetCoreID ( ) ] = pdTRUE ;
}
else
{
xYieldPending [ xPortGetCoreID ( ) ] = pdFALSE ;
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM
xSwitchingContext [ xPortGetCoreID ( ) ] = pdTRUE ;
# endif // ESP_PLATFORM
2021-08-10 05:13:43 +08:00
traceTASK_SWITCHED_OUT ( ) ;
# if ( configGENERATE_RUN_TIME_STATS == 1 )
2021-08-26 13:45:51 +08:00
{
2021-08-10 05:13:43 +08:00
# ifdef portALT_GET_RUN_TIME_COUNTER_VALUE
portALT_GET_RUN_TIME_COUNTER_VALUE ( ulTotalRunTime ) ;
# else
ulTotalRunTime = portGET_RUN_TIME_COUNTER_VALUE ( ) ;
# endif
/* Add the amount of time the task has been running to the
2021-08-26 13:45:51 +08:00
* accumulated time so far . The time the task started running was
* stored in ulTaskSwitchedInTime . Note that there is no overflow
* protection here so count values are only valid until the timer
* overflows . The guard against negative values is to protect
* against suspect run time stat counter implementations - which
* are provided by the application , not the kernel . */
2021-08-10 05:13:43 +08:00
if ( ulTotalRunTime > ulTaskSwitchedInTime [ xPortGetCoreID ( ) ] )
{
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ulRunTimeCounter + = ( ulTotalRunTime - ulTaskSwitchedInTime [ xPortGetCoreID ( ) ] ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-10-14 01:31:05 +08:00
2021-08-10 05:13:43 +08:00
ulTaskSwitchedInTime [ xPortGetCoreID ( ) ] = ulTotalRunTime ;
2021-08-26 13:45:51 +08:00
}
2021-08-10 05:13:43 +08:00
# endif /* configGENERATE_RUN_TIME_STATS */
/* Check for stack overflow, if configured. */
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM
taskFIRST_CHECK_FOR_STACK_OVERFLOW ( ) ;
taskSECOND_CHECK_FOR_STACK_OVERFLOW ( ) ;
# else
taskCHECK_FOR_STACK_OVERFLOW ( ) ;
2021-08-10 05:13:43 +08:00
2022-11-02 19:49:40 +08:00
/* Before the currently running task is switched out, save its errno. */
# if ( configUSE_POSIX_ERRNO == 1 )
{
pxCurrentTCB - > iTaskErrno = FreeRTOS_errno ;
}
# endif
# endif // ESP_PLATFORM
2021-08-10 05:13:43 +08:00
2021-10-14 01:31:05 +08:00
/* Select a new task to run using either the generic C or port
* optimised asm code . */
taskSELECT_HIGHEST_PRIORITY_TASK ( ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
2021-08-10 05:13:43 +08:00
traceTASK_SWITCHED_IN ( ) ;
2016-08-17 23:08:22 +08:00
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM
xSwitchingContext [ xPortGetCoreID ( ) ] = pdFALSE ;
# if CONFIG_FREERTOS_WATCHPOINT_END_OF_STACK
vPortSetStackWatchpoint ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > pxStack ) ;
# endif
# else
/* After the new task is switched in, update the global errno. */
# if ( configUSE_POSIX_ERRNO == 1 )
{
FreeRTOS_errno = pxCurrentTCB - > iTaskErrno ;
}
# endif
2017-01-09 16:42:45 +08:00
2022-11-02 19:49:40 +08:00
# if ( configUSE_NEWLIB_REENTRANT == 1 )
{
/* Switch Newlib's _impure_ptr variable to point to the _reent
* structure specific to this task .
* See the third party link http : //www.nadler.com/embedded/newlibAndFreeRTOS.html
* for additional information . */
_impure_ptr = & ( pxCurrentTCB - > xNewLib_reent ) ;
}
# endif /* configUSE_NEWLIB_REENTRANT */
# endif // ESP_PLATFORM
2021-08-10 05:13:43 +08:00
}
2022-11-02 19:49:40 +08:00
2022-11-14 15:28:58 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock as we have finished
* accessing the kernel data structures . */
2022-11-02 19:49:40 +08:00
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
2022-11-14 15:28:58 +08:00
# endif /* ( configNUM_CORES > 1 ) */
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2021-09-09 14:13:07 +08:00
void vTaskPlaceOnEventList ( List_t * const pxEventList ,
const TickType_t xTicksToWait )
2016-08-17 23:08:22 +08:00
{
2021-08-10 05:13:43 +08:00
configASSERT ( pxEventList ) ;
2022-08-01 17:32:40 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* In SMP, we need to take the kernel lock as we are about to access the
* task lists . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* configNUM_CORES > 1 */
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* THIS FUNCTION MUST BE CALLED WITH EITHER INTERRUPTS DISABLED OR THE
2021-08-26 13:45:51 +08:00
* SCHEDULER SUSPENDED AND THE QUEUE BEING ACCESSED LOCKED . */
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
/* Place the event list item of the TCB in the appropriate event list.
2021-08-26 13:45:51 +08:00
* This is placed in the list in priority order so the highest priority task
* is the first to be woken by the event . The queue that contains the event
* list is locked , preventing simultaneous access from interrupts . */
2022-08-01 17:32:40 +08:00
vListInsert ( pxEventList , & ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > xEventListItem ) ) ;
2016-08-17 23:08:22 +08:00
2022-08-01 19:06:44 +08:00
prvAddCurrentTaskToDelayedList ( xTicksToWait , pdTRUE ) ;
2022-08-01 17:32:40 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif /* configNUM_CORES > 1 */
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2021-09-09 14:13:07 +08:00
void vTaskPlaceOnUnorderedEventList ( List_t * pxEventList ,
const TickType_t xItemValue ,
const TickType_t xTicksToWait )
2016-08-17 23:08:22 +08:00
{
2021-08-10 05:13:43 +08:00
configASSERT ( pxEventList ) ;
2022-08-01 17:32:40 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
2021-08-10 05:13:43 +08:00
2022-11-14 15:27:00 +08:00
/* In SMP, the event groups haven't suspended the scheduler at this
* point . We need to take the kernel lock instead as we are about to
* access the task lists . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# else /* configNUM_CORES > 1 */
2022-11-02 19:49:40 +08:00
2022-11-14 15:27:00 +08:00
/* THIS FUNCTION MUST BE CALLED WITH THE SCHEDULER SUSPENDED. It is used by
* the event groups implementation . */
configASSERT ( uxSchedulerSuspended [ 0 ] ! = 0 ) ;
# endif /* configNUM_CORES > 1 */
2022-08-01 17:32:40 +08:00
2021-08-10 05:13:43 +08:00
/* Store the item value in the event list item. It is safe to access the
2021-08-26 13:45:51 +08:00
* event list item here as interrupts won ' t access the event list item of a
* task that is not in the Blocked state . */
2022-08-01 17:32:40 +08:00
listSET_LIST_ITEM_VALUE ( & ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > xEventListItem ) , xItemValue | taskEVENT_LIST_ITEM_VALUE_IN_USE ) ;
2021-08-10 05:13:43 +08:00
/* Place the event list item of the TCB at the end of the appropriate event
2021-08-26 13:45:51 +08:00
* list . It is safe to access the event list here because it is part of an
* event group implementation - and interrupts don ' t access event groups
* directly ( instead they access them indirectly by pending function calls to
* the task level ) . */
2022-08-01 17:32:40 +08:00
vListInsertEnd ( pxEventList , & ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > xEventListItem ) ) ;
2021-08-10 05:13:43 +08:00
2022-08-01 19:06:44 +08:00
prvAddCurrentTaskToDelayedList ( xTicksToWait , pdTRUE ) ;
2022-08-01 17:32:40 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif /* configNUM_CORES > 1 */
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
# if ( configUSE_TIMERS == 1 )
2016-08-17 23:08:22 +08:00
2022-11-02 19:49:40 +08:00
void vTaskPlaceOnEventListRestricted ( List_t * const pxEventList ,
TickType_t xTicksToWait ,
const BaseType_t xWaitIndefinitely )
2021-08-10 05:13:43 +08:00
{
configASSERT ( pxEventList ) ;
2016-08-17 23:08:22 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* In SMP, we need to take the kernel lock as we are about to access
* the task lists . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* configNUM_CORES > 1 */
2022-08-01 17:32:40 +08:00
2021-08-10 05:13:43 +08:00
/* This function should not be called by application code hence the
2021-08-26 13:45:51 +08:00
* ' Restricted ' in its name . It is not part of the public API . It is
* designed for use by kernel code , and has special calling requirements -
* it should be called with the scheduler suspended . */
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* Place the event list item of the TCB in the appropriate event list.
2021-08-26 13:45:51 +08:00
* In this case it is assume that this is the only task that is going to
* be waiting on this event list , so the faster vListInsertEnd ( ) function
* can be used in place of vListInsert . */
2022-08-01 17:32:40 +08:00
vListInsertEnd ( pxEventList , & ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > xEventListItem ) ) ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* If the task should block indefinitely then set the block time to a
2021-08-26 13:45:51 +08:00
* value that will be recognised as an indefinite delay inside the
* prvAddCurrentTaskToDelayedList ( ) function . */
2021-08-10 05:13:43 +08:00
if ( xWaitIndefinitely ! = pdFALSE )
{
xTicksToWait = portMAX_DELAY ;
}
2016-08-17 23:08:22 +08:00
2022-08-01 17:32:40 +08:00
traceTASK_DELAY_UNTIL ( ( xTickCount + xTicksToWait ) ) ;
2022-08-01 19:06:44 +08:00
prvAddCurrentTaskToDelayedList ( xTicksToWait , xWaitIndefinitely ) ;
2022-08-01 17:32:40 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TIMERS */
/*-----------------------------------------------------------*/
BaseType_t xTaskRemoveFromEventList ( const List_t * const pxEventList )
{
2021-08-26 13:45:51 +08:00
TCB_t * pxUnblockedTCB ;
BaseType_t xReturn ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* THIS FUNCTION MUST BE CALLED FROM A CRITICAL SECTION. It can also be
2022-11-14 15:27:00 +08:00
* called from a critical section within an ISR . */
# if ( configNUM_CORES > 1 )
/* In SMP, we need to take the kernel lock (even if the caller is
* already in a critical section by taking a different lock ) as we are
* about to access the task lists , which are protected by the kernel
* lock . This function can also be called from an ISR context , so we
* need to check whether we are in an ISR . */
if ( portCHECK_IF_IN_ISR ( ) = = pdFALSE )
{
taskENTER_CRITICAL ( & xKernelLock ) ;
}
else
{
taskENTER_CRITICAL_ISR ( & xKernelLock ) ;
}
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
{
2022-08-01 18:38:19 +08:00
/* Before taking the kernel lock, another task/ISR could have already
* emptied the pxEventList . So we insert a check here to see if
* pxEventList is empty before attempting to remove an item from it . */
if ( listLIST_IS_EMPTY ( pxEventList ) = = pdFALSE )
{
BaseType_t xCurCoreID = xPortGetCoreID ( ) ;
/* The event list is sorted in priority order, so the first in the list can
* be removed as it is known to be the highest priority . Remove the TCB from
* the delayed list , and add it to the ready list .
*
* If an event is for a queue that is locked then this function will never
* get called - the lock count on the queue will get modified instead . This
* means exclusive access to the event list is guaranteed here .
*
* This function assumes that a check has already been made to ensure that
* pxEventList is not empty . */
pxUnblockedTCB = listGET_OWNER_OF_HEAD_ENTRY ( pxEventList ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
configASSERT ( pxUnblockedTCB ) ;
( void ) uxListRemove ( & ( pxUnblockedTCB - > xEventListItem ) ) ;
/* Add the task to the ready list if a core with compatible affinity
* has NOT suspended its scheduler . This occurs when :
* - The task is pinned , and the pinned core ' s scheduler is running
* - The task is unpinned , and at least one of the core ' s scheduler is running */
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
if ( ( ( uxSchedulerSuspended [ 0 ] = = ( UBaseType_t ) pdFALSE ) & & ( taskCAN_RUN_ON_CORE ( 0 , pxUnblockedTCB - > xCoreID ) = = pdTRUE ) ) | |
( ( uxSchedulerSuspended [ 1 ] = = ( UBaseType_t ) pdFALSE ) & & ( taskCAN_RUN_ON_CORE ( 1 , pxUnblockedTCB - > xCoreID ) = = pdTRUE ) ) )
2022-08-01 18:38:19 +08:00
# else
if ( uxSchedulerSuspended [ 0 ] = = ( UBaseType_t ) pdFALSE )
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
{
2022-08-01 18:38:19 +08:00
( void ) uxListRemove ( & ( pxUnblockedTCB - > xStateListItem ) ) ;
prvAddTaskToReadyList ( pxUnblockedTCB ) ;
2021-08-10 05:13:43 +08:00
2022-08-01 18:38:19 +08:00
# if ( configUSE_TICKLESS_IDLE != 0 )
{
/* If a task is blocked on a kernel object then xNextTaskUnblockTime
* might be set to the blocked task ' s time out time . If the task is
* unblocked for a reason other than a timeout xNextTaskUnblockTime is
* normally left unchanged , because it is automatically reset to a new
* value when the tick count equals xNextTaskUnblockTime . However if
* tickless idling is used it might be more important to enter sleep mode
* at the earliest possible time - so reset xNextTaskUnblockTime here to
* ensure it is updated at the earliest possible time . */
prvResetNextTaskUnblockTime ( ) ;
}
# endif
}
else
{
/* We arrive here due to one of the following possibilities:
* - The task is pinned to core X and core X has suspended its scheduler
* - The task is unpinned and both cores have suspend their schedulers
* Therefore , we add the task to one of the pending lists :
* - If the task is pinned to core X , add it to core X ' s pending list
* - If the task is unpinned , add it to the current core ' s pending list */
BaseType_t xPendingListCore ;
# if ( configNUM_CORES > 1 )
xPendingListCore = ( ( pxUnblockedTCB - > xCoreID = = tskNO_AFFINITY ) ? xCurCoreID : pxUnblockedTCB - > xCoreID ) ;
# else
xPendingListCore = 0 ;
# endif /* configNUM_CORES > 1 */
configASSERT ( uxSchedulerSuspended [ xPendingListCore ] = = pdTRUE ) ;
2021-08-10 05:13:43 +08:00
2022-08-01 18:38:19 +08:00
/* The delayed and ready lists cannot be accessed, so hold this task
* pending until the scheduler is resumed . */
vListInsertEnd ( & ( xPendingReadyList [ xPendingListCore ] ) , & ( pxUnblockedTCB - > xEventListItem ) ) ;
}
2021-08-10 05:13:43 +08:00
2022-08-01 18:38:19 +08:00
if ( prvCheckForYield ( pxUnblockedTCB , xCurCoreID , pdFALSE ) )
{
/* Return true if the task removed from the event list has a higher
* priority than the calling task . This allows the calling task to know if
* it should force a context switch now . */
xReturn = pdTRUE ;
2021-08-10 05:13:43 +08:00
2022-08-01 18:38:19 +08:00
/* Mark that a yield is pending in case the user is not using the
* " xHigherPriorityTaskWoken " parameter to an ISR safe FreeRTOS function . */
xYieldPending [ xCurCoreID ] = pdTRUE ;
}
else
{
xReturn = pdFALSE ;
}
}
else
{
/* The pxEventList was emptied before we entered the critical section,
* Nothing to do except return pdFALSE . */
xReturn = pdFALSE ;
}
2021-08-10 05:13:43 +08:00
}
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
if ( portCHECK_IF_IN_ISR ( ) = = pdFALSE )
{
taskEXIT_CRITICAL ( & xKernelLock ) ;
}
else
{
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
}
# endif /* configNUM_CORES > 1 */
2022-08-01 18:38:19 +08:00
2021-08-10 05:13:43 +08:00
return xReturn ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
void vTaskTakeKernelLock ( void )
{
/* We call the tasks.c critical section macro to take xKernelLock */
taskENTER_CRITICAL ( & xKernelLock ) ;
}
2022-06-29 00:58:08 +08:00
2022-11-02 19:49:40 +08:00
void vTaskReleaseKernelLock ( void )
{
/* We call the tasks.c critical section macro to release xKernelLock */
taskEXIT_CRITICAL ( & xKernelLock ) ;
}
2022-11-14 15:27:00 +08:00
# endif /* configNUM_CORES > 1 */
2022-06-29 00:58:08 +08:00
2021-09-09 14:13:07 +08:00
void vTaskRemoveFromUnorderedEventList ( ListItem_t * pxEventListItem ,
const TickType_t xItemValue )
2016-08-17 23:08:22 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * pxUnblockedTCB ;
2022-08-01 18:38:19 +08:00
BaseType_t xCurCoreID = xPortGetCoreID ( ) ;
2016-08-17 23:08:22 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
2022-11-02 19:49:40 +08:00
2022-11-14 15:27:00 +08:00
/* THIS FUNCTION MUST BE CALLED WITH THE KERNEL LOCK ALREADY TAKEN.
* It is used by the event flags implementation , thus those functions
* should call vTaskTakeKernelLock ( ) before calling this function . */
# else /* configNUM_CORES > 1 */
/* THIS FUNCTION MUST BE CALLED WITH THE SCHEDULER SUSPENDED. It is used by
* the event flags implementation . */
configASSERT ( uxSchedulerSuspended ! = pdFALSE ) ;
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
/* Store the new item value in the event list. */
listSET_LIST_ITEM_VALUE ( pxEventListItem , xItemValue | taskEVENT_LIST_ITEM_VALUE_IN_USE ) ;
/* Remove the event list form the event flag. Interrupts do not access
2021-08-26 13:45:51 +08:00
* event flags . */
2022-08-01 18:38:19 +08:00
pxUnblockedTCB = listGET_LIST_ITEM_OWNER ( pxEventListItem ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
2021-08-10 05:13:43 +08:00
configASSERT ( pxUnblockedTCB ) ;
( void ) uxListRemove ( pxEventListItem ) ;
2022-08-01 18:38:19 +08:00
# if ( configUSE_TICKLESS_IDLE != 0 )
{
/* If a task is blocked on a kernel object then xNextTaskUnblockTime
* might be set to the blocked task ' s time out time . If the task is
* unblocked for a reason other than a timeout xNextTaskUnblockTime is
* normally left unchanged , because it is automatically reset to a new
* value when the tick count equals xNextTaskUnblockTime . However if
* tickless idling is used it might be more important to enter sleep mode
* at the earliest possible time - so reset xNextTaskUnblockTime here to
* ensure it is updated at the earliest possible time . */
prvResetNextTaskUnblockTime ( ) ;
}
# endif
2021-08-10 05:13:43 +08:00
/* Remove the task from the delayed list and add it to the ready list. The
2021-08-26 13:45:51 +08:00
* scheduler is suspended so interrupts will not be accessing the ready
* lists . */
2021-08-10 05:13:43 +08:00
( void ) uxListRemove ( & ( pxUnblockedTCB - > xStateListItem ) ) ;
prvAddTaskToReadyList ( pxUnblockedTCB ) ;
2022-08-01 18:38:19 +08:00
if ( prvCheckForYield ( pxUnblockedTCB , xCurCoreID , pdFALSE ) )
2021-08-10 05:13:43 +08:00
{
2022-08-01 18:38:19 +08:00
/* The unblocked task has a priority above that of the calling task, so
* a context switch is required . This function is called with the
* scheduler suspended so xYieldPending is set so the context switch
* occurs immediately that the scheduler is resumed ( unsuspended ) . */
xYieldPending [ xCurCoreID ] = pdTRUE ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
void vTaskSetTimeOutState ( TimeOut_t * const pxTimeOut )
{
2021-08-10 05:13:43 +08:00
configASSERT ( pxTimeOut ) ;
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
pxTimeOut - > xOverflowCount = xNumOfOverflows ;
pxTimeOut - > xTimeOnEntering = xTickCount ;
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2019-11-28 15:27:47 -03:00
}
/*-----------------------------------------------------------*/
void vTaskInternalSetTimeOutState ( TimeOut_t * const pxTimeOut )
2016-08-17 23:08:22 +08:00
{
2022-07-27 11:40:34 +08:00
/**
* In case of we are building for SMP , we need to protect the following instructions in order to make them
* atomic .
* Indeed , without this , it would be possible to get preempted by the tick hook right after storing the number
* of overflows with ` pxTimeOut - > xOverflowCount = xNumOfOverflows ` . Then , the tick hook increments the timer ,
* which overflows , and thus resets the xTickCount to 0.
* Resuming our task would result in an invalid state of the timer where the number of overflow corresponds
* to the previous value and not the current one .
*
* On a single core configuration , this problem doesn ' t appear as this function is meant to be called from
* a critical section , disabling the ( tick ) interrupts .
*/
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
2022-07-27 11:40:34 +08:00
configASSERT ( pxTimeOut ) ;
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2022-11-14 15:27:00 +08:00
# endif /* configNUM_CORES > 1 */
2022-07-27 11:40:34 +08:00
2022-11-02 19:49:40 +08:00
/* For internal use only as it does not use a critical section. */
pxTimeOut - > xOverflowCount = xNumOfOverflows ;
pxTimeOut - > xTimeOnEntering = xTickCount ;
2022-07-27 11:40:34 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2022-11-14 15:27:00 +08:00
# endif /* configNUM_CORES > 1 */
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2021-09-09 14:13:07 +08:00
BaseType_t xTaskCheckForTimeOut ( TimeOut_t * const pxTimeOut ,
TickType_t * const pxTicksToWait )
2016-08-17 23:08:22 +08:00
{
2021-08-26 13:45:51 +08:00
BaseType_t xReturn ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
configASSERT ( pxTimeOut ) ;
configASSERT ( pxTicksToWait ) ;
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* Minor optimisation. The tick count cannot change in this block. */
const TickType_t xConstTickCount = xTickCount ;
const TickType_t xElapsedTime = xConstTickCount - pxTimeOut - > xTimeOnEntering ;
2021-08-26 13:45:51 +08:00
# if ( INCLUDE_xTaskAbortDelay == 1 )
2022-11-02 19:49:40 +08:00
if ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > ucDelayAborted ! = ( uint8_t ) pdFALSE )
2021-08-10 05:13:43 +08:00
{
/* The delay was aborted, which is not the same as a time out,
2021-08-26 13:45:51 +08:00
* but has the same result . */
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ucDelayAborted = pdFALSE ;
2021-08-10 05:13:43 +08:00
xReturn = pdTRUE ;
}
else
# endif
# if ( INCLUDE_vTaskSuspend == 1 )
if ( * pxTicksToWait = = portMAX_DELAY )
{
/* If INCLUDE_vTaskSuspend is set to 1 and the block time
2021-08-26 13:45:51 +08:00
* specified is the maximum block time then the task should block
* indefinitely , and therefore never time out . */
2021-08-10 05:13:43 +08:00
xReturn = pdFALSE ;
}
else
# endif
if ( ( xNumOfOverflows ! = pxTimeOut - > xOverflowCount ) & & ( xConstTickCount > = pxTimeOut - > xTimeOnEntering ) ) /*lint !e525 Indentation preferred as is to make code within pre-processor directives clearer. */
{
/* The tick count is greater than the time at which
2021-08-26 13:45:51 +08:00
* vTaskSetTimeout ( ) was called , but has also overflowed since
* vTaskSetTimeOut ( ) was called . It must have wrapped all the way
* around and gone past again . This passed since vTaskSetTimeout ( )
* was called . */
2021-08-10 05:13:43 +08:00
xReturn = pdTRUE ;
2022-09-05 17:31:22 +08:00
* pxTicksToWait = ( TickType_t ) 0 ;
2021-08-10 05:13:43 +08:00
}
else if ( xElapsedTime < * pxTicksToWait ) /*lint !e961 Explicit casting is only redundant with some compilers, whereas others require it to prevent integer conversion errors. */
{
/* Not a genuine timeout. Adjust parameters for time remaining. */
* pxTicksToWait - = xElapsedTime ;
vTaskInternalSetTimeOutState ( pxTimeOut ) ;
xReturn = pdFALSE ;
}
else
{
2022-09-05 17:31:22 +08:00
* pxTicksToWait = ( TickType_t ) 0 ;
2021-08-10 05:13:43 +08:00
xReturn = pdTRUE ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return xReturn ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
void vTaskMissedYield ( void )
{
2022-11-02 19:49:40 +08:00
xYieldPending [ xPortGetCoreID ( ) ] = pdTRUE ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
# if ( configUSE_TRACE_FACILITY == 1 )
2021-08-10 05:13:43 +08:00
UBaseType_t uxTaskGetTaskNumber ( TaskHandle_t xTask )
{
2021-08-26 13:45:51 +08:00
UBaseType_t uxReturn ;
TCB_t const * pxTCB ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
if ( xTask ! = NULL )
{
pxTCB = xTask ;
uxReturn = pxTCB - > uxTaskNumber ;
}
else
{
uxReturn = 0U ;
}
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
return uxReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TRACE_FACILITY */
/*-----------------------------------------------------------*/
# if ( configUSE_TRACE_FACILITY == 1 )
2021-09-09 14:13:07 +08:00
void vTaskSetTaskNumber ( TaskHandle_t xTask ,
const UBaseType_t uxHandle )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
if ( xTask ! = NULL )
{
pxTCB = xTask ;
pxTCB - > uxTaskNumber = uxHandle ;
}
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TRACE_FACILITY */
/*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* The Idle task .
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
*
* The portTASK_FUNCTION ( ) macro is used to allow port / compiler specific
* language extensions . The equivalent prototype for this function is :
*
* void prvIdleTask ( void * pvParameters ) ;
*
*/
static portTASK_FUNCTION ( prvIdleTask , pvParameters )
{
2021-08-10 05:13:43 +08:00
/* Stop warnings. */
( void ) pvParameters ;
/** THIS IS THE RTOS IDLE TASK - WHICH IS CREATED AUTOMATICALLY WHEN THE
2021-08-26 13:45:51 +08:00
* SCHEDULER IS STARTED . * */
2021-08-10 05:13:43 +08:00
/* In case a task that has a secure context deletes itself, in which case
2021-08-26 13:45:51 +08:00
* the idle task is responsible for deleting the task ' s secure context , if
* any . */
2021-08-10 05:13:43 +08:00
portALLOCATE_SECURE_CONTEXT ( configMINIMAL_SECURE_STACK_SIZE ) ;
2021-08-26 13:45:51 +08:00
for ( ; ; )
2021-08-10 05:13:43 +08:00
{
/* See if any tasks have deleted themselves - if so then the idle task
2021-08-26 13:45:51 +08:00
* is responsible for freeing the deleted task ' s TCB and stack . */
2021-08-10 05:13:43 +08:00
prvCheckTasksWaitingTermination ( ) ;
# if ( configUSE_PREEMPTION == 0 )
{
2021-08-26 13:45:51 +08:00
/* If we are not using preemption we keep forcing a task switch to
* see if any other task has become available . If we are using
* preemption we don ' t need to do this as any task becoming available
* will automatically get the processor anyway . */
2021-08-10 05:13:43 +08:00
taskYIELD ( ) ;
}
2021-08-26 13:45:51 +08:00
# endif /* configUSE_PREEMPTION */
# if ( ( configUSE_PREEMPTION == 1 ) && ( configIDLE_SHOULD_YIELD == 1 ) )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
/* When using preemption tasks of equal priority will be
* timesliced . If a task that is sharing the idle priority is ready
* to run then the idle task should yield before the end of the
* timeslice .
*
* A critical region is not required here as we are just reading from
* the list , and an occasional incorrect value will not matter . If
* the ready list at the idle priority contains more than one task
* then a task other than the idle task is ready to execute . */
if ( listCURRENT_LIST_LENGTH ( & ( pxReadyTasksLists [ tskIDLE_PRIORITY ] ) ) > ( UBaseType_t ) 1 )
{
taskYIELD ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-10 05:13:43 +08:00
}
# endif /* ( ( configUSE_PREEMPTION == 1 ) && ( configIDLE_SHOULD_YIELD == 1 ) ) */
# if ( configUSE_IDLE_HOOK == 1 )
2021-08-26 13:45:51 +08:00
{
extern void vApplicationIdleHook ( void ) ;
/* Call the user defined function from within the idle task. This
* allows the application designer to add background functionality
* without the overhead of a separate task .
* NOTE : vApplicationIdleHook ( ) MUST NOT , UNDER ANY CIRCUMSTANCES ,
* CALL A FUNCTION THAT MIGHT BLOCK . */
vApplicationIdleHook ( ) ;
}
2021-08-10 05:13:43 +08:00
# endif /* configUSE_IDLE_HOOK */
2022-02-17 16:40:52 +08:00
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM
/* Call the esp-idf idle hook system */
esp_vApplicationIdleHook ( ) ;
# endif // ESP_PLATFORM
2021-08-10 05:13:43 +08:00
/* This conditional compilation should use inequality to 0, not equality
2021-08-26 13:45:51 +08:00
* to 1. This is to ensure portSUPPRESS_TICKS_AND_SLEEP ( ) is called when
* user defined low power mode implementations require
* configUSE_TICKLESS_IDLE to be set to a value other than 1. */
2021-08-10 05:13:43 +08:00
# if ( configUSE_TICKLESS_IDLE != 0 )
{
2021-08-26 13:45:51 +08:00
TickType_t xExpectedIdleTime ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
/* It is not desirable to suspend then resume the scheduler on
* each iteration of the idle task . Therefore , a preliminary
* test of the expected idle time is performed without the
* scheduler suspended . The result here is not necessarily
* valid . */
xExpectedIdleTime = prvGetExpectedIdleTime ( ) ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
if ( xExpectedIdleTime > = configEXPECTED_IDLE_TIME_BEFORE_SLEEP )
{
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_SUSPEND_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
/* Now the scheduler is suspended, the expected idle
* time can be sampled again , and this time its value can
* be used . */
configASSERT ( xNextTaskUnblockTime > = xTickCount ) ;
xExpectedIdleTime = prvGetExpectedIdleTime ( ) ;
/* Define the following macro to set xExpectedIdleTime to 0
* if the application does not want
* portSUPPRESS_TICKS_AND_SLEEP ( ) to be called . */
configPRE_SUPPRESS_TICKS_AND_SLEEP_PROCESSING ( xExpectedIdleTime ) ;
if ( xExpectedIdleTime > = configEXPECTED_IDLE_TIME_BEFORE_SLEEP )
{
traceLOW_POWER_IDLE_BEGIN ( ) ;
portSUPPRESS_TICKS_AND_SLEEP ( xExpectedIdleTime ) ;
traceLOW_POWER_IDLE_END ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-10 05:13:43 +08:00
}
2022-11-14 15:27:00 +08:00
( void ) prvEXIT_CRITICAL_OR_RESUME_ALL ( & xKernelLock ) ;
2021-08-26 13:45:51 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
2021-08-10 05:13:43 +08:00
}
}
# endif /* configUSE_TICKLESS_IDLE */
}
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
# if ( configUSE_TICKLESS_IDLE != 0 )
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
eSleepModeStatus eTaskConfirmSleepModeStatus ( void )
{
2021-08-26 13:45:51 +08:00
/* The idle task exists in addition to the application tasks. */
const UBaseType_t uxNonApplicationTasks = 1 ;
eSleepModeStatus eReturn = eStandardSleep ;
2021-08-10 05:13:43 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2022-11-02 19:49:40 +08:00
if ( listCURRENT_LIST_LENGTH ( & xPendingReadyList [ xPortGetCoreID ( ) ] ) ! = 0 )
2021-08-10 05:13:43 +08:00
{
/* A task was made ready while the scheduler was suspended. */
eReturn = eAbortSleep ;
}
2022-11-02 19:49:40 +08:00
else if ( xYieldPending [ xPortGetCoreID ( ) ] ! = pdFALSE )
2021-08-10 05:13:43 +08:00
{
/* A yield was pended while the scheduler was suspended. */
eReturn = eAbortSleep ;
}
else
{
/* If all the tasks are in the suspended list (which might mean they
2021-08-26 13:45:51 +08:00
* have an infinite block time rather than actually being suspended )
* then it is safe to turn all clocks off and just wait for external
* interrupts . */
2021-08-10 05:13:43 +08:00
if ( listCURRENT_LIST_LENGTH ( & xSuspendedTaskList ) = = ( uxCurrentNumberOfTasks - uxNonApplicationTasks ) )
{
eReturn = eNoTasksWaitingTimeout ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-11-02 19:49:40 +08:00
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return eReturn ;
}
2019-11-28 15:27:47 -03:00
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TICKLESS_IDLE */
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
2016-08-17 23:08:22 +08:00
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS != 0 )
2022-11-08 17:48:48 +08:00
# if ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 )
2016-08-24 13:29:06 +08:00
2022-11-02 19:49:40 +08:00
void vTaskSetThreadLocalStoragePointerAndDelCallback ( TaskHandle_t xTaskToSet ,
BaseType_t xIndex ,
void * pvValue ,
TlsDeleteCallbackFunction_t xDelCallback )
2021-08-10 05:13:43 +08:00
{
2022-11-02 19:49:40 +08:00
TCB_t * pxTCB ;
2016-08-24 13:29:06 +08:00
2022-11-02 19:49:40 +08:00
if ( xIndex < configNUM_THREAD_LOCAL_STORAGE_POINTERS )
{
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* For SMP, we need to take the kernel lock here as we
* another core could also update this task ' s TLSP at the
* same time . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2022-11-02 19:49:40 +08:00
pxTCB = prvGetTCBFromHandle ( xTaskToSet ) ;
pxTCB - > pvThreadLocalStoragePointers [ xIndex ] = pvValue ;
pxTCB - > pvThreadLocalStoragePointersDelCallback [ xIndex ] = xDelCallback ;
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif /* configNUM_CORES > 1 */
2022-11-02 19:49:40 +08:00
}
}
2016-08-24 13:29:06 +08:00
2022-11-02 19:49:40 +08:00
void vTaskSetThreadLocalStoragePointer ( TaskHandle_t xTaskToSet ,
BaseType_t xIndex ,
void * pvValue )
{
vTaskSetThreadLocalStoragePointerAndDelCallback ( xTaskToSet , xIndex , pvValue , ( TlsDeleteCallbackFunction_t ) NULL ) ;
}
2016-08-24 13:29:06 +08:00
2021-08-10 05:13:43 +08:00
2022-11-08 17:48:48 +08:00
# else /* if ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 ) */
2022-11-02 19:49:40 +08:00
void vTaskSetThreadLocalStoragePointer ( TaskHandle_t xTaskToSet ,
BaseType_t xIndex ,
void * pvValue )
2021-08-10 05:13:43 +08:00
{
2022-11-02 19:49:40 +08:00
TCB_t * pxTCB ;
if ( xIndex < configNUM_THREAD_LOCAL_STORAGE_POINTERS )
{
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* For SMP, we need to take the kernel lock here as we
* another core could also update this task ' s TLSP at the
* same time . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2022-11-02 19:49:40 +08:00
pxTCB = prvGetTCBFromHandle ( xTaskToSet ) ;
2022-11-14 15:27:00 +08:00
configASSERT ( pxTCB ! = NULL ) ;
2022-11-02 19:49:40 +08:00
pxTCB - > pvThreadLocalStoragePointers [ xIndex ] = pvValue ;
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif /* configNUM_CORES > 1 */
2022-11-02 19:49:40 +08:00
}
2021-08-10 05:13:43 +08:00
}
2022-11-08 17:48:48 +08:00
# endif /* configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 */
2016-08-17 23:08:22 +08:00
# endif /* configNUM_THREAD_LOCAL_STORAGE_POINTERS */
/*-----------------------------------------------------------*/
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS != 0 )
2021-08-26 13:45:51 +08:00
void * pvTaskGetThreadLocalStoragePointer ( TaskHandle_t xTaskToQuery ,
BaseType_t xIndex )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
void * pvReturn = NULL ;
TCB_t * pxTCB ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
if ( xIndex < configNUM_THREAD_LOCAL_STORAGE_POINTERS )
{
pxTCB = prvGetTCBFromHandle ( xTaskToQuery ) ;
pvReturn = pxTCB - > pvThreadLocalStoragePointers [ xIndex ] ;
}
else
{
pvReturn = NULL ;
}
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
return pvReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configNUM_THREAD_LOCAL_STORAGE_POINTERS */
2019-11-28 15:27:47 -03:00
/*-----------------------------------------------------------*/
2016-08-17 23:08:22 +08:00
# if ( portUSING_MPU_WRAPPERS == 1 )
2019-11-28 15:27:47 -03:00
2021-08-26 13:45:51 +08:00
void vTaskAllocateMPURegions ( TaskHandle_t xTaskToModify ,
const MemoryRegion_t * const xRegions )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* If null is passed in here then we are modifying the MPU settings of
2021-08-26 13:45:51 +08:00
* the calling task . */
2021-08-10 05:13:43 +08:00
pxTCB = prvGetTCBFromHandle ( xTaskToModify ) ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
vPortStoreTaskMPUSettings ( & ( pxTCB - > xMPUSettings ) , xRegions , NULL , 0 ) ;
}
2016-08-17 23:08:22 +08:00
# endif /* portUSING_MPU_WRAPPERS */
/*-----------------------------------------------------------*/
static void prvInitialiseTaskLists ( void )
{
2021-08-26 13:45:51 +08:00
UBaseType_t uxPriority ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
for ( uxPriority = ( UBaseType_t ) 0U ; uxPriority < ( UBaseType_t ) configMAX_PRIORITIES ; uxPriority + + )
{
vListInitialise ( & ( pxReadyTasksLists [ uxPriority ] ) ) ;
}
vListInitialise ( & xDelayedTaskList1 ) ;
vListInitialise ( & xDelayedTaskList2 ) ;
2022-08-01 17:29:03 +08:00
for ( BaseType_t x = 0 ; x < configNUM_CORES ; x + + )
{
vListInitialise ( & xPendingReadyList [ x ] ) ;
2021-08-10 05:13:43 +08:00
}
# if ( INCLUDE_vTaskDelete == 1 )
2021-08-26 13:45:51 +08:00
{
vListInitialise ( & xTasksWaitingTermination ) ;
}
2021-08-10 05:13:43 +08:00
# endif /* INCLUDE_vTaskDelete */
# if ( INCLUDE_vTaskSuspend == 1 )
2021-08-26 13:45:51 +08:00
{
vListInitialise ( & xSuspendedTaskList ) ;
}
2021-08-10 05:13:43 +08:00
# endif /* INCLUDE_vTaskSuspend */
/* Start with pxDelayedTaskList using list1 and the pxOverflowDelayedTaskList
2021-08-26 13:45:51 +08:00
* using list2 . */
2021-08-10 05:13:43 +08:00
pxDelayedTaskList = & xDelayedTaskList1 ;
pxOverflowDelayedTaskList = & xDelayedTaskList2 ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2022-12-14 21:07:48 +08:00
2016-08-17 23:08:22 +08:00
static void prvCheckTasksWaitingTermination ( void )
{
2021-08-10 05:13:43 +08:00
/** THIS FUNCTION IS CALLED FROM THE RTOS IDLE TASK **/
# if ( INCLUDE_vTaskDelete == 1 )
{
2022-12-14 21:07:48 +08:00
TCB_t * pxTCB ;
2021-08-10 05:13:43 +08:00
2022-12-14 21:07:48 +08:00
/* uxDeletedTasksWaitingCleanUp is used to prevent taskENTER_CRITICAL()
2021-08-26 13:45:51 +08:00
* being called too often in the idle task . */
while ( uxDeletedTasksWaitingCleanUp > ( UBaseType_t ) 0U )
2021-08-10 05:13:43 +08:00
{
2022-12-14 21:07:48 +08:00
# if ( configNUM_CORES > 1 )
pxTCB = NULL ;
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-26 13:45:51 +08:00
{
2022-12-14 21:07:48 +08:00
/* List may have already been cleared by the other core. Check again */
if ( listLIST_IS_EMPTY ( & xTasksWaitingTermination ) = = pdFALSE )
2022-11-02 19:49:40 +08:00
{
2022-12-14 21:07:48 +08:00
/* We can't delete a task if it is still running on
* the other core . Keep walking the list until we
* find a task we can free , or until we walk the
* entire list . */
ListItem_t * xEntry ;
for ( xEntry = listGET_HEAD_ENTRY ( & xTasksWaitingTermination ) ; xEntry ! = listGET_END_MARKER ( & xTasksWaitingTermination ) ; xEntry = listGET_NEXT ( xEntry ) )
2022-11-02 19:49:40 +08:00
{
2022-12-14 21:07:48 +08:00
if ( ! taskIS_CURRENTLY_RUNNING ( ( ( TCB_t * ) listGET_LIST_ITEM_OWNER ( xEntry ) ) ) )
{
pxTCB = ( TCB_t * ) listGET_LIST_ITEM_OWNER ( xEntry ) ;
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
- - uxCurrentNumberOfTasks ;
- - uxDeletedTasksWaitingCleanUp ;
break ;
}
2021-08-26 13:45:51 +08:00
}
2021-08-10 05:13:43 +08:00
}
2022-12-14 21:07:48 +08:00
}
taskEXIT_CRITICAL ( & xKernelLock ) ;
2022-11-02 19:49:40 +08:00
2022-12-14 21:07:48 +08:00
if ( pxTCB ! = NULL )
{
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS > 0 ) && ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 )
prvDeleteTLS ( pxTCB ) ;
# endif
prvDeleteTCB ( pxTCB ) ;
2021-08-10 05:13:43 +08:00
}
2022-12-14 21:07:48 +08:00
else
{
/* No task found to delete, break out of loop */
break ;
}
# else
taskENTER_CRITICAL ( & xKernelLock ) ;
{
pxTCB = listGET_OWNER_OF_HEAD_ENTRY ( ( & xTasksWaitingTermination ) ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
- - uxCurrentNumberOfTasks ;
- - uxDeletedTasksWaitingCleanUp ;
}
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
2022-11-08 17:48:48 +08:00
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS > 0 ) && ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 )
2021-08-26 13:45:51 +08:00
prvDeleteTLS ( pxTCB ) ;
# endif
prvDeleteTCB ( pxTCB ) ;
2022-12-14 21:07:48 +08:00
# endif /* configNUM_CORES > 1 */
2021-08-10 05:13:43 +08:00
}
}
# endif /* INCLUDE_vTaskDelete */
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2021-08-26 13:45:51 +08:00
# if ( configUSE_TRACE_FACILITY == 1 )
2016-08-17 23:08:22 +08:00
2021-08-26 13:45:51 +08:00
void vTaskGetInfo ( TaskHandle_t xTask ,
TaskStatus_t * pxTaskStatus ,
BaseType_t xGetFreeStackSpace ,
eTaskState eState )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
2021-08-10 05:13:43 +08:00
/* xTask is NULL then get the state of the calling task. */
pxTCB = prvGetTCBFromHandle ( xTask ) ;
pxTaskStatus - > xHandle = ( TaskHandle_t ) pxTCB ;
2021-08-26 13:45:51 +08:00
pxTaskStatus - > pcTaskName = ( const char * ) & ( pxTCB - > pcTaskName [ 0 ] ) ;
2021-08-10 05:13:43 +08:00
pxTaskStatus - > uxCurrentPriority = pxTCB - > uxPriority ;
pxTaskStatus - > pxStackBase = pxTCB - > pxStack ;
pxTaskStatus - > xTaskNumber = pxTCB - > uxTCBNumber ;
# if ( configTASKLIST_INCLUDE_COREID == 1 )
2022-09-05 17:31:22 +08:00
pxTaskStatus - > xCoreID = pxTCB - > xCoreID ;
2021-08-10 05:13:43 +08:00
# endif /* configTASKLIST_INCLUDE_COREID */
# if ( configUSE_MUTEXES == 1 )
2021-08-26 13:45:51 +08:00
{
pxTaskStatus - > uxBasePriority = pxTCB - > uxBasePriority ;
}
2021-08-10 05:13:43 +08:00
# else
2021-08-26 13:45:51 +08:00
{
pxTaskStatus - > uxBasePriority = 0 ;
}
2021-08-10 05:13:43 +08:00
# endif
# if ( configGENERATE_RUN_TIME_STATS == 1 )
2021-08-26 13:45:51 +08:00
{
pxTaskStatus - > ulRunTimeCounter = pxTCB - > ulRunTimeCounter ;
}
2021-08-10 05:13:43 +08:00
# else
2021-08-26 13:45:51 +08:00
{
pxTaskStatus - > ulRunTimeCounter = 0 ;
}
2021-08-10 05:13:43 +08:00
# endif
/* Obtaining the task state is a little fiddly, so is only done if the
2021-08-26 13:45:51 +08:00
* value of eState passed into this function is eInvalid - otherwise the
* state is just set to whatever is passed in . */
2021-08-10 05:13:43 +08:00
if ( eState ! = eInvalid )
{
2022-11-02 19:49:40 +08:00
if ( pxTCB = = pxCurrentTCB [ xPortGetCoreID ( ) ] )
2021-08-10 05:13:43 +08:00
{
pxTaskStatus - > eCurrentState = eRunning ;
}
else
{
pxTaskStatus - > eCurrentState = eState ;
# if ( INCLUDE_vTaskSuspend == 1 )
{
2021-08-26 13:45:51 +08:00
/* If the task is in the suspended list then there is a
* chance it is actually just blocked indefinitely - so really
* it should be reported as being in the Blocked state . */
if ( eState = = eSuspended )
2021-08-10 05:13:43 +08:00
{
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_SUSPEND_ALL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
if ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) ! = NULL )
{
pxTaskStatus - > eCurrentState = eBlocked ;
}
2021-08-10 05:13:43 +08:00
}
2022-11-14 15:27:00 +08:00
( void ) prvEXIT_CRITICAL_OR_RESUME_ALL ( & xKernelLock ) ;
2021-08-26 13:45:51 +08:00
}
2021-08-10 05:13:43 +08:00
}
# endif /* INCLUDE_vTaskSuspend */
}
}
else
{
pxTaskStatus - > eCurrentState = eTaskGetState ( pxTCB ) ;
}
/* Obtaining the stack space takes some time, so the xGetFreeStackSpace
2021-08-26 13:45:51 +08:00
* parameter is provided to allow it to be skipped . */
2021-08-10 05:13:43 +08:00
if ( xGetFreeStackSpace ! = pdFALSE )
{
# if ( portSTACK_GROWTH > 0 )
2021-08-26 13:45:51 +08:00
{
pxTaskStatus - > usStackHighWaterMark = prvTaskCheckFreeStackSpace ( ( uint8_t * ) pxTCB - > pxEndOfStack ) ;
}
2021-08-10 05:13:43 +08:00
# else
2021-08-26 13:45:51 +08:00
{
pxTaskStatus - > usStackHighWaterMark = prvTaskCheckFreeStackSpace ( ( uint8_t * ) pxTCB - > pxStack ) ;
}
2021-08-10 05:13:43 +08:00
# endif
}
else
{
pxTaskStatus - > usStackHighWaterMark = 0 ;
}
}
2019-11-28 15:27:47 -03:00
# endif /* configUSE_TRACE_FACILITY */
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
2016-09-28 17:02:44 +08:00
BaseType_t xTaskGetAffinity ( TaskHandle_t xTask )
{
2022-11-02 19:49:40 +08:00
TCB_t * pxTCB ;
2016-09-28 17:02:44 +08:00
2021-08-10 05:13:43 +08:00
pxTCB = prvGetTCBFromHandle ( xTask ) ;
2016-09-28 17:02:44 +08:00
2021-08-10 05:13:43 +08:00
return pxTCB - > xCoreID ;
2016-09-28 17:02:44 +08:00
}
/*-----------------------------------------------------------*/
2016-08-17 23:08:22 +08:00
# if ( configUSE_TRACE_FACILITY == 1 )
2021-08-26 13:45:51 +08:00
static UBaseType_t prvListTasksWithinSingleList ( TaskStatus_t * pxTaskStatusArray ,
List_t * pxList ,
eTaskState eState )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
configLIST_VOLATILE TCB_t * pxNextTCB , * pxFirstTCB ;
UBaseType_t uxTask = 0 ;
2021-08-10 05:13:43 +08:00
if ( listCURRENT_LIST_LENGTH ( pxList ) > ( UBaseType_t ) 0 )
{
listGET_OWNER_OF_NEXT_ENTRY ( pxFirstTCB , pxList ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
/* Populate an TaskStatus_t structure within the
2021-08-26 13:45:51 +08:00
* pxTaskStatusArray array for each task that is referenced from
* pxList . See the definition of TaskStatus_t in task . h for the
* meaning of each TaskStatus_t structure member . */
2021-08-10 05:13:43 +08:00
do
{
listGET_OWNER_OF_NEXT_ENTRY ( pxNextTCB , pxList ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
vTaskGetInfo ( ( TaskHandle_t ) pxNextTCB , & ( pxTaskStatusArray [ uxTask ] ) , pdTRUE , eState ) ;
uxTask + + ;
} while ( pxNextTCB ! = pxFirstTCB ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
return uxTask ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TRACE_FACILITY */
/*-----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
# if ( ( configUSE_TRACE_FACILITY == 1 ) || ( INCLUDE_uxTaskGetStackHighWaterMark == 1 ) || ( INCLUDE_uxTaskGetStackHighWaterMark2 == 1 ) )
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
static configSTACK_DEPTH_TYPE prvTaskCheckFreeStackSpace ( const uint8_t * pucStackByte )
{
2021-08-26 13:45:51 +08:00
uint32_t ulCount = 0U ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
while ( * pucStackByte = = ( uint8_t ) tskSTACK_FILL_BYTE )
{
pucStackByte - = portSTACK_GROWTH ;
ulCount + + ;
}
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
ulCount / = ( uint32_t ) sizeof ( StackType_t ) ; /*lint !e961 Casting is not redundant on smaller architectures. */
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
return ( configSTACK_DEPTH_TYPE ) ulCount ;
}
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
# endif /* ( ( configUSE_TRACE_FACILITY == 1 ) || ( INCLUDE_uxTaskGetStackHighWaterMark == 1 ) || ( INCLUDE_uxTaskGetStackHighWaterMark2 == 1 ) ) */
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
# if ( INCLUDE_uxTaskGetStackHighWaterMark2 == 1 )
2016-08-17 23:08:22 +08:00
2021-08-26 13:45:51 +08:00
/* uxTaskGetStackHighWaterMark() and uxTaskGetStackHighWaterMark2() are the
* same except for their return type . Using configSTACK_DEPTH_TYPE allows the
* user to determine the return type . It gets around the problem of the value
* overflowing on 8 - bit types without breaking backward compatibility for
* applications that expect an 8 - bit return type . */
2021-08-10 05:13:43 +08:00
configSTACK_DEPTH_TYPE uxTaskGetStackHighWaterMark2 ( TaskHandle_t xTask )
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
uint8_t * pucEndOfStack ;
configSTACK_DEPTH_TYPE uxReturn ;
2021-08-10 05:13:43 +08:00
/* uxTaskGetStackHighWaterMark() and uxTaskGetStackHighWaterMark2() are
2021-08-26 13:45:51 +08:00
* the same except for their return type . Using configSTACK_DEPTH_TYPE
* allows the user to determine the return type . It gets around the
* problem of the value overflowing on 8 - bit types without breaking
* backward compatibility for applications that expect an 8 - bit return
* type . */
2021-08-10 05:13:43 +08:00
pxTCB = prvGetTCBFromHandle ( xTask ) ;
# if portSTACK_GROWTH < 0
2021-08-26 13:45:51 +08:00
{
pucEndOfStack = ( uint8_t * ) pxTCB - > pxStack ;
}
2021-08-10 05:13:43 +08:00
# else
2021-08-26 13:45:51 +08:00
{
pucEndOfStack = ( uint8_t * ) pxTCB - > pxEndOfStack ;
}
2021-08-10 05:13:43 +08:00
# endif
uxReturn = prvTaskCheckFreeStackSpace ( pucEndOfStack ) ;
return uxReturn ;
}
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
# endif /* INCLUDE_uxTaskGetStackHighWaterMark2 */
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
# if ( INCLUDE_uxTaskGetStackHighWaterMark == 1 )
2017-11-17 13:05:54 +01:00
2021-08-10 05:13:43 +08:00
UBaseType_t uxTaskGetStackHighWaterMark ( TaskHandle_t xTask )
{
2021-08-26 13:45:51 +08:00
TCB_t * pxTCB ;
uint8_t * pucEndOfStack ;
UBaseType_t uxReturn ;
2017-11-17 13:05:54 +01:00
2021-08-10 05:13:43 +08:00
pxTCB = prvGetTCBFromHandle ( xTask ) ;
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
# if portSTACK_GROWTH < 0
2021-08-26 13:45:51 +08:00
{
pucEndOfStack = ( uint8_t * ) pxTCB - > pxStack ;
}
2021-08-10 05:13:43 +08:00
# else
2021-08-26 13:45:51 +08:00
{
pucEndOfStack = ( uint8_t * ) pxTCB - > pxEndOfStack ;
}
2021-08-10 05:13:43 +08:00
# endif
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
uxReturn = ( UBaseType_t ) prvTaskCheckFreeStackSpace ( pucEndOfStack ) ;
2017-11-17 13:05:54 +01:00
2021-08-10 05:13:43 +08:00
return uxReturn ;
}
2017-03-22 06:07:37 +03:00
2019-11-28 15:27:47 -03:00
# endif /* INCLUDE_uxTaskGetStackHighWaterMark */
2017-03-22 06:07:37 +03:00
/*-----------------------------------------------------------*/
2022-11-02 19:49:40 +08:00
# if ( INCLUDE_pxTaskGetStackStart == 1 )
2020-09-15 17:47:47 -03:00
2022-11-02 19:49:40 +08:00
uint8_t * pxTaskGetStackStart ( TaskHandle_t xTask )
2021-08-10 05:13:43 +08:00
{
2022-11-02 19:49:40 +08:00
TCB_t * pxTCB ;
uint8_t * uxReturn ;
2020-09-15 17:47:47 -03:00
2021-08-10 05:13:43 +08:00
pxTCB = prvGetTCBFromHandle ( xTask ) ;
2022-11-02 19:49:40 +08:00
uxReturn = ( uint8_t * ) pxTCB - > pxStack ;
2020-09-15 17:47:47 -03:00
2021-08-10 05:13:43 +08:00
return uxReturn ;
}
2020-09-15 17:47:47 -03:00
# endif /* INCLUDE_pxTaskGetStackStart */
2017-03-22 06:07:37 +03:00
2016-08-17 23:08:22 +08:00
# if ( INCLUDE_vTaskDelete == 1 )
2021-08-26 13:45:51 +08:00
static void prvDeleteTCB ( TCB_t * pxTCB )
2021-08-10 05:13:43 +08:00
{
/* This call is required specifically for the TriCore port. It must be
2021-08-26 13:45:51 +08:00
* above the vPortFree ( ) calls . The call is also used by ports / demos that
* want to allocate and clean RAM statically . */
2021-08-10 05:13:43 +08:00
portCLEAN_UP_TCB ( pxTCB ) ;
/* Free up the memory allocated by the scheduler for the task. It is up
2021-08-26 13:45:51 +08:00
* to the task to free any memory allocated at the application level .
* See the third party link http : //www.nadler.com/embedded/newlibAndFreeRTOS.html
* for additional information . */
2021-08-10 05:13:43 +08:00
# if ( configUSE_NEWLIB_REENTRANT == 1 )
2021-08-26 13:45:51 +08:00
{
_reclaim_reent ( & ( pxTCB - > xNewLib_reent ) ) ;
}
2021-08-10 05:13:43 +08:00
# endif /* configUSE_NEWLIB_REENTRANT */
# if ( portUSING_MPU_WRAPPERS == 1 )
2022-11-02 19:49:40 +08:00
vPortReleaseTaskMPUSettings ( & ( pxTCB - > xMPUSettings ) ) ;
2021-08-10 05:13:43 +08:00
# endif
2022-12-14 20:44:13 +08:00
# ifdef portCLEAN_UP_COPROC
portCLEAN_UP_COPROC ( ( void * ) pxTCB ) ;
# endif
2021-08-26 13:45:51 +08:00
# if ( ( configSUPPORT_DYNAMIC_ALLOCATION == 1 ) && ( configSUPPORT_STATIC_ALLOCATION == 0 ) && ( portUSING_MPU_WRAPPERS == 0 ) )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
/* The task can only have been allocated dynamically - free both
* the stack and TCB . */
2021-08-10 05:13:43 +08:00
vPortFree ( pxTCB - > pxStack ) ;
vPortFree ( pxTCB ) ;
}
2021-08-26 13:45:51 +08:00
# elif ( tskSTATIC_AND_DYNAMIC_ALLOCATION_POSSIBLE != 0 ) /*lint !e731 !e9029 Macro has been consolidated for readability reasons. */
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
/* The task could have been allocated statically or dynamically, so
* check what was statically allocated before trying to free the
* memory . */
if ( pxTCB - > ucStaticallyAllocated = = tskDYNAMICALLY_ALLOCATED_STACK_AND_TCB )
{
/* Both the stack and TCB were allocated dynamically, so both
* must be freed . */
vPortFree ( pxTCB - > pxStack ) ;
vPortFree ( pxTCB ) ;
}
else if ( pxTCB - > ucStaticallyAllocated = = tskSTATICALLY_ALLOCATED_STACK_ONLY )
{
/* Only the stack was statically allocated, so the TCB is the
* only memory that must be freed . */
vPortFree ( pxTCB ) ;
}
else
{
/* Neither the stack nor the TCB were allocated dynamically, so
* nothing needs to be freed . */
configASSERT ( pxTCB - > ucStaticallyAllocated = = tskSTATICALLY_ALLOCATED_STACK_AND_TCB ) ;
mtCOVERAGE_TEST_MARKER ( ) ;
}
2021-08-10 05:13:43 +08:00
}
# endif /* configSUPPORT_DYNAMIC_ALLOCATION */
}
2016-08-17 23:08:22 +08:00
# endif /* INCLUDE_vTaskDelete */
/*-----------------------------------------------------------*/
2022-11-08 17:48:48 +08:00
# if ( configNUM_THREAD_LOCAL_STORAGE_POINTERS > 0 ) && ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 )
2017-11-23 22:35:54 +08:00
2022-11-02 19:49:40 +08:00
static void prvDeleteTLS ( TCB_t * pxTCB )
2021-08-10 05:13:43 +08:00
{
configASSERT ( pxTCB ) ;
2022-11-02 19:49:40 +08:00
2021-08-10 05:13:43 +08:00
for ( int x = 0 ; x < configNUM_THREAD_LOCAL_STORAGE_POINTERS ; x + + )
{
2022-11-02 19:49:40 +08:00
if ( pxTCB - > pvThreadLocalStoragePointersDelCallback [ x ] ! = NULL ) /*If del cb is set */
2021-08-10 05:13:43 +08:00
{
2022-11-02 19:49:40 +08:00
pxTCB - > pvThreadLocalStoragePointersDelCallback [ x ] ( x , pxTCB - > pvThreadLocalStoragePointers [ x ] ) ; /*Call del cb */
2021-08-10 05:13:43 +08:00
}
}
}
2017-11-23 22:35:54 +08:00
2022-11-08 17:48:48 +08:00
# endif /* ( configNUM_THREAD_LOCAL_STORAGE_POINTERS > 0 ) && ( configTHREAD_LOCAL_STORAGE_DELETE_CALLBACKS == 1 ) */
2017-11-23 22:35:54 +08:00
/*-----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
2016-08-17 23:08:22 +08:00
static void prvResetNextTaskUnblockTime ( void )
{
2021-09-07 10:30:30 +08:00
TCB_t * pxTCB ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
if ( listLIST_IS_EMPTY ( pxDelayedTaskList ) ! = pdFALSE )
{
/* The new current delayed list is empty. Set xNextTaskUnblockTime to
2021-08-26 13:45:51 +08:00
* the maximum possible value so it is extremely unlikely that the
* if ( xTickCount > = xNextTaskUnblockTime ) test will pass until
* there is an item in the delayed list . */
2021-08-10 05:13:43 +08:00
xNextTaskUnblockTime = portMAX_DELAY ;
}
else
{
/* The new current delayed list is not empty, get the value of
2021-08-26 13:45:51 +08:00
* the item at the head of the delayed list . This is the time at
* which the task at the head of the delayed list should be removed
* from the Blocked state . */
2021-08-10 05:13:43 +08:00
( pxTCB ) = listGET_OWNER_OF_HEAD_ENTRY ( pxDelayedTaskList ) ; /*lint !e9079 void * is used as this macro is used with timers and co-routines too. Alignment is known to be fine as the type of the pointer stored and retrieved is the same. */
xNextTaskUnblockTime = listGET_LIST_ITEM_VALUE ( & ( ( pxTCB ) - > xStateListItem ) ) ;
}
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
2022-11-02 19:49:40 +08:00
# if ( ( INCLUDE_xTaskGetCurrentTaskHandle == 1 ) || ( configUSE_MUTEXES == 1 ) || ( configNUM_CORES > 1 ) )
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
TaskHandle_t xTaskGetCurrentTaskHandle ( void )
{
2021-08-26 13:45:51 +08:00
TaskHandle_t xReturn ;
unsigned state ;
2016-08-17 23:08:22 +08:00
2021-10-16 00:14:27 +08:00
state = portSET_INTERRUPT_MASK_FROM_ISR ( ) ;
2021-08-10 05:13:43 +08:00
xReturn = pxCurrentTCB [ xPortGetCoreID ( ) ] ;
2022-11-02 19:49:40 +08:00
portCLEAR_INTERRUPT_MASK_FROM_ISR ( state ) ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
TaskHandle_t xTaskGetCurrentTaskHandleForCPU ( BaseType_t cpuid )
{
2022-11-02 19:49:40 +08:00
TaskHandle_t xReturn = NULL ;
2016-10-25 18:08:55 +08:00
2022-11-02 19:49:40 +08:00
/*Xtensa-specific: the pxCurrentPCB pointer is atomic so we shouldn't need a lock. */
if ( cpuid < configNUM_CORES )
{
2021-08-10 05:13:43 +08:00
xReturn = pxCurrentTCB [ cpuid ] ;
}
2016-10-25 18:08:55 +08:00
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2019-11-28 15:27:47 -03:00
2016-08-17 23:08:22 +08:00
# endif /* ( ( INCLUDE_xTaskGetCurrentTaskHandle == 1 ) || ( configUSE_MUTEXES == 1 ) ) */
/*-----------------------------------------------------------*/
# if ( ( INCLUDE_xTaskGetSchedulerState == 1 ) || ( configUSE_TIMERS == 1 ) )
2021-08-10 05:13:43 +08:00
BaseType_t xTaskGetSchedulerState ( void )
{
2021-08-26 13:45:51 +08:00
BaseType_t xReturn ;
2021-11-24 14:34:40 +05:30
unsigned state ;
2021-08-10 05:13:43 +08:00
2022-09-05 17:31:22 +08:00
/* Known issue. This should use critical sections. See IDF-5889 */
2021-11-24 14:34:40 +05:30
state = portSET_INTERRUPT_MASK_FROM_ISR ( ) ;
2022-11-02 19:49:40 +08:00
2021-08-10 05:13:43 +08:00
if ( xSchedulerRunning = = pdFALSE )
{
xReturn = taskSCHEDULER_NOT_STARTED ;
}
else
{
2022-11-02 19:49:40 +08:00
if ( uxSchedulerSuspended [ xPortGetCoreID ( ) ] = = ( UBaseType_t ) pdFALSE )
2021-08-10 05:13:43 +08:00
{
xReturn = taskSCHEDULER_RUNNING ;
}
else
{
xReturn = taskSCHEDULER_SUSPENDED ;
}
}
2022-11-02 19:49:40 +08:00
portCLEAR_INTERRUPT_MASK_FROM_ISR ( state ) ;
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* ( ( INCLUDE_xTaskGetSchedulerState == 1 ) || ( configUSE_TIMERS == 1 ) ) */
/*-----------------------------------------------------------*/
# if ( configUSE_MUTEXES == 1 )
2021-08-10 05:13:43 +08:00
BaseType_t xTaskPriorityInherit ( TaskHandle_t const pxMutexHolder )
{
2021-08-26 13:45:51 +08:00
TCB_t * const pxMutexHolderTCB = pxMutexHolder ;
BaseType_t xReturn = pdFALSE ;
2021-08-10 05:13:43 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* For SMP, we need to take the kernel lock here as we are about to
* access kernel data structures . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2022-11-02 19:49:40 +08:00
2021-08-10 05:13:43 +08:00
/* If the mutex was given back by an interrupt while the queue was
2021-08-26 13:45:51 +08:00
* locked then the mutex holder might now be NULL . _RB_ Is this still
* needed as interrupts can no longer use mutexes ? */
2021-08-10 05:13:43 +08:00
if ( pxMutexHolder ! = NULL )
{
/* If the holder of the mutex has a priority below the priority of
2021-08-26 13:45:51 +08:00
* the task attempting to obtain the mutex then it will temporarily
* inherit the priority of the task attempting to obtain the mutex . */
2022-11-02 19:49:40 +08:00
if ( pxMutexHolderTCB - > uxPriority < pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxPriority )
2021-08-10 05:13:43 +08:00
{
/* Adjust the mutex holder state to account for its new
2021-08-26 13:45:51 +08:00
* priority . Only reset the event list item value if the value is
* not being used for anything else . */
2021-08-10 05:13:43 +08:00
if ( ( listGET_LIST_ITEM_VALUE ( & ( pxMutexHolderTCB - > xEventListItem ) ) & taskEVENT_LIST_ITEM_VALUE_IN_USE ) = = 0UL )
{
2022-11-02 19:49:40 +08:00
listSET_LIST_ITEM_VALUE ( & ( pxMutexHolderTCB - > xEventListItem ) , ( TickType_t ) configMAX_PRIORITIES - ( TickType_t ) pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxPriority ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* If the task being modified is in the ready state it will need
2021-08-26 13:45:51 +08:00
* to be moved into a new list . */
2021-08-10 05:13:43 +08:00
if ( listIS_CONTAINED_WITHIN ( & ( pxReadyTasksLists [ pxMutexHolderTCB - > uxPriority ] ) , & ( pxMutexHolderTCB - > xStateListItem ) ) ! = pdFALSE )
{
if ( uxListRemove ( & ( pxMutexHolderTCB - > xStateListItem ) ) = = ( UBaseType_t ) 0 )
{
/* It is known that the task is in its ready list so
2021-08-26 13:45:51 +08:00
* there is no need to check again and the port level
* reset macro can be called directly . */
2021-08-10 05:13:43 +08:00
portRESET_READY_PRIORITY ( pxMutexHolderTCB - > uxPriority , uxTopReadyPriority ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* Inherit the priority before being moved into the new list. */
2022-11-02 19:49:40 +08:00
pxMutexHolderTCB - > uxPriority = pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxPriority ;
2021-08-10 05:13:43 +08:00
prvAddTaskToReadyList ( pxMutexHolderTCB ) ;
}
else
{
/* Just inherit the priority. */
2022-11-02 19:49:40 +08:00
pxMutexHolderTCB - > uxPriority = pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxPriority ;
2021-08-10 05:13:43 +08:00
}
2022-11-02 19:49:40 +08:00
traceTASK_PRIORITY_INHERIT ( pxMutexHolderTCB , pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxPriority ) ;
2021-08-10 05:13:43 +08:00
/* Inheritance occurred. */
xReturn = pdTRUE ;
}
else
{
2022-11-02 19:49:40 +08:00
if ( pxMutexHolderTCB - > uxBasePriority < pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxPriority )
2021-08-10 05:13:43 +08:00
{
/* The base priority of the mutex holder is lower than the
2021-08-26 13:45:51 +08:00
* priority of the task attempting to take the mutex , but the
* current priority of the mutex holder is not lower than the
* priority of the task attempting to take the mutex .
* Therefore the mutex holder must have already inherited a
* priority , but inheritance would have occurred if that had
* not been the case . */
2021-08-10 05:13:43 +08:00
xReturn = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-11-02 19:49:40 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2019-11-28 15:27:47 -03:00
# endif /* configUSE_MUTEXES */
/*-----------------------------------------------------------*/
# if ( configUSE_MUTEXES == 1 )
2021-08-10 05:13:43 +08:00
BaseType_t xTaskPriorityDisinherit ( TaskHandle_t const pxMutexHolder )
{
2021-08-26 13:45:51 +08:00
TCB_t * const pxTCB = pxMutexHolder ;
BaseType_t xReturn = pdFALSE ;
2021-08-10 05:13:43 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* For SMP, we need to take the kernel lock here as we are about to
* access kernel data structures . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2022-11-02 19:49:40 +08:00
2021-08-10 05:13:43 +08:00
if ( pxMutexHolder ! = NULL )
{
/* A task can only have an inherited priority if it holds the mutex.
2021-08-26 13:45:51 +08:00
* If the mutex is held by a task then it cannot be given from an
* interrupt , and if a mutex is given by the holding task then it must
* be the running state task . */
2022-11-02 19:49:40 +08:00
configASSERT ( pxTCB = = pxCurrentTCB [ xPortGetCoreID ( ) ] ) ;
2021-08-10 05:13:43 +08:00
configASSERT ( pxTCB - > uxMutexesHeld ) ;
( pxTCB - > uxMutexesHeld ) - - ;
/* Has the holder of the mutex inherited the priority of another
2021-08-26 13:45:51 +08:00
* task ? */
2021-08-10 05:13:43 +08:00
if ( pxTCB - > uxPriority ! = pxTCB - > uxBasePriority )
{
/* Only disinherit if no other mutexes are held. */
if ( pxTCB - > uxMutexesHeld = = ( UBaseType_t ) 0 )
{
/* A task can only have an inherited priority if it holds
2021-08-26 13:45:51 +08:00
* the mutex . If the mutex is held by a task then it cannot be
* given from an interrupt , and if a mutex is given by the
* holding task then it must be the running state task . Remove
* the holding task from the ready list . */
2021-08-10 05:13:43 +08:00
if ( uxListRemove ( & ( pxTCB - > xStateListItem ) ) = = ( UBaseType_t ) 0 )
{
taskRESET_READY_PRIORITY ( pxTCB - > uxPriority ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* Disinherit the priority before adding the task into the
2021-08-26 13:45:51 +08:00
* new ready list . */
2021-08-10 05:13:43 +08:00
traceTASK_PRIORITY_DISINHERIT ( pxTCB , pxTCB - > uxBasePriority ) ;
pxTCB - > uxPriority = pxTCB - > uxBasePriority ;
/* Reset the event list item value. It cannot be in use for
2021-08-26 13:45:51 +08:00
* any other purpose if this task is running , and it must be
* running to give back the mutex . */
2021-08-10 05:13:43 +08:00
listSET_LIST_ITEM_VALUE ( & ( pxTCB - > xEventListItem ) , ( TickType_t ) configMAX_PRIORITIES - ( TickType_t ) pxTCB - > uxPriority ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
prvAddTaskToReadyList ( pxTCB ) ;
/* Return true to indicate that a context switch is required.
2021-08-26 13:45:51 +08:00
* This is only actually required in the corner case whereby
* multiple mutexes were held and the mutexes were given back
* in an order different to that in which they were taken .
* If a context switch did not occur when the first mutex was
* returned , even if a task was waiting on it , then a context
* switch should occur when the last mutex is returned whether
* a task is waiting on it or not . */
2021-08-10 05:13:43 +08:00
xReturn = pdTRUE ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-11-02 19:49:40 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_MUTEXES */
/*-----------------------------------------------------------*/
# if ( configUSE_MUTEXES == 1 )
2021-08-26 13:45:51 +08:00
void vTaskPriorityDisinheritAfterTimeout ( TaskHandle_t const pxMutexHolder ,
UBaseType_t uxHighestPriorityWaitingTask )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TCB_t * const pxTCB = pxMutexHolder ;
UBaseType_t uxPriorityUsedOnEntry , uxPriorityToUse ;
const UBaseType_t uxOnlyOneMutexHeld = ( UBaseType_t ) 1 ;
2021-08-10 05:13:43 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* For SMP, we need to take the kernel lock here as we are about to
* access kernel data structures . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2022-11-02 19:49:40 +08:00
2021-08-10 05:13:43 +08:00
if ( pxMutexHolder ! = NULL )
{
/* If pxMutexHolder is not NULL then the holder must hold at least
2021-08-26 13:45:51 +08:00
* one mutex . */
2021-08-10 05:13:43 +08:00
configASSERT ( pxTCB - > uxMutexesHeld ) ;
/* Determine the priority to which the priority of the task that
2021-08-26 13:45:51 +08:00
* holds the mutex should be set . This will be the greater of the
* holding task ' s base priority and the priority of the highest
* priority task that is waiting to obtain the mutex . */
2021-08-10 05:13:43 +08:00
if ( pxTCB - > uxBasePriority < uxHighestPriorityWaitingTask )
{
uxPriorityToUse = uxHighestPriorityWaitingTask ;
}
else
{
uxPriorityToUse = pxTCB - > uxBasePriority ;
}
/* Does the priority need to change? */
if ( pxTCB - > uxPriority ! = uxPriorityToUse )
{
/* Only disinherit if no other mutexes are held. This is a
2021-08-26 13:45:51 +08:00
* simplification in the priority inheritance implementation . If
* the task that holds the mutex is also holding other mutexes then
* the other mutexes may have caused the priority inheritance . */
2021-08-10 05:13:43 +08:00
if ( pxTCB - > uxMutexesHeld = = uxOnlyOneMutexHeld )
{
/* If a task has timed out because it already holds the
2021-08-26 13:45:51 +08:00
* mutex it was trying to obtain then it cannot of inherited
* its own priority . */
2022-11-02 19:49:40 +08:00
configASSERT ( pxTCB ! = pxCurrentTCB [ xPortGetCoreID ( ) ] ) ;
2021-08-10 05:13:43 +08:00
/* Disinherit the priority, remembering the previous
2021-08-26 13:45:51 +08:00
* priority to facilitate determining the subject task ' s
* state . */
2021-08-10 05:13:43 +08:00
traceTASK_PRIORITY_DISINHERIT ( pxTCB , pxTCB - > uxBasePriority ) ;
uxPriorityUsedOnEntry = pxTCB - > uxPriority ;
pxTCB - > uxPriority = uxPriorityToUse ;
/* Only reset the event list item value if the value is not
2021-08-26 13:45:51 +08:00
* being used for anything else . */
2021-08-10 05:13:43 +08:00
if ( ( listGET_LIST_ITEM_VALUE ( & ( pxTCB - > xEventListItem ) ) & taskEVENT_LIST_ITEM_VALUE_IN_USE ) = = 0UL )
{
listSET_LIST_ITEM_VALUE ( & ( pxTCB - > xEventListItem ) , ( TickType_t ) configMAX_PRIORITIES - ( TickType_t ) uxPriorityToUse ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* If the running task is not the task that holds the mutex
2021-08-26 13:45:51 +08:00
* then the task that holds the mutex could be in either the
* Ready , Blocked or Suspended states . Only remove the task
* from its current state list if it is in the Ready state as
* the task ' s priority is going to change and there is one
* Ready list per priority . */
2021-08-10 05:13:43 +08:00
if ( listIS_CONTAINED_WITHIN ( & ( pxReadyTasksLists [ uxPriorityUsedOnEntry ] ) , & ( pxTCB - > xStateListItem ) ) ! = pdFALSE )
{
if ( uxListRemove ( & ( pxTCB - > xStateListItem ) ) = = ( UBaseType_t ) 0 )
{
/* It is known that the task is in its ready list so
2021-08-26 13:45:51 +08:00
* there is no need to check again and the port level
* reset macro can be called directly . */
2021-08-10 05:13:43 +08:00
portRESET_READY_PRIORITY ( pxTCB - > uxPriority , uxTopReadyPriority ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
prvAddTaskToReadyList ( pxTCB ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-11-02 19:49:40 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_MUTEXES */
/*-----------------------------------------------------------*/
# if ( portCRITICAL_NESTING_IN_TCB == 1 )
2021-08-10 05:13:43 +08:00
void vTaskEnterCritical ( void )
{
portDISABLE_INTERRUPTS ( ) ;
if ( xSchedulerRunning ! = pdFALSE )
{
2022-11-02 19:49:40 +08:00
( pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxCriticalNesting ) + + ;
2021-08-10 05:13:43 +08:00
/* This is not the interrupt safe version of the enter critical
2021-08-26 13:45:51 +08:00
* function so assert ( ) if it is being called from an interrupt
* context . Only API functions that end in " FromISR " can be used in an
* interrupt . Only assert if the critical nesting count is 1 to
* protect against recursive calls if the assert function also uses a
* critical section . */
2022-11-02 19:49:40 +08:00
if ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxCriticalNesting = = 1 )
2021-08-10 05:13:43 +08:00
{
portASSERT_IF_IN_ISR ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2016-08-17 23:08:22 +08:00
# endif /* portCRITICAL_NESTING_IN_TCB */
/*-----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
2016-08-17 23:08:22 +08:00
# if ( portCRITICAL_NESTING_IN_TCB == 1 )
2021-08-10 05:13:43 +08:00
void vTaskExitCritical ( void )
{
if ( xSchedulerRunning ! = pdFALSE )
{
2022-11-02 19:49:40 +08:00
if ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxCriticalNesting > 0U )
2021-08-10 05:13:43 +08:00
{
2022-11-02 19:49:40 +08:00
( pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxCriticalNesting ) - - ;
2021-08-10 05:13:43 +08:00
2022-11-02 19:49:40 +08:00
if ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > uxCriticalNesting = = 0U )
2021-08-10 05:13:43 +08:00
{
portENABLE_INTERRUPTS ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2017-03-22 06:07:37 +03:00
2016-08-17 23:08:22 +08:00
# endif /* portCRITICAL_NESTING_IN_TCB */
/*-----------------------------------------------------------*/
# if ( ( configUSE_TRACE_FACILITY == 1 ) && ( configUSE_STATS_FORMATTING_FUNCTIONS > 0 ) )
2021-09-07 10:30:30 +08:00
static char * prvWriteNameToBuffer ( char * pcBuffer ,
const char * pcTaskName )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
size_t x ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* Start by copying the entire string. */
strcpy ( pcBuffer , pcTaskName ) ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* Pad the end of the string with spaces to ensure columns line up when
2021-08-26 13:45:51 +08:00
* printed out . */
2021-08-10 05:13:43 +08:00
for ( x = strlen ( pcBuffer ) ; x < ( size_t ) ( configMAX_TASK_NAME_LEN - 1 ) ; x + + )
{
pcBuffer [ x ] = ' ' ;
}
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* Terminate. */
pcBuffer [ x ] = ( char ) 0x00 ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* Return the new end of string. */
return & ( pcBuffer [ x ] ) ;
}
2016-08-17 23:08:22 +08:00
# endif /* ( configUSE_TRACE_FACILITY == 1 ) && ( configUSE_STATS_FORMATTING_FUNCTIONS > 0 ) */
/*-----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
# if ( ( configUSE_TRACE_FACILITY == 1 ) && ( configUSE_STATS_FORMATTING_FUNCTIONS > 0 ) && ( configSUPPORT_DYNAMIC_ALLOCATION == 1 ) )
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
void vTaskList ( char * pcWriteBuffer )
{
2021-08-26 13:45:51 +08:00
TaskStatus_t * pxTaskStatusArray ;
UBaseType_t uxArraySize , x ;
char cStatus ;
2021-08-10 05:13:43 +08:00
/*
* PLEASE NOTE :
*
* This function is provided for convenience only , and is used by many
* of the demo applications . Do not consider it to be part of the
* scheduler .
*
* vTaskList ( ) calls uxTaskGetSystemState ( ) , then formats part of the
* uxTaskGetSystemState ( ) output into a human readable table that
* displays task names , states and stack usage .
*
* vTaskList ( ) has a dependency on the sprintf ( ) C library function that
* might bloat the code size , use a lot of stack , and provide different
* results on different platforms . An alternative , tiny , third party ,
* and limited functionality implementation of sprintf ( ) is provided in
* many of the FreeRTOS / Demo sub - directories in a file called
* printf - stdarg . c ( note printf - stdarg . c does not provide a full
* snprintf ( ) implementation ! ) .
*
* It is recommended that production systems call uxTaskGetSystemState ( )
* directly to get access to raw stats data , rather than indirectly
* through a call to vTaskList ( ) .
*/
/* Make sure the write buffer does not contain a string. */
* pcWriteBuffer = ( char ) 0x00 ;
/* Take a snapshot of the number of tasks in case it changes while this
2021-08-26 13:45:51 +08:00
* function is executing . */
2021-08-10 05:13:43 +08:00
uxArraySize = uxCurrentNumberOfTasks ;
/* Allocate an array index for each task. NOTE! if
2021-08-26 13:45:51 +08:00
* configSUPPORT_DYNAMIC_ALLOCATION is set to 0 then pvPortMalloc ( ) will
* equate to NULL . */
2021-08-10 05:13:43 +08:00
pxTaskStatusArray = pvPortMalloc ( uxCurrentNumberOfTasks * sizeof ( TaskStatus_t ) ) ; /*lint !e9079 All values returned by pvPortMalloc() have at least the alignment required by the MCU's stack and this allocation allocates a struct that has the alignment requirements of a pointer. */
if ( pxTaskStatusArray ! = NULL )
{
/* Generate the (binary) data. */
uxArraySize = uxTaskGetSystemState ( pxTaskStatusArray , uxArraySize , NULL ) ;
/* Create a human readable table from the binary data. */
for ( x = 0 ; x < uxArraySize ; x + + )
{
switch ( pxTaskStatusArray [ x ] . eCurrentState )
{
2021-08-26 13:45:51 +08:00
case eRunning :
cStatus = tskRUNNING_CHAR ;
break ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
case eReady :
cStatus = tskREADY_CHAR ;
break ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
case eBlocked :
cStatus = tskBLOCKED_CHAR ;
break ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
case eSuspended :
cStatus = tskSUSPENDED_CHAR ;
break ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
case eDeleted :
cStatus = tskDELETED_CHAR ;
break ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
case eInvalid : /* Fall through. */
default : /* Should not get here, but it is included
* to prevent static checking errors . */
cStatus = ( char ) 0x00 ;
break ;
2021-08-10 05:13:43 +08:00
}
/* Write the task name to the string, padding with spaces so it
2021-08-26 13:45:51 +08:00
* can be printed in tabular form more easily . */
2021-08-10 05:13:43 +08:00
pcWriteBuffer = prvWriteNameToBuffer ( pcWriteBuffer , pxTaskStatusArray [ x ] . pcTaskName ) ;
/* Write the rest of the string. */
2022-11-02 19:49:40 +08:00
# if configTASKLIST_INCLUDE_COREID
sprintf ( pcWriteBuffer , " \t %c \t %u \t %u \t %u \t %hd \r \n " , cStatus , ( unsigned int ) pxTaskStatusArray [ x ] . uxCurrentPriority , ( unsigned int ) pxTaskStatusArray [ x ] . usStackHighWaterMark , ( unsigned int ) pxTaskStatusArray [ x ] . xTaskNumber , ( int ) pxTaskStatusArray [ x ] . xCoreID ) ;
# else
sprintf ( pcWriteBuffer , " \t %c \t %u \t %u \t %u \r \n " , cStatus , ( unsigned int ) pxTaskStatusArray [ x ] . uxCurrentPriority , ( unsigned int ) pxTaskStatusArray [ x ] . usStackHighWaterMark , ( unsigned int ) pxTaskStatusArray [ x ] . xTaskNumber ) ; /*lint !e586 sprintf() allowed as this is compiled with many compilers and this is a utility function only - not part of the core kernel implementation. */
# endif
pcWriteBuffer + = strlen ( pcWriteBuffer ) ; /*lint !e9016 Pointer arithmetic ok on char pointers especially as in this case where it best denotes the intent of the code. */
2021-08-10 05:13:43 +08:00
}
/* Free the array again. NOTE! If configSUPPORT_DYNAMIC_ALLOCATION
2021-08-26 13:45:51 +08:00
* is 0 then vPortFree ( ) will be # defined to nothing . */
2021-08-10 05:13:43 +08:00
vPortFree ( pxTaskStatusArray ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
# endif /* ( ( configUSE_TRACE_FACILITY == 1 ) && ( configUSE_STATS_FORMATTING_FUNCTIONS > 0 ) && ( configSUPPORT_DYNAMIC_ALLOCATION == 1 ) ) */
2016-08-17 23:08:22 +08:00
/*----------------------------------------------------------*/
2019-11-28 15:27:47 -03:00
# if ( ( configGENERATE_RUN_TIME_STATS == 1 ) && ( configUSE_STATS_FORMATTING_FUNCTIONS > 0 ) && ( configSUPPORT_DYNAMIC_ALLOCATION == 1 ) )
2016-08-17 23:08:22 +08:00
2021-08-26 13:45:51 +08:00
void vTaskGetRunTimeStats ( char * pcWriteBuffer )
2021-08-10 05:13:43 +08:00
{
2021-08-26 13:45:51 +08:00
TaskStatus_t * pxTaskStatusArray ;
UBaseType_t uxArraySize , x ;
uint32_t ulTotalTime , ulStatsAsPercentage ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
# if ( configUSE_TRACE_FACILITY != 1 )
{
# error configUSE_TRACE_FACILITY must also be set to 1 in FreeRTOSConfig.h to use vTaskGetRunTimeStats().
}
2021-08-10 05:13:43 +08:00
# endif
/*
* PLEASE NOTE :
*
* This function is provided for convenience only , and is used by many
* of the demo applications . Do not consider it to be part of the
* scheduler .
*
* vTaskGetRunTimeStats ( ) calls uxTaskGetSystemState ( ) , then formats part
* of the uxTaskGetSystemState ( ) output into a human readable table that
* displays the amount of time each task has spent in the Running state
* in both absolute and percentage terms .
*
* vTaskGetRunTimeStats ( ) has a dependency on the sprintf ( ) C library
* function that might bloat the code size , use a lot of stack , and
* provide different results on different platforms . An alternative ,
* tiny , third party , and limited functionality implementation of
* sprintf ( ) is provided in many of the FreeRTOS / Demo sub - directories in
* a file called printf - stdarg . c ( note printf - stdarg . c does not provide
* a full snprintf ( ) implementation ! ) .
*
* It is recommended that production systems call uxTaskGetSystemState ( )
* directly to get access to raw stats data , rather than indirectly
* through a call to vTaskGetRunTimeStats ( ) .
*/
/* Make sure the write buffer does not contain a string. */
* pcWriteBuffer = ( char ) 0x00 ;
/* Take a snapshot of the number of tasks in case it changes while this
2021-08-26 13:45:51 +08:00
* function is executing . */
2021-08-10 05:13:43 +08:00
uxArraySize = uxCurrentNumberOfTasks ;
/* Allocate an array index for each task. NOTE! If
2021-08-26 13:45:51 +08:00
* configSUPPORT_DYNAMIC_ALLOCATION is set to 0 then pvPortMalloc ( ) will
* equate to NULL . */
2021-08-10 05:13:43 +08:00
pxTaskStatusArray = pvPortMalloc ( uxCurrentNumberOfTasks * sizeof ( TaskStatus_t ) ) ; /*lint !e9079 All values returned by pvPortMalloc() have at least the alignment required by the MCU's stack and this allocation allocates a struct that has the alignment requirements of a pointer. */
if ( pxTaskStatusArray ! = NULL )
{
/* Generate the (binary) data. */
uxArraySize = uxTaskGetSystemState ( pxTaskStatusArray , uxArraySize , & ulTotalTime ) ;
/* For percentage calculations. */
ulTotalTime / = 100UL ;
/* Avoid divide by zero errors. */
if ( ulTotalTime > 0UL )
{
/* Create a human readable table from the binary data. */
for ( x = 0 ; x < uxArraySize ; x + + )
{
/* What percentage of the total run time has the task used?
2021-08-26 13:45:51 +08:00
* This will always be rounded down to the nearest integer .
* ulTotalRunTimeDiv100 has already been divided by 100. */
2021-08-10 05:13:43 +08:00
ulStatsAsPercentage = pxTaskStatusArray [ x ] . ulRunTimeCounter / ulTotalTime ;
/* Write the task name to the string, padding with
2021-08-26 13:45:51 +08:00
* spaces so it can be printed in tabular form more
* easily . */
2021-08-10 05:13:43 +08:00
pcWriteBuffer = prvWriteNameToBuffer ( pcWriteBuffer , pxTaskStatusArray [ x ] . pcTaskName ) ;
if ( ulStatsAsPercentage > 0UL )
{
# ifdef portLU_PRINTF_SPECIFIER_REQUIRED
2021-08-26 13:45:51 +08:00
{
sprintf ( pcWriteBuffer , " \t %lu \t \t %lu%% \r \n " , pxTaskStatusArray [ x ] . ulRunTimeCounter , ulStatsAsPercentage ) ;
}
2021-08-10 05:13:43 +08:00
# else
2021-08-26 13:45:51 +08:00
{
/* sizeof( int ) == sizeof( long ) so a smaller
* printf ( ) library can be used . */
sprintf ( pcWriteBuffer , " \t %u \t \t %u%% \r \n " , ( unsigned int ) pxTaskStatusArray [ x ] . ulRunTimeCounter , ( unsigned int ) ulStatsAsPercentage ) ; /*lint !e586 sprintf() allowed as this is compiled with many compilers and this is a utility function only - not part of the core kernel implementation. */
}
2021-08-10 05:13:43 +08:00
# endif
}
else
{
/* If the percentage is zero here then the task has
2021-08-26 13:45:51 +08:00
* consumed less than 1 % of the total run time . */
2021-08-10 05:13:43 +08:00
# ifdef portLU_PRINTF_SPECIFIER_REQUIRED
2021-08-26 13:45:51 +08:00
{
sprintf ( pcWriteBuffer , " \t %lu \t \t <1%% \r \n " , pxTaskStatusArray [ x ] . ulRunTimeCounter ) ;
}
2021-08-10 05:13:43 +08:00
# else
2021-08-26 13:45:51 +08:00
{
/* sizeof( int ) == sizeof( long ) so a smaller
* printf ( ) library can be used . */
sprintf ( pcWriteBuffer , " \t %u \t \t <1%% \r \n " , ( unsigned int ) pxTaskStatusArray [ x ] . ulRunTimeCounter ) ; /*lint !e586 sprintf() allowed as this is compiled with many compilers and this is a utility function only - not part of the core kernel implementation. */
}
2021-08-10 05:13:43 +08:00
# endif
}
pcWriteBuffer + = strlen ( pcWriteBuffer ) ; /*lint !e9016 Pointer arithmetic ok on char pointers especially as in this case where it best denotes the intent of the code. */
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
/* Free the array again. NOTE! If configSUPPORT_DYNAMIC_ALLOCATION
2021-08-26 13:45:51 +08:00
* is 0 then vPortFree ( ) will be # defined to nothing . */
2021-08-10 05:13:43 +08:00
vPortFree ( pxTaskStatusArray ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2016-08-17 23:08:22 +08:00
2019-11-28 15:27:47 -03:00
# endif /* ( ( configGENERATE_RUN_TIME_STATS == 1 ) && ( configUSE_STATS_FORMATTING_FUNCTIONS > 0 ) && ( configSUPPORT_STATIC_ALLOCATION == 1 ) ) */
2016-08-17 23:08:22 +08:00
/*-----------------------------------------------------------*/
TickType_t uxTaskResetEventItemValue ( void )
{
2021-08-26 13:45:51 +08:00
TickType_t uxReturn ;
2022-11-14 15:27:00 +08:00
BaseType_t xCoreID ;
2019-11-28 15:27:47 -03:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* For SMP, we need to take the kernel lock here to ensure nothing else
* modifies the task ' s event item value simultaneously . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2022-09-05 17:31:22 +08:00
2022-11-14 15:27:00 +08:00
xCoreID = xPortGetCoreID ( ) ;
uxReturn = listGET_LIST_ITEM_VALUE ( & ( pxCurrentTCB [ xCoreID ] - > xEventListItem ) ) ;
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
/* Reset the event list item to its normal value - so it can be used with
2021-08-26 13:45:51 +08:00
* queues and semaphores . */
2022-11-14 15:27:00 +08:00
listSET_LIST_ITEM_VALUE ( & ( pxCurrentTCB [ xCoreID ] - > xEventListItem ) , ( ( TickType_t ) configMAX_PRIORITIES - ( TickType_t ) pxCurrentTCB [ xCoreID ] - > uxPriority ) ) ; /*lint !e961 MISRA exception as the casts are only redundant for some ports. */
2022-09-05 17:31:22 +08:00
2022-11-14 15:27:00 +08:00
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL_ISR ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2016-08-17 23:08:22 +08:00
2021-08-10 05:13:43 +08:00
return uxReturn ;
2016-08-17 23:08:22 +08:00
}
/*-----------------------------------------------------------*/
# if ( configUSE_MUTEXES == 1 )
2021-11-02 10:24:30 +05:30
TaskHandle_t pvTaskIncrementMutexHeldCount ( void )
2021-08-10 05:13:43 +08:00
{
2022-11-14 15:27:00 +08:00
TCB_t * pxCurTCB ;
BaseType_t xCoreID ;
# if ( configNUM_CORES > 1 )
/* For SMP, we need to take the kernel lock here as we are about to
* access kernel data structures . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
xCoreID = xPortGetCoreID ( ) ;
2020-09-30 12:56:52 -03:00
2021-08-10 05:13:43 +08:00
/* If xSemaphoreCreateMutex() is called before any tasks have been created
2021-08-26 13:45:51 +08:00
* then pxCurrentTCB will be NULL . */
2022-11-14 15:27:00 +08:00
if ( pxCurrentTCB [ xCoreID ] ! = NULL )
2021-08-10 05:13:43 +08:00
{
2022-11-14 15:27:00 +08:00
( pxCurrentTCB [ xCoreID ] - > uxMutexesHeld ) + + ;
2021-08-10 05:13:43 +08:00
}
2022-11-02 19:49:40 +08:00
2022-11-14 15:27:00 +08:00
pxCurTCB = pxCurrentTCB [ xCoreID ] ;
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2016-08-17 23:08:22 +08:00
2022-11-14 15:27:00 +08:00
return pxCurTCB ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_MUTEXES */
/*-----------------------------------------------------------*/
2021-09-06 12:46:14 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM /* IDF-3851 */
/* included here for backward binary compatibility */
# undef ulTaskNotifyTake
uint32_t ulTaskNotifyTake ( BaseType_t xClearCountOnExit ,
TickType_t xTicksToWait )
{
return ulTaskGenericNotifyTake ( tskDEFAULT_INDEX_TO_NOTIFY , xClearCountOnExit , xTicksToWait ) ;
}
# endif // ESP-PLATFORM
2016-08-17 23:08:22 +08:00
2021-09-06 12:46:14 +08:00
uint32_t ulTaskGenericNotifyTake ( UBaseType_t uxIndexToWait ,
BaseType_t xClearCountOnExit ,
TickType_t xTicksToWait )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
uint32_t ulReturn ;
configASSERT ( uxIndexToWait < configTASK_NOTIFICATION_ARRAY_ENTRIES ) ;
2021-08-10 05:13:43 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* Only block if the notification count is not already non-zero. */
2022-11-02 19:49:40 +08:00
if ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > ulNotifiedValue [ uxIndexToWait ] = = 0UL )
2021-08-10 05:13:43 +08:00
{
/* Mark this task as waiting for a notification. */
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ucNotifyState [ uxIndexToWait ] = taskWAITING_NOTIFICATION ;
2021-08-10 05:13:43 +08:00
if ( xTicksToWait > ( TickType_t ) 0 )
{
2022-08-01 19:06:44 +08:00
prvAddCurrentTaskToDelayedList ( xTicksToWait , pdTRUE ) ;
2021-09-07 15:57:50 +08:00
traceTASK_NOTIFY_TAKE_BLOCK ( uxIndexToWait ) ;
2021-08-10 05:13:43 +08:00
/* All ports are written to allow a yield in a critical
2021-09-06 12:46:14 +08:00
* section ( some will yield immediately , others wait until the
* critical section exits ) - but it is not something that
* application code should ever do . */
2021-08-10 05:13:43 +08:00
portYIELD_WITHIN_API ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
2021-09-07 15:57:50 +08:00
traceTASK_NOTIFY_TAKE ( uxIndexToWait ) ;
2022-11-02 19:49:40 +08:00
ulReturn = pxCurrentTCB [ xPortGetCoreID ( ) ] - > ulNotifiedValue [ uxIndexToWait ] ;
2021-08-10 05:13:43 +08:00
if ( ulReturn ! = 0UL )
{
if ( xClearCountOnExit ! = pdFALSE )
{
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ulNotifiedValue [ uxIndexToWait ] = 0UL ;
2021-08-10 05:13:43 +08:00
}
else
{
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ulNotifiedValue [ uxIndexToWait ] = ulReturn - ( uint32_t ) 1 ;
2021-08-10 05:13:43 +08:00
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ucNotifyState [ uxIndexToWait ] = taskNOT_WAITING_NOTIFICATION ;
2021-08-10 05:13:43 +08:00
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return ulReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TASK_NOTIFICATIONS */
/*-----------------------------------------------------------*/
2021-09-06 12:46:14 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2016-08-17 23:08:22 +08:00
2022-11-02 19:49:40 +08:00
# ifdef ESP_PLATFORM /* IDF-3851 */
/* included for backward compatibility */
# undef xTaskNotifyWait
BaseType_t xTaskNotifyWait ( uint32_t ulBitsToClearOnEntry ,
uint32_t ulBitsToClearOnExit ,
uint32_t * pulNotificationValue ,
TickType_t xTicksToWait )
{
return xTaskGenericNotifyWait ( tskDEFAULT_INDEX_TO_NOTIFY , ulBitsToClearOnEntry , ulBitsToClearOnExit , pulNotificationValue , xTicksToWait ) ;
}
# endif // ESP-PLATFORM
2021-09-06 12:46:14 +08:00
BaseType_t xTaskGenericNotifyWait ( UBaseType_t uxIndexToWait ,
uint32_t ulBitsToClearOnEntry ,
uint32_t ulBitsToClearOnExit ,
uint32_t * pulNotificationValue ,
TickType_t xTicksToWait )
{
BaseType_t xReturn ;
2021-08-10 05:13:43 +08:00
2021-09-06 12:46:14 +08:00
configASSERT ( uxIndexToWait < configTASK_NOTIFICATION_ARRAY_ENTRIES ) ;
2021-09-09 14:13:07 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
/* Only block if a notification is not already pending. */
2022-11-02 19:49:40 +08:00
if ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > ucNotifyState [ uxIndexToWait ] ! = taskNOTIFICATION_RECEIVED )
2021-08-10 05:13:43 +08:00
{
/* Clear bits in the task's notification value as bits may get
2021-09-06 12:46:14 +08:00
* set by the notifying task or interrupt . This can be used to
* clear the value to zero . */
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ulNotifiedValue [ uxIndexToWait ] & = ~ ulBitsToClearOnEntry ;
2021-08-10 05:13:43 +08:00
/* Mark this task as waiting for a notification. */
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ucNotifyState [ uxIndexToWait ] = taskWAITING_NOTIFICATION ;
2021-08-10 05:13:43 +08:00
if ( xTicksToWait > ( TickType_t ) 0 )
{
2022-08-01 19:06:44 +08:00
prvAddCurrentTaskToDelayedList ( xTicksToWait , pdTRUE ) ;
2021-09-07 15:57:50 +08:00
traceTASK_NOTIFY_WAIT_BLOCK ( uxIndexToWait ) ;
2021-08-10 05:13:43 +08:00
/* All ports are written to allow a yield in a critical
2021-09-06 12:46:14 +08:00
* section ( some will yield immediately , others wait until the
* critical section exits ) - but it is not something that
* application code should ever do . */
2021-08-10 05:13:43 +08:00
portYIELD_WITHIN_API ( ) ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
2021-09-07 15:57:50 +08:00
traceTASK_NOTIFY_WAIT ( uxIndexToWait ) ;
2021-08-10 05:13:43 +08:00
if ( pulNotificationValue ! = NULL )
{
/* Output the current notification value, which may or may not
2021-09-06 12:46:14 +08:00
* have changed . */
2022-11-02 19:49:40 +08:00
* pulNotificationValue = pxCurrentTCB [ xPortGetCoreID ( ) ] - > ulNotifiedValue [ uxIndexToWait ] ;
2021-08-10 05:13:43 +08:00
}
/* If ucNotifyValue is set then either the task never entered the
2021-09-06 12:46:14 +08:00
* blocked state ( because a notification was already pending ) or the
* task unblocked because of a notification . Otherwise the task
* unblocked because of a timeout . */
2022-11-02 19:49:40 +08:00
if ( pxCurrentTCB [ xPortGetCoreID ( ) ] - > ucNotifyState [ uxIndexToWait ] ! = taskNOTIFICATION_RECEIVED )
2021-08-10 05:13:43 +08:00
{
/* A notification was not received. */
xReturn = pdFALSE ;
}
else
{
/* A notification was already pending or a notification was
2021-09-06 12:46:14 +08:00
* received while the task was waiting . */
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ulNotifiedValue [ uxIndexToWait ] & = ~ ulBitsToClearOnExit ;
2021-08-10 05:13:43 +08:00
xReturn = pdTRUE ;
}
2022-11-02 19:49:40 +08:00
pxCurrentTCB [ xPortGetCoreID ( ) ] - > ucNotifyState [ uxIndexToWait ] = taskNOT_WAITING_NOTIFICATION ;
2021-08-10 05:13:43 +08:00
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TASK_NOTIFICATIONS */
/*-----------------------------------------------------------*/
2021-09-06 12:46:14 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2016-08-17 23:08:22 +08:00
2021-09-06 12:46:14 +08:00
BaseType_t xTaskGenericNotify ( TaskHandle_t xTaskToNotify ,
UBaseType_t uxIndexToNotify ,
uint32_t ulValue ,
eNotifyAction eAction ,
uint32_t * pulPreviousNotificationValue )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
TCB_t * pxTCB ;
BaseType_t xReturn = pdPASS ;
uint8_t ucOriginalNotifyState ;
2021-08-10 05:13:43 +08:00
2021-09-06 12:46:14 +08:00
configASSERT ( uxIndexToNotify < configTASK_NOTIFICATION_ARRAY_ENTRIES ) ;
2021-08-10 05:13:43 +08:00
configASSERT ( xTaskToNotify ) ;
pxTCB = xTaskToNotify ;
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
if ( pulPreviousNotificationValue ! = NULL )
{
2021-09-06 12:46:14 +08:00
* pulPreviousNotificationValue = pxTCB - > ulNotifiedValue [ uxIndexToNotify ] ;
2021-08-10 05:13:43 +08:00
}
2021-09-06 12:46:14 +08:00
ucOriginalNotifyState = pxTCB - > ucNotifyState [ uxIndexToNotify ] ;
2021-08-10 05:13:43 +08:00
2021-09-06 12:46:14 +08:00
pxTCB - > ucNotifyState [ uxIndexToNotify ] = taskNOTIFICATION_RECEIVED ;
2021-08-10 05:13:43 +08:00
switch ( eAction )
{
2021-09-06 12:46:14 +08:00
case eSetBits :
pxTCB - > ulNotifiedValue [ uxIndexToNotify ] | = ulValue ;
2021-08-10 05:13:43 +08:00
break ;
2021-09-06 12:46:14 +08:00
case eIncrement :
( pxTCB - > ulNotifiedValue [ uxIndexToNotify ] ) + + ;
2021-08-10 05:13:43 +08:00
break ;
2021-09-06 12:46:14 +08:00
case eSetValueWithOverwrite :
pxTCB - > ulNotifiedValue [ uxIndexToNotify ] = ulValue ;
2021-08-10 05:13:43 +08:00
break ;
2021-09-06 12:46:14 +08:00
case eSetValueWithoutOverwrite :
2021-08-10 05:13:43 +08:00
if ( ucOriginalNotifyState ! = taskNOTIFICATION_RECEIVED )
{
2021-09-06 12:46:14 +08:00
pxTCB - > ulNotifiedValue [ uxIndexToNotify ] = ulValue ;
2021-08-10 05:13:43 +08:00
}
else
{
/* The value could not be written to the task. */
xReturn = pdFAIL ;
}
2021-09-06 12:46:14 +08:00
2021-08-10 05:13:43 +08:00
break ;
case eNoAction :
2021-09-06 12:46:14 +08:00
2021-08-10 05:13:43 +08:00
/* The task is being notified without its notify value being
2021-09-06 12:46:14 +08:00
* updated . */
2021-08-10 05:13:43 +08:00
break ;
default :
2021-09-06 12:46:14 +08:00
2021-08-10 05:13:43 +08:00
/* Should not get here if all enums are handled.
2021-09-06 12:46:14 +08:00
* Artificially force an assert by testing a value the
* compiler can ' t assume is const . */
configASSERT ( pxTCB - > ulNotifiedValue [ uxIndexToNotify ] = = ~ 0UL ) ;
2021-08-10 05:13:43 +08:00
break ;
}
2021-09-07 15:57:50 +08:00
traceTASK_NOTIFY ( uxIndexToNotify ) ;
2021-08-10 05:13:43 +08:00
/* If the task is in the blocked state specifically to wait for a
2021-09-06 12:46:14 +08:00
* notification then unblock it now . */
2021-08-10 05:13:43 +08:00
if ( ucOriginalNotifyState = = taskWAITING_NOTIFICATION )
{
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
prvAddTaskToReadyList ( pxTCB ) ;
/* The task should not have been on an event list. */
configASSERT ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) = = NULL ) ;
2021-09-06 12:46:14 +08:00
# if ( configUSE_TICKLESS_IDLE != 0 )
{
/* If a task is blocked waiting for a notification then
* xNextTaskUnblockTime might be set to the blocked task ' s time
* out time . If the task is unblocked for a reason other than
* a timeout xNextTaskUnblockTime is normally left unchanged ,
* because it will automatically get reset to a new value when
* the tick count equals xNextTaskUnblockTime . However if
* tickless idling is used it might be more important to enter
* sleep mode at the earliest possible time - so reset
* xNextTaskUnblockTime here to ensure it is updated at the
* earliest possible time . */
prvResetNextTaskUnblockTime ( ) ;
}
2021-08-10 05:13:43 +08:00
# endif
2022-08-01 15:57:51 +08:00
if ( prvCheckForYield ( pxTCB , xPortGetCoreID ( ) , pdFALSE ) )
2021-08-10 05:13:43 +08:00
{
/* The notified task has a priority above the currently
2021-08-26 13:45:51 +08:00
* executing task so a yield is required . */
2022-08-01 15:57:51 +08:00
taskYIELD_IF_USING_PREEMPTION ( ) ;
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TASK_NOTIFICATIONS */
/*-----------------------------------------------------------*/
2021-09-06 12:46:14 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2016-08-17 23:08:22 +08:00
2021-09-06 12:46:14 +08:00
BaseType_t xTaskGenericNotifyFromISR ( TaskHandle_t xTaskToNotify ,
UBaseType_t uxIndexToNotify ,
uint32_t ulValue ,
eNotifyAction eAction ,
uint32_t * pulPreviousNotificationValue ,
BaseType_t * pxHigherPriorityTaskWoken )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
TCB_t * pxTCB ;
uint8_t ucOriginalNotifyState ;
BaseType_t xReturn = pdPASS ;
2022-11-14 15:27:00 +08:00
UBaseType_t uxSavedInterruptStatus ;
2021-08-10 05:13:43 +08:00
configASSERT ( xTaskToNotify ) ;
2021-09-06 12:46:14 +08:00
configASSERT ( uxIndexToNotify < configTASK_NOTIFICATION_ARRAY_ENTRIES ) ;
2021-08-10 05:13:43 +08:00
/* RTOS ports that support interrupt nesting have the concept of a
2021-09-06 12:46:14 +08:00
* maximum system call ( or maximum API call ) interrupt priority .
* Interrupts that are above the maximum system call priority are keep
* permanently enabled , even when the RTOS kernel is in a critical section ,
* but cannot make any calls to FreeRTOS API functions . If configASSERT ( )
* is defined in FreeRTOSConfig . h then
* portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) will result in an assertion
* failure if a FreeRTOS API function is called from an interrupt that has
* been assigned a priority above the configured maximum system call
* priority . Only FreeRTOS functions that end in FromISR can be called
* from interrupts that have been assigned a priority at or ( logically )
* below the maximum system call interrupt priority . FreeRTOS maintains a
* separate interrupt safe API to ensure interrupt entry is as fast and as
* simple as possible . More information ( albeit Cortex - M specific ) is
* provided on the following link :
* https : //www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */
2021-08-10 05:13:43 +08:00
portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) ;
pxTCB = xTaskToNotify ;
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_MASK_ISR ( & xKernelLock , uxSavedInterruptStatus ) ;
2021-08-10 05:13:43 +08:00
{
if ( pulPreviousNotificationValue ! = NULL )
{
2021-09-06 12:46:14 +08:00
* pulPreviousNotificationValue = pxTCB - > ulNotifiedValue [ uxIndexToNotify ] ;
2021-08-10 05:13:43 +08:00
}
2021-09-06 12:46:14 +08:00
ucOriginalNotifyState = pxTCB - > ucNotifyState [ uxIndexToNotify ] ;
pxTCB - > ucNotifyState [ uxIndexToNotify ] = taskNOTIFICATION_RECEIVED ;
2021-08-10 05:13:43 +08:00
switch ( eAction )
{
2021-09-06 12:46:14 +08:00
case eSetBits :
pxTCB - > ulNotifiedValue [ uxIndexToNotify ] | = ulValue ;
2021-08-10 05:13:43 +08:00
break ;
2021-09-06 12:46:14 +08:00
case eIncrement :
( pxTCB - > ulNotifiedValue [ uxIndexToNotify ] ) + + ;
2021-08-10 05:13:43 +08:00
break ;
2021-09-06 12:46:14 +08:00
case eSetValueWithOverwrite :
pxTCB - > ulNotifiedValue [ uxIndexToNotify ] = ulValue ;
2021-08-10 05:13:43 +08:00
break ;
2021-09-06 12:46:14 +08:00
case eSetValueWithoutOverwrite :
2021-08-10 05:13:43 +08:00
if ( ucOriginalNotifyState ! = taskNOTIFICATION_RECEIVED )
{
2021-09-06 12:46:14 +08:00
pxTCB - > ulNotifiedValue [ uxIndexToNotify ] = ulValue ;
2021-08-10 05:13:43 +08:00
}
else
{
/* The value could not be written to the task. */
xReturn = pdFAIL ;
}
2021-09-06 12:46:14 +08:00
2021-08-10 05:13:43 +08:00
break ;
2021-09-06 12:46:14 +08:00
case eNoAction :
2021-08-10 05:13:43 +08:00
/* The task is being notified without its notify value being
2021-09-06 12:46:14 +08:00
* updated . */
2021-08-10 05:13:43 +08:00
break ;
default :
2021-09-06 12:46:14 +08:00
2021-08-10 05:13:43 +08:00
/* Should not get here if all enums are handled.
2021-09-06 12:46:14 +08:00
* Artificially force an assert by testing a value the
* compiler can ' t assume is const . */
configASSERT ( pxTCB - > ulNotifiedValue [ uxIndexToNotify ] = = ~ 0UL ) ;
2021-08-10 05:13:43 +08:00
break ;
}
2021-09-07 15:57:50 +08:00
traceTASK_NOTIFY_FROM_ISR ( uxIndexToNotify ) ;
2021-08-10 05:13:43 +08:00
/* If the task is in the blocked state specifically to wait for a
2021-09-06 12:46:14 +08:00
* notification then unblock it now . */
2021-08-10 05:13:43 +08:00
if ( ucOriginalNotifyState = = taskWAITING_NOTIFICATION )
{
/* The task should not have been on an event list. */
configASSERT ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) = = NULL ) ;
2022-11-02 19:49:40 +08:00
if ( uxSchedulerSuspended [ xPortGetCoreID ( ) ] = = ( UBaseType_t ) pdFALSE )
2021-08-10 05:13:43 +08:00
{
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
prvAddTaskToReadyList ( pxTCB ) ;
}
else
{
/* The delayed and ready lists cannot be accessed, so hold
2021-09-06 12:46:14 +08:00
* this task pending until the scheduler is resumed . */
2022-11-02 19:49:40 +08:00
vListInsertEnd ( & ( xPendingReadyList [ xPortGetCoreID ( ) ] ) , & ( pxTCB - > xEventListItem ) ) ;
2021-08-10 05:13:43 +08:00
}
2022-08-01 15:57:51 +08:00
if ( prvCheckForYield ( pxTCB , xPortGetCoreID ( ) , pdFALSE ) )
2021-08-10 05:13:43 +08:00
{
/* The notified task has a priority above the currently
2021-09-06 12:46:14 +08:00
* executing task so a yield is required . */
2021-08-10 05:13:43 +08:00
if ( pxHigherPriorityTaskWoken ! = NULL )
{
* pxHigherPriorityTaskWoken = pdTRUE ;
}
2022-08-01 15:57:51 +08:00
/* Mark that a yield is pending in case the user is not
* using the " xHigherPriorityTaskWoken " parameter to an ISR
* safe FreeRTOS function . */
xYieldPending [ xPortGetCoreID ( ) ] = pdTRUE ;
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
}
2022-11-14 15:27:00 +08:00
prvEXIT_CRITICAL_OR_UNMASK_ISR ( & xKernelLock , uxSavedInterruptStatus ) ;
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TASK_NOTIFICATIONS */
/*-----------------------------------------------------------*/
2021-09-07 10:30:30 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2016-08-17 23:08:22 +08:00
2021-09-06 12:46:14 +08:00
void vTaskGenericNotifyGiveFromISR ( TaskHandle_t xTaskToNotify ,
UBaseType_t uxIndexToNotify ,
BaseType_t * pxHigherPriorityTaskWoken )
2021-08-10 05:13:43 +08:00
{
2021-09-07 10:30:30 +08:00
TCB_t * pxTCB ;
uint8_t ucOriginalNotifyState ;
2022-11-14 15:27:00 +08:00
UBaseType_t uxSavedInterruptStatus ;
2021-08-10 05:13:43 +08:00
configASSERT ( xTaskToNotify ) ;
2021-09-06 12:46:14 +08:00
configASSERT ( uxIndexToNotify < configTASK_NOTIFICATION_ARRAY_ENTRIES ) ;
2021-08-10 05:13:43 +08:00
/* RTOS ports that support interrupt nesting have the concept of a
2021-09-06 12:46:14 +08:00
* maximum system call ( or maximum API call ) interrupt priority .
* Interrupts that are above the maximum system call priority are keep
* permanently enabled , even when the RTOS kernel is in a critical section ,
* but cannot make any calls to FreeRTOS API functions . If configASSERT ( )
* is defined in FreeRTOSConfig . h then
* portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) will result in an assertion
* failure if a FreeRTOS API function is called from an interrupt that has
* been assigned a priority above the configured maximum system call
* priority . Only FreeRTOS functions that end in FromISR can be called
* from interrupts that have been assigned a priority at or ( logically )
* below the maximum system call interrupt priority . FreeRTOS maintains a
* separate interrupt safe API to ensure interrupt entry is as fast and as
* simple as possible . More information ( albeit Cortex - M specific ) is
* provided on the following link :
* https : //www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */
2021-08-10 05:13:43 +08:00
portASSERT_IF_INTERRUPT_PRIORITY_INVALID ( ) ;
pxTCB = xTaskToNotify ;
2022-11-14 15:27:00 +08:00
prvENTER_CRITICAL_OR_MASK_ISR ( & xKernelLock , uxSavedInterruptStatus ) ;
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
ucOriginalNotifyState = pxTCB - > ucNotifyState [ uxIndexToNotify ] ;
pxTCB - > ucNotifyState [ uxIndexToNotify ] = taskNOTIFICATION_RECEIVED ;
2021-08-10 05:13:43 +08:00
/* 'Giving' is equivalent to incrementing a count in a counting
2021-09-06 12:46:14 +08:00
* semaphore . */
( pxTCB - > ulNotifiedValue [ uxIndexToNotify ] ) + + ;
2021-08-10 05:13:43 +08:00
2021-09-07 15:57:50 +08:00
traceTASK_NOTIFY_GIVE_FROM_ISR ( uxIndexToNotify ) ;
2021-08-10 05:13:43 +08:00
/* If the task is in the blocked state specifically to wait for a
2021-09-06 12:46:14 +08:00
* notification then unblock it now . */
2021-08-10 05:13:43 +08:00
if ( ucOriginalNotifyState = = taskWAITING_NOTIFICATION )
{
/* The task should not have been on an event list. */
configASSERT ( listLIST_ITEM_CONTAINER ( & ( pxTCB - > xEventListItem ) ) = = NULL ) ;
if ( uxSchedulerSuspended [ xPortGetCoreID ( ) ] = = ( UBaseType_t ) pdFALSE )
{
( void ) uxListRemove ( & ( pxTCB - > xStateListItem ) ) ;
prvAddTaskToReadyList ( pxTCB ) ;
}
else
{
/* The delayed and ready lists cannot be accessed, so hold
2021-09-06 12:46:14 +08:00
* this task pending until the scheduler is resumed . */
2022-11-02 19:49:40 +08:00
vListInsertEnd ( & ( xPendingReadyList [ xPortGetCoreID ( ) ] ) , & ( pxTCB - > xEventListItem ) ) ;
2021-08-10 05:13:43 +08:00
}
2022-08-01 15:57:51 +08:00
if ( prvCheckForYield ( pxTCB , xPortGetCoreID ( ) , pdFALSE ) )
2021-08-10 05:13:43 +08:00
{
/* The notified task has a priority above the currently
2021-09-06 12:46:14 +08:00
* executing task so a yield is required . */
2021-08-10 05:13:43 +08:00
if ( pxHigherPriorityTaskWoken ! = NULL )
{
* pxHigherPriorityTaskWoken = pdTRUE ;
}
2022-08-01 15:57:51 +08:00
/* Mark that a yield is pending in case the user is not
* using the " xHigherPriorityTaskWoken " parameter in an ISR
* safe FreeRTOS function . */
xYieldPending [ xPortGetCoreID ( ) ] = pdTRUE ;
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
}
2022-11-14 15:27:00 +08:00
prvEXIT_CRITICAL_OR_UNMASK_ISR ( & xKernelLock , uxSavedInterruptStatus ) ;
2021-08-10 05:13:43 +08:00
}
2016-08-17 23:08:22 +08:00
# endif /* configUSE_TASK_NOTIFICATIONS */
2019-11-28 15:27:47 -03:00
/*-----------------------------------------------------------*/
2021-09-06 12:46:14 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
2019-11-28 15:27:47 -03:00
2021-09-06 12:46:14 +08:00
BaseType_t xTaskGenericNotifyStateClear ( TaskHandle_t xTask ,
UBaseType_t uxIndexToClear )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
TCB_t * pxTCB ;
BaseType_t xReturn ;
configASSERT ( uxIndexToClear < configTASK_NOTIFICATION_ARRAY_ENTRIES ) ;
2021-08-10 05:13:43 +08:00
/* If null is passed in here then it is the calling task that is having
2021-09-06 12:46:14 +08:00
* its notification state cleared . */
2021-08-10 05:13:43 +08:00
pxTCB = prvGetTCBFromHandle ( xTask ) ;
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
if ( pxTCB - > ucNotifyState [ uxIndexToClear ] = = taskNOTIFICATION_RECEIVED )
2021-08-10 05:13:43 +08:00
{
2021-09-06 12:46:14 +08:00
pxTCB - > ucNotifyState [ uxIndexToClear ] = taskNOT_WAITING_NOTIFICATION ;
2021-08-10 05:13:43 +08:00
xReturn = pdPASS ;
}
else
{
xReturn = pdFAIL ;
}
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-08-10 05:13:43 +08:00
return xReturn ;
}
2019-11-28 15:27:47 -03:00
# endif /* configUSE_TASK_NOTIFICATIONS */
/*-----------------------------------------------------------*/
2021-09-06 12:46:14 +08:00
# if ( configUSE_TASK_NOTIFICATIONS == 1 )
uint32_t ulTaskGenericNotifyValueClear ( TaskHandle_t xTask ,
UBaseType_t uxIndexToClear ,
uint32_t ulBitsToClear )
{
TCB_t * pxTCB ;
uint32_t ulReturn ;
/* If null is passed in here then it is the calling task that is having
* its notification state cleared . */
pxTCB = prvGetTCBFromHandle ( xTask ) ;
2022-08-12 15:52:14 +08:00
taskENTER_CRITICAL ( & xKernelLock ) ;
2021-09-06 12:46:14 +08:00
{
/* Return the notification as it was before the bits were cleared,
* then clear the bit mask . */
ulReturn = pxTCB - > ulNotifiedValue [ uxIndexToClear ] ;
pxTCB - > ulNotifiedValue [ uxIndexToClear ] & = ~ ulBitsToClear ;
}
2022-08-12 15:52:14 +08:00
taskEXIT_CRITICAL ( & xKernelLock ) ;
2021-09-06 12:46:14 +08:00
return ulReturn ;
}
# endif /* configUSE_TASK_NOTIFICATIONS */
/*-----------------------------------------------------------*/
# if ( ( configGENERATE_RUN_TIME_STATS == 1 ) && ( INCLUDE_xTaskGetIdleTaskHandle == 1 ) )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
uint32_t ulTaskGetIdleRunTimeCounter ( void )
{
2022-11-14 15:27:00 +08:00
uint32_t ulRunTimeCounter ;
# if ( configNUM_CORES > 1 )
/* For SMP, we need to take the kernel lock here as we are about to
* access kernel data structures . */
taskENTER_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
ulRunTimeCounter = xIdleTaskHandle [ xPortGetCoreID ( ) ] - > ulRunTimeCounter ;
# if ( configNUM_CORES > 1 )
/* Release the previously taken kernel lock. */
taskEXIT_CRITICAL ( & xKernelLock ) ;
# endif /* ( configNUM_CORES > 1 ) */
2020-09-30 12:56:52 -03:00
2022-11-14 15:27:00 +08:00
return ulRunTimeCounter ;
2021-08-10 05:13:43 +08:00
}
2019-11-28 15:27:47 -03:00
2022-11-02 19:49:40 +08:00
# endif /* if ( ( configGENERATE_RUN_TIME_STATS == 1 ) && ( INCLUDE_xTaskGetIdleTaskHandle == 1 ) ) */
2019-11-28 15:27:47 -03:00
/*-----------------------------------------------------------*/
2022-08-01 19:06:44 +08:00
static void prvAddCurrentTaskToDelayedList ( TickType_t xTicksToWait ,
const BaseType_t xCanBlockIndefinitely )
2019-11-28 15:27:47 -03:00
{
2021-08-26 13:45:51 +08:00
TickType_t xTimeToWake ;
const TickType_t xConstTickCount = xTickCount ;
2022-08-01 19:06:44 +08:00
BaseType_t xCurCoreID = xPortGetCoreID ( ) ;
2019-11-28 15:27:47 -03:00
2022-09-05 17:31:22 +08:00
# if ( configNUM_CORES > 1 )
if ( listIS_CONTAINED_WITHIN ( & xTasksWaitingTermination , & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) ) = = pdTRUE )
{
/* vTaskDelete() has been called to delete this task. This would have happened from the other core while this task was spinning on xTaskQueueMutex,
2022-11-02 19:49:40 +08:00
* so don ' t move the running task to the delayed list - as soon as this core re - enables interrupts this task will
* be suspended permanently . Todo : IDF - 5844. */
2022-09-05 17:31:22 +08:00
return ;
}
# endif
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
# if ( INCLUDE_xTaskAbortDelay == 1 )
{
/* About to enter a delayed list, so ensure the ucDelayAborted flag is
* reset to pdFALSE so it can be detected as having been set to pdTRUE
* when the task leaves the Blocked state . */
2022-08-01 19:06:44 +08:00
pxCurrentTCB [ xCurCoreID ] - > ucDelayAborted = pdFALSE ;
2021-08-26 13:45:51 +08:00
}
2021-08-10 05:13:43 +08:00
# endif
/* Remove the task from the ready list before adding it to the blocked list
2021-08-26 13:45:51 +08:00
* as the same list item is used for both lists . */
2022-08-01 19:06:44 +08:00
if ( uxListRemove ( & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) ) = = ( UBaseType_t ) 0 )
2021-08-10 05:13:43 +08:00
{
/* The current task must be in a ready list, so there is no need to
2021-08-26 13:45:51 +08:00
* check , and the port reset macro can be called directly . */
2022-08-01 19:06:44 +08:00
portRESET_READY_PRIORITY ( pxCurrentTCB [ xCurCoreID ] - > uxPriority , uxTopReadyPriority ) ; /*lint !e931 pxCurrentTCB cannot change as it is the calling task. pxCurrentTCB->uxPriority and uxTopReadyPriority cannot change as called with scheduler suspended or in a critical section. */
2021-08-10 05:13:43 +08:00
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
# if ( INCLUDE_vTaskSuspend == 1 )
{
2022-08-01 19:06:44 +08:00
if ( ( xTicksToWait = = portMAX_DELAY ) & & ( xCanBlockIndefinitely ! = pdFALSE ) )
2021-08-26 13:45:51 +08:00
{
/* Add the task to the suspended task list instead of a delayed task
* list to ensure it is not woken by a timing event . It will block
* indefinitely . */
2022-08-01 19:06:44 +08:00
vListInsertEnd ( & xSuspendedTaskList , & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) ) ;
2021-08-26 13:45:51 +08:00
}
else
{
/* Calculate the time at which the task should be woken if the event
* does not occur . This may overflow but this doesn ' t matter , the
* kernel will manage it correctly . */
xTimeToWake = xConstTickCount + xTicksToWait ;
/* The list item will be inserted in wake time order. */
2022-08-01 19:06:44 +08:00
listSET_LIST_ITEM_VALUE ( & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) , xTimeToWake ) ;
2021-08-26 13:45:51 +08:00
if ( xTimeToWake < xConstTickCount )
{
/* Wake time has overflowed. Place this item in the overflow
* list . */
2022-08-01 19:06:44 +08:00
vListInsert ( pxOverflowDelayedTaskList , & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) ) ;
2021-08-26 13:45:51 +08:00
}
else
{
/* The wake time has not overflowed, so the current block list
* is used . */
2022-08-01 19:06:44 +08:00
vListInsert ( pxDelayedTaskList , & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) ) ;
2021-08-26 13:45:51 +08:00
/* If the task entering the blocked state was placed at the
* head of the list of blocked tasks then xNextTaskUnblockTime
* needs to be updated too . */
if ( xTimeToWake < xNextTaskUnblockTime )
{
xNextTaskUnblockTime = xTimeToWake ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
}
2021-08-10 05:13:43 +08:00
}
2021-08-26 13:45:51 +08:00
# else /* INCLUDE_vTaskSuspend */
2021-08-10 05:13:43 +08:00
{
/* Calculate the time at which the task should be woken if the event
2021-08-26 13:45:51 +08:00
* does not occur . This may overflow but this doesn ' t matter , the kernel
* will manage it correctly . */
2021-08-10 05:13:43 +08:00
xTimeToWake = xConstTickCount + xTicksToWait ;
/* The list item will be inserted in wake time order. */
2022-08-01 19:06:44 +08:00
listSET_LIST_ITEM_VALUE ( & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) , xTimeToWake ) ;
2021-08-10 05:13:43 +08:00
if ( xTimeToWake < xConstTickCount )
{
2021-08-26 13:45:51 +08:00
/* Wake time has overflowed. Place this item in the overflow list. */
2022-08-01 19:06:44 +08:00
vListInsert ( pxOverflowDelayedTaskList , & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) ) ;
2021-08-10 05:13:43 +08:00
}
else
{
2021-08-26 13:45:51 +08:00
/* The wake time has not overflowed, so the current block list is used. */
2022-08-01 19:06:44 +08:00
vListInsert ( pxDelayedTaskList , & ( pxCurrentTCB [ xCurCoreID ] - > xStateListItem ) ) ;
2021-08-10 05:13:43 +08:00
2021-08-26 13:45:51 +08:00
/* If the task entering the blocked state was placed at the head of the
* list of blocked tasks then xNextTaskUnblockTime needs to be updated
* too . */
2021-08-10 05:13:43 +08:00
if ( xTimeToWake < xNextTaskUnblockTime )
{
xNextTaskUnblockTime = xTimeToWake ;
}
else
{
mtCOVERAGE_TEST_MARKER ( ) ;
}
}
2022-08-01 19:06:44 +08:00
/* Avoid compiler warning when INCLUDE_vTaskSuspend is not 1. */
( void ) xCanBlockIndefinitely ;
2021-08-10 05:13:43 +08:00
}
# endif /* INCLUDE_vTaskSuspend */
2019-11-28 15:27:47 -03:00
}
/* Code below here allows additional code to be inserted into this source file,
2021-08-26 13:45:51 +08:00
* especially where access to file scope functions and data is needed ( for example
* when performing module tests ) . */
2019-11-28 15:27:47 -03:00
2016-08-17 23:08:22 +08:00
# ifdef FREERTOS_MODULE_TEST
2021-08-10 05:13:43 +08:00
# include "tasks_test_access_functions.h"
2016-08-17 23:08:22 +08:00
# endif
2021-08-26 13:45:51 +08:00
# if ( configINCLUDE_FREERTOS_TASK_C_ADDITIONS_H == 1 )
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
# include "freertos_tasks_c_additions.h"
2019-11-28 15:27:47 -03:00
2021-08-10 05:13:43 +08:00
# ifdef FREERTOS_TASKS_C_ADDITIONS_INIT
static void freertos_tasks_c_additions_init ( void )
{
FREERTOS_TASKS_C_ADDITIONS_INIT ( ) ;
}
# endif
2019-11-28 15:27:47 -03:00
2021-08-26 13:45:51 +08:00
# endif /* if ( configINCLUDE_FREERTOS_TASK_C_ADDITIONS_H == 1 ) */
2019-11-28 15:27:47 -03:00
2020-10-02 17:47:55 +02:00
/* If timers.c is not referenced anywhere, don't create the timer task to save RAM */
2022-11-02 19:49:40 +08:00
BaseType_t __attribute__ ( ( weak ) ) xTimerCreateTimerTask ( void )
2020-10-02 17:47:55 +02:00
{
2021-08-10 05:13:43 +08:00
return pdPASS ;
2020-10-02 17:47:55 +02:00
}