1 /* 2 * SPDX-FileCopyrightText: 2017 Amazon.com, Inc. or its affiliates 3 * SPDX-FileCopyrightText: 2015-2019 Cadence Design Systems, Inc. 4 * 5 * SPDX-License-Identifier: MIT 6 * 7 * SPDX-FileContributor: 2016-2022 Espressif Systems (Shanghai) CO LTD 8 */ 9 10 /* 11 * FreeRTOS Kernel V11.1.0 12 * Copyright (C) 2017 Amazon.com, Inc. or its affiliates. All Rights Reserved. 13 * 14 * Permission is hereby granted, free of charge, to any person obtaining a copy of 15 * this software and associated documentation files (the "Software"), to deal in 16 * the Software without restriction, including without limitation the rights to 17 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of 18 * the Software, and to permit persons to whom the Software is furnished to do so, 19 * subject to the following conditions: 20 * 21 * The above copyright notice and this permission notice shall be included in all 22 * copies or substantial portions of the Software. If you wish to use our Amazon 23 * FreeRTOS name, please do so in a fair use way that does not cause confusion. 24 * 25 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 26 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS 27 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR 28 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER 29 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 30 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 31 * 32 * https://www.FreeRTOS.org 33 * https://github.com/FreeRTOS 34 * 35 * 1 tab == 4 spaces! 36 */ 37 38 /* 39 * Copyright (c) 2015-2019 Cadence Design Systems, Inc. 40 * 41 * Permission is hereby granted, free of charge, to any person obtaining 42 * a copy of this software and associated documentation files (the 43 * "Software"), to deal in the Software without restriction, including 44 * without limitation the rights to use, copy, modify, merge, publish, 45 * distribute, sublicense, and/or sell copies of the Software, and to 46 * permit persons to whom the Software is furnished to do so, subject to 47 * the following conditions: 48 * 49 * The above copyright notice and this permission notice shall be included 50 * in all copies or substantial portions of the Software. 51 * 52 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 53 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 54 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. 55 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY 56 * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, 57 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE 58 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 59 */ 60 61 #ifndef PORTMACRO_H 62 #define PORTMACRO_H 63 64 /* *INDENT-OFF* */ 65 #ifdef __cplusplus 66 extern "C" { 67 #endif 68 /* *INDENT-ON* */ 69 70 #ifndef __ASSEMBLER__ 71 72 #include <stdint.h> 73 74 #include <xtensa/hal.h> 75 #include <xtensa/config/core.h> 76 #include <xtensa/config/system.h> /* required for XSHAL_CLIB */ 77 #include <xtensa/xtruntime.h> 78 #include "soc/spinlock.h" 79 #include "esp_timer.h" /* required for FreeRTOS run time stats */ 80 #include "esp_system.h" 81 #include "esp_idf_version.h" 82 #include "esp_heap_caps.h" 83 84 /* TODO: Resolve build warnings generated due to this header inclusion */ 85 #include "hal/cpu_hal.h" 86 87 /* TODO: These includes are not directly used in this file. They are kept into to prevent a breaking change. Remove these. */ 88 #include <limits.h> 89 #include <xtensa/xtensa_api.h> 90 91 #include "soc/cpu.h" 92 #include "soc/soc_memory_layout.h" 93 #if ( ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL( 4, 2, 0 ) ) 94 #include "soc/compare_set.h" 95 #endif /* ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(4, 2, 0) */ 96 97 /*#include "xtensa_context.h" */ 98 99 /*----------------------------------------------------------- 100 * Port specific definitions. 101 * 102 * The settings in this file configure FreeRTOS correctly for the 103 * given hardware and compiler. 104 * 105 * These settings should not be altered. 106 *----------------------------------------------------------- 107 */ 108 109 /* Type definitions. */ 110 111 #define portCHAR int8_t 112 #define portFLOAT float 113 #define portDOUBLE double 114 #define portLONG int32_t 115 #define portSHORT int16_t 116 #define portSTACK_TYPE uint8_t 117 #define portBASE_TYPE int 118 119 typedef portSTACK_TYPE StackType_t; 120 typedef portBASE_TYPE BaseType_t; 121 typedef unsigned portBASE_TYPE UBaseType_t; 122 123 #if ( configTICK_TYPE_WIDTH_IN_BITS == TICK_TYPE_WIDTH_16_BITS ) 124 typedef uint16_t TickType_t; 125 #define portMAX_DELAY ( TickType_t ) 0xffff 126 #elif ( configTICK_TYPE_WIDTH_IN_BITS == TICK_TYPE_WIDTH_32_BITS ) 127 typedef uint32_t TickType_t; 128 #define portMAX_DELAY ( TickType_t ) 0xffffffffUL 129 #else 130 #error configTICK_TYPE_WIDTH_IN_BITS set to unsupported tick type width. 131 #endif 132 /*-----------------------------------------------------------*/ 133 134 /* portbenchmark */ 135 #include "portbenchmark.h" 136 137 #include "sdkconfig.h" 138 #include "esp_attr.h" 139 140 /* "mux" data structure (spinlock) */ 141 typedef spinlock_t portMUX_TYPE; /**< Spinlock type used by FreeRTOS critical sections */ 142 #define portMUX_INITIALIZER_UNLOCKED SPINLOCK_INITIALIZER /**< Spinlock initializer */ 143 #define portMUX_FREE_VAL SPINLOCK_FREE /**< Spinlock is free. [refactor-todo] check if this is still required */ 144 #define portMUX_NO_TIMEOUT SPINLOCK_WAIT_FOREVER /**< When passed for 'timeout_cycles', spin forever if necessary. [refactor-todo] check if this is still required */ 145 #define portMUX_TRY_LOCK SPINLOCK_NO_WAIT /**< Try to acquire the spinlock a single time only. [refactor-todo] check if this is still required */ 146 #define portMUX_INITIALIZE( mux ) spinlock_initialize( mux ) /*< Initialize a spinlock to its unlocked state */ 147 148 #define portCRITICAL_NESTING_IN_TCB 1 149 150 /* 151 * Modifications to portENTER_CRITICAL. 152 * 153 * For an introduction, see "Critical Sections & Disabling Interrupts" in docs/api-guides/freertos-smp.rst 154 * 155 * The original portENTER_CRITICAL only disabled the ISRs. This is enough for single-CPU operation: by 156 * disabling the interrupts, there is no task switch so no other tasks can meddle in the data, and because 157 * interrupts are disabled, ISRs can't corrupt data structures either. 158 * 159 * For multiprocessing, things get a bit more hairy. First of all, disabling the interrupts doesn't stop 160 * the tasks or ISRs on the other processors meddling with our CPU. For tasks, this is solved by adding 161 * a spinlock to the portENTER_CRITICAL macro. A task running on the other CPU accessing the same data will 162 * spinlock in the portENTER_CRITICAL code until the first CPU is done. 163 * 164 * For ISRs, we now also need muxes: while portENTER_CRITICAL disabling interrupts will stop ISRs on the same 165 * CPU from meddling with the data, it does not stop interrupts on the other cores from interfering with the 166 * data. For this, we also use a spinlock in the routines called by the ISR, but these spinlocks 167 * do not disable the interrupts (because they already are). 168 * 169 * This all assumes that interrupts are either entirely disabled or enabled. Interrupt priority levels 170 * will break this scheme. 171 * 172 * Remark: For the ESP32, portENTER_CRITICAL and portENTER_CRITICAL_ISR both alias vPortEnterCritical, meaning 173 * that either function can be called both from ISR as well as task context. This is not standard FreeRTOS 174 * behaviour; please keep this in mind if you need any compatibility with other FreeRTOS implementations. 175 */ 176 void vPortCPUInitializeMutex( portMUX_TYPE * mux ); 177 #ifdef CONFIG_FREERTOS_PORTMUX_DEBUG 178 #error CONFIG_FREERTOS_PORTMUX_DEBUG not supported in Amazon FreeRTOS 179 #endif 180 181 void vTaskExitCritical(); 182 void vTaskEnterCritical(); vPortConsumeSpinlockArg(int unused,...)183 static inline void vPortConsumeSpinlockArg( int unused, 184 ... ) 185 { 186 } 187 188 /** @brief Acquire a portmux spinlock with a timeout 189 * 190 * @param mux Pointer to portmux to acquire. 191 * @param timeout_cycles Timeout to spin, in CPU cycles. Pass portMUX_NO_TIMEOUT to wait forever, 192 * portMUX_TRY_LOCK to try a single time to acquire the lock. 193 * 194 * @return true if mutex is successfully acquired, false on timeout. 195 */ 196 bool vPortCPUAcquireMutexTimeout( portMUX_TYPE * mux, 197 int timeout_cycles ); 198 void vPortCPUReleaseMutex( portMUX_TYPE * mux ); 199 200 #define portENTER_CRITICAL( ... ) do { vTaskEnterCritical(); vPortConsumeSpinlockArg( 0, ## __VA_ARGS__ ); } while( 0 ) 201 #define portEXIT_CRITICAL( ... ) do { vTaskExitCritical(); vPortConsumeSpinlockArg( 0, ## __VA_ARGS__ ); } while( 0 ) 202 203 204 #define portENTER_CRITICAL_ISR( mux ) vPortCPUAcquireMutexTimeout( mux, portMUX_NO_TIMEOUT ) 205 #define portEXIT_CRITICAL_ISR( mux ) vPortCPUReleaseMutex( mux ) 206 207 #define portENTER_CRITICAL_SAFE( mux ) \ 208 do { \ 209 if( xPortInIsrContext() ) { \ 210 portENTER_CRITICAL_ISR( mux ); \ 211 } \ 212 else { \ 213 portENTER_CRITICAL( mux ); \ 214 } \ 215 } while( 0 ) 216 217 #define portEXIT_CRITICAL_SAFE( mux ) \ 218 do { \ 219 if( xPortInIsrContext() ) { \ 220 portEXIT_CRITICAL_ISR( mux ); \ 221 } \ 222 else { \ 223 portEXIT_CRITICAL( mux ); \ 224 } \ 225 } while( 0 ) 226 227 #define portASSERT_IF_IN_ISR() vPortAssertIfInISR() 228 void vPortAssertIfInISR( void ); 229 230 /* Critical section management. NW-TODO: replace XTOS_SET_INTLEVEL with more efficient version, if any? */ 231 /* These cannot be nested. They should be used with a lot of care and cannot be called from interrupt level. */ 232 /* */ 233 /* Only applies to one CPU. See notes above & below for reasons not to use these. */ 234 #define portDISABLE_INTERRUPTS() do { XTOS_SET_INTLEVEL( XCHAL_EXCM_LEVEL ); portbenchmarkINTERRUPT_DISABLE(); } while( 0 ) 235 #define portENABLE_INTERRUPTS() do { portbenchmarkINTERRUPT_RESTORE( 0 ); XTOS_SET_INTLEVEL( 0 ); } while( 0 ) 236 237 /* Cleaner solution allows nested interrupts disabling and restoring via local registers or stack. */ 238 /* They can be called from interrupts too. */ 239 /* WARNING: Only applies to current CPU. See notes above. */ xPortSetInterruptMaskFromISR(void)240 static inline UBaseType_t __attribute__( ( always_inline ) ) xPortSetInterruptMaskFromISR( void ) 241 { 242 UBaseType_t prev_int_level = XTOS_SET_INTLEVEL( XCHAL_EXCM_LEVEL ); 243 244 portbenchmarkINTERRUPT_DISABLE(); 245 return prev_int_level; 246 } 247 vPortClearInterruptMaskFromISR(UBaseType_t prev_level)248 static inline void __attribute__( ( always_inline ) ) vPortClearInterruptMaskFromISR( UBaseType_t prev_level ) 249 { 250 portbenchmarkINTERRUPT_RESTORE( prev_level ); 251 XTOS_RESTORE_JUST_INTLEVEL( prev_level ); 252 } 253 254 /* These FreeRTOS versions are similar to the nested versions above */ 255 #define portSET_INTERRUPT_MASK_FROM_ISR() xPortSetInterruptMaskFromISR() 256 #define portCLEAR_INTERRUPT_MASK_FROM_ISR( prev_level ) vPortClearInterruptMaskFromISR( prev_level ) 257 258 /*Because the ROM routines don't necessarily handle a stack in external RAM correctly, we force */ 259 /*the stack memory to always be internal. */ 260 #define portTcbMemoryCaps ( MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT ) 261 #define portStackMemoryCaps ( MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT ) 262 263 #define pvPortMallocTcbMem( size ) heap_caps_malloc( size, portTcbMemoryCaps ) 264 #define pvPortMallocStackMem( size ) heap_caps_malloc( size, portStackMemoryCaps ) 265 266 /*xTaskCreateStatic uses these functions to check incoming memory. */ 267 #define portVALID_TCB_MEM( ptr ) ( esp_ptr_internal( ptr ) && esp_ptr_byte_accessible( ptr ) ) 268 #ifdef CONFIG_SPIRAM_ALLOW_STACK_EXTERNAL_MEMORY 269 #define portVALID_STACK_MEM( ptr ) esp_ptr_byte_accessible( ptr ) 270 #else 271 #define portVALID_STACK_MEM( ptr ) ( esp_ptr_internal( ptr ) && esp_ptr_byte_accessible( ptr ) ) 272 #endif 273 274 /* 275 * Wrapper for the Xtensa compare-and-set instruction. This subroutine will atomically compare 276 * *addr to 'compare'. If *addr == compare, *addr is set to *set. *set is updated with the previous 277 * value of *addr (either 'compare' or some other value.) 278 * 279 * Warning: From the ISA docs: in some (unspecified) cases, the s32c1i instruction may return the 280 * *bitwise inverse* of the old mem if the mem wasn't written. This doesn't seem to happen on the 281 * ESP32 (portMUX assertions would fail). 282 */ uxPortCompareSet(volatile uint32_t * addr,uint32_t compare,uint32_t * set)283 static inline void uxPortCompareSet( volatile uint32_t * addr, 284 uint32_t compare, 285 uint32_t * set ) 286 { 287 #if ( ESP_IDF_VERSION < ESP_IDF_VERSION_VAL( 4, 2, 0 ) ) 288 __asm__ __volatile__ ( 289 "WSR %2,SCOMPARE1 \n" 290 "S32C1I %0, %1, 0 \n" 291 : "=r" ( *set ) 292 : "r" ( addr ), "r" ( compare ), "0" ( *set ) 293 ); 294 #else 295 #if ( XCHAL_HAVE_S32C1I > 0 ) 296 __asm__ __volatile__ ( 297 "WSR %2,SCOMPARE1 \n" 298 "S32C1I %0, %1, 0 \n" 299 : "=r" ( *set ) 300 : "r" ( addr ), "r" ( compare ), "0" ( *set ) 301 ); 302 #else 303 /* No S32C1I, so do this by disabling and re-enabling interrupts (slower) */ 304 uint32_t intlevel, old_value; 305 __asm__ __volatile__ ( "rsil %0, " XTSTR( XCHAL_EXCM_LEVEL ) "\n" 306 : "=r" ( intlevel ) ); 307 308 old_value = *addr; 309 310 if( old_value == compare ) 311 { 312 *addr = *set; 313 } 314 315 __asm__ __volatile__ ( "memw \n" 316 "wsr %0, ps\n" 317 : : "r" ( intlevel ) ); 318 319 *set = old_value; 320 #endif /* if ( XCHAL_HAVE_S32C1I > 0 ) */ 321 #endif /* #if (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(4, 2, 0)) */ 322 } 323 324 #if ( ESP_IDF_VERSION < ESP_IDF_VERSION_VAL( 4, 2, 0 ) ) 325 void uxPortCompareSetExtram( volatile uint32_t * addr, 326 uint32_t compare, 327 uint32_t * set ); 328 #else uxPortCompareSetExtram(volatile uint32_t * addr,uint32_t compare,uint32_t * set)329 static inline void uxPortCompareSetExtram( volatile uint32_t * addr, 330 uint32_t compare, 331 uint32_t * set ) 332 { 333 #if defined( CONFIG_SPIRAM ) 334 compare_and_set_extram( addr, compare, set ); 335 #endif 336 } 337 #endif /* if ( ESP_IDF_VERSION < ESP_IDF_VERSION_VAL( 4, 2, 0 ) ) */ 338 339 /*-----------------------------------------------------------*/ 340 341 /* Architecture specifics. */ 342 #define portSTACK_GROWTH ( -1 ) 343 #define portTICK_PERIOD_MS ( ( TickType_t ) 1000 / configTICK_RATE_HZ ) 344 #define portBYTE_ALIGNMENT 4 345 #define portNOP() XT_NOP() 346 /*-----------------------------------------------------------*/ 347 348 /* Fine resolution time */ 349 #define portGET_RUN_TIME_COUNTER_VALUE() xthal_get_ccount() 350 /*ccount or esp_timer are initialized elsewhere */ 351 #define portCONFIGURE_TIMER_FOR_RUN_TIME_STATS() 352 353 #ifdef CONFIG_FREERTOS_RUN_TIME_STATS_USING_ESP_TIMER 354 /* Coarse resolution time (us) */ 355 #define portALT_GET_RUN_TIME_COUNTER_VALUE( x ) do { x = ( uint32_t ) esp_timer_get_time(); } while( 0 ) 356 #endif 357 358 359 360 /* Kernel utilities. */ 361 void vPortYield( void ); 362 void vPortEvaluateYieldFromISR( int argc, 363 ... ); 364 void _frxt_setup_switch( void ); 365 366 /* Macro to count number of arguments of a __VA_ARGS__ used to support portYIELD_FROM_ISR with, 367 * or without arguments. The macro counts only 0 or 1 arguments. 368 * 369 * In the future, we want to switch to C++20. We also want to become compatible with clang. 370 * Hence, we provide two versions of the following macros which are using variadic arguments. 371 * The first one is using the GNU extension ##__VA_ARGS__. The second one is using the C++20 feature __VA_OPT__(,). 372 * This allows users to compile their code with standard C++20 enabled instead of the GNU extension. 373 * Below C++20, we haven't found any good alternative to using ##__VA_ARGS__. 374 */ 375 #if defined( __cplusplus ) && ( __cplusplus > 201703L ) 376 #define portGET_ARGUMENT_COUNT( ... ) portGET_ARGUMENT_COUNT_INNER( 0 __VA_OPT__(, ) __VA_ARGS__, 1, 0 ) 377 #else 378 #define portGET_ARGUMENT_COUNT( ... ) portGET_ARGUMENT_COUNT_INNER( 0, ## __VA_ARGS__, 1, 0 ) 379 #endif 380 #define portGET_ARGUMENT_COUNT_INNER( zero, one, count, ... ) count 381 382 _Static_assert( portGET_ARGUMENT_COUNT() == 0, "portGET_ARGUMENT_COUNT() result does not match for 0 arguments" ); 383 _Static_assert( portGET_ARGUMENT_COUNT( 1 ) == 1, "portGET_ARGUMENT_COUNT() result does not match for 1 argument" ); 384 385 #define portYIELD() vPortYield() 386 387 /* The macro below could be used when passing a single argument, or without any argument, 388 * it was developed to support both usages of portYIELD inside of an ISR. Any other usage form 389 * might result in undesired behaviour 390 */ 391 #if defined( __cplusplus ) && ( __cplusplus > 201703L ) 392 #define portYIELD_FROM_ISR( ... ) vPortEvaluateYieldFromISR( portGET_ARGUMENT_COUNT( __VA_ARGS__ ) __VA_OPT__(, ) __VA_ARGS__ ) 393 #else 394 #define portYIELD_FROM_ISR( ... ) vPortEvaluateYieldFromISR( portGET_ARGUMENT_COUNT( __VA_ARGS__ ), ## __VA_ARGS__ ) 395 #endif 396 397 static inline BaseType_t xPortGetCoreID(); 398 399 /*-----------------------------------------------------------*/ 400 401 /* Task function macros as described on the FreeRTOS.org WEB site. */ 402 #define portTASK_FUNCTION_PROTO( vFunction, pvParameters ) void vFunction( void * pvParameters ) 403 #define portTASK_FUNCTION( vFunction, pvParameters ) void vFunction( void * pvParameters ) 404 405 /* When coprocessors are defined, we to maintain a pointer to coprocessors area. */ 406 /* We currently use a hack: redefine field xMPU_SETTINGS in TCB block as a structure that can hold: */ 407 /* MPU wrappers, coprocessor area pointer, trace code structure, and more if needed. */ 408 /* The field is normally used for memory protection. FreeRTOS should create another general purpose field. */ 409 typedef struct 410 { 411 #if XCHAL_CP_NUM > 0 412 volatile StackType_t * coproc_area; /* Pointer to coprocessor save area; MUST BE FIRST */ 413 #endif 414 415 #if portUSING_MPU_WRAPPERS 416 /* Define here mpu_settings, which is port dependent */ 417 int mpu_setting; /* Just a dummy example here; MPU not ported to Xtensa yet */ 418 #endif 419 420 #if configUSE_TRACE_FACILITY_2 421 struct 422 { 423 /* Cf. porttraceStamp() */ 424 int taskstamp; /* Stamp from inside task to see where we are */ 425 int taskstampcount; /* A counter usually incremented when we restart the task's loop */ 426 } porttrace; 427 #endif 428 } xMPU_SETTINGS; 429 430 /* Main hack to use MPU_wrappers even when no MPU is defined (warning: mpu_setting should not be accessed; otherwise move this above xMPU_SETTINGS) */ 431 #if ( XCHAL_CP_NUM > 0 || configUSE_TRACE_FACILITY_2 ) && !portUSING_MPU_WRAPPERS /* If MPU wrappers not used, we still need to allocate coproc area */ 432 #undef portUSING_MPU_WRAPPERS 433 #define portUSING_MPU_WRAPPERS 1 /* Enable it to allocate coproc area */ 434 #define MPU_WRAPPERS_H /* Override mpu_wrapper.h to disable unwanted code */ 435 #define PRIVILEGED_FUNCTION 436 #define PRIVILEGED_DATA 437 #endif 438 439 void vApplicationSleep( TickType_t xExpectedIdleTime ); 440 441 #define portSUPPRESS_TICKS_AND_SLEEP( idleTime ) vApplicationSleep( idleTime ) 442 443 void _xt_coproc_release( volatile void * coproc_sa_base ); 444 445 /*-----------------------------------------------------------*/ 446 447 #if ( ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL( 4, 2, 0 ) ) 448 /* Architecture specific optimisations. */ 449 450 #if configUSE_PORT_OPTIMISED_TASK_SELECTION == 1 451 452 /* Check the configuration. */ 453 #if ( configMAX_PRIORITIES > 32 ) 454 #error configUSE_PORT_OPTIMISED_TASK_SELECTION can only be set to 1 when configMAX_PRIORITIES is less than or equal to 32. It is very rare that a system requires more than 10 to 15 different priorities as tasks that share a priority will time slice. 455 #endif 456 457 /* Store/clear the ready priorities in a bit map. */ 458 #define portRECORD_READY_PRIORITY( uxPriority, uxReadyPriorities ) ( uxReadyPriorities ) |= ( 1UL << ( uxPriority ) ) 459 #define portRESET_READY_PRIORITY( uxPriority, uxReadyPriorities ) ( uxReadyPriorities ) &= ~( 1UL << ( uxPriority ) ) 460 461 /*-----------------------------------------------------------*/ 462 463 #define portGET_HIGHEST_PRIORITY( uxTopPriority, uxReadyPriorities ) uxTopPriority = ( 31 - __builtin_clz( ( uxReadyPriorities ) ) ) 464 465 #endif /* configUSE_PORT_OPTIMISED_TASK_SELECTION */ 466 467 #endif /* ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(4, 2, 0) */ 468 469 /*-----------------------------------------------------------*/ 470 471 /* 472 * Map to the memory management routines required for the port. 473 * 474 * Note that libc standard malloc/free are also available for 475 * non-FreeRTOS-specific code, and behave the same as 476 * pvPortMalloc()/vPortFree(). 477 */ 478 #define pvPortMalloc heap_caps_malloc_default 479 #define vPortFree heap_caps_free 480 #define xPortGetFreeHeapSize esp_get_free_heap_size 481 #define xPortGetMinimumEverFreeHeapSize esp_get_minimum_free_heap_size 482 483 #if ( ESP_IDF_VERSION < ESP_IDF_VERSION_VAL( 4, 2, 0 ) ) 484 485 /* 486 * Send an interrupt to another core in order to make the task running 487 * on it yield for a higher-priority task. 488 */ 489 490 void vPortYieldOtherCore( BaseType_t coreid ) PRIVILEGED_FUNCTION; 491 492 #endif /* ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(4, 2, 0) */ 493 494 /* 495 * Callback to set a watchpoint on the end of the stack. Called every context switch to change the stack 496 * watchpoint around. 497 */ 498 void vPortSetStackWatchpoint( void * pxStackStart ); 499 500 /* 501 * Returns true if the current core is in ISR context; low prio ISR, med prio ISR or timer tick ISR. High prio ISRs 502 * aren't detected here, but they normally cannot call C code, so that should not be an issue anyway. 503 */ 504 BaseType_t xPortInIsrContext(); 505 506 507 /* 508 * This function will be called in High prio ISRs. Returns true if the current core was in ISR context 509 * before calling into high prio ISR context. 510 */ 511 BaseType_t xPortInterruptedFromISRContext(); 512 513 /* 514 * The structures and methods of manipulating the MPU are contained within the 515 * port layer. 516 * 517 * Fills the xMPUSettings structure with the memory region information 518 * contained in xRegions. 519 */ 520 #if ( portUSING_MPU_WRAPPERS == 1 ) 521 struct xMEMORY_REGION; 522 void vPortStoreTaskMPUSettings( xMPU_SETTINGS * xMPUSettings, 523 const struct xMEMORY_REGION * const xRegions, 524 StackType_t * pxBottomOfStack, 525 configSTACK_DEPTH_TYPE uxStackDepth ) PRIVILEGED_FUNCTION; 526 void vPortReleaseTaskMPUSettings( xMPU_SETTINGS * xMPUSettings ); 527 #endif 528 529 /* Multi-core: get current core ID */ xPortGetCoreID()530 static inline BaseType_t IRAM_ATTR xPortGetCoreID() 531 { 532 return ( uint32_t ) cpu_hal_get_core_id(); 533 } 534 535 /* Get tick rate per second */ 536 uint32_t xPortGetTickRateHz( void ); 537 xPortCanYield(void)538 static inline bool IRAM_ATTR xPortCanYield( void ) 539 { 540 uint32_t ps_reg = 0; 541 542 /*Get the current value of PS (processor status) register */ 543 RSR( PS, ps_reg ); 544 545 /* 546 * intlevel = (ps_reg & 0xf); 547 * excm = (ps_reg >> 4) & 0x1; 548 * CINTLEVEL is max(excm * EXCMLEVEL, INTLEVEL), where EXCMLEVEL is 3. 549 * However, just return true, only intlevel is zero. 550 */ 551 552 return( ( ps_reg & PS_INTLEVEL_MASK ) == 0 ); 553 } 554 555 /* porttrace */ 556 #if configUSE_TRACE_FACILITY_2 557 #include "porttrace.h" 558 #endif 559 560 /* configASSERT_2 if requested */ 561 #if configASSERT_2 562 #include <stdio.h> 563 void exit( int ); 564 #define configASSERT( x ) if( !( x ) ) { porttracePrint( -1 ); printf( "\nAssertion failed in %s:%d\n", __FILE__, __LINE__ ); exit( -1 ); } 565 #endif 566 567 /* Barriers */ 568 #define portMEMORY_BARRIER() __asm volatile ( "" ::: "memory" ) 569 570 571 #endif // __ASSEMBLER__ 572 573 /* *INDENT-OFF* */ 574 #ifdef __cplusplus 575 } 576 #endif 577 /* *INDENT-ON* */ 578 579 #endif /* PORTMACRO_H */ 580