当前位置:   article > 正文

TC397 FreeRTOS SMP多核开发(一):FreeRTOS SMP介绍

freertos smp

一、TC397 多核介绍

  • AURIX TC3xx微控制器架构具有多达6个独立的处理器内核CPU0…CPU5, 可在一个统一平台上无缝托管多个应用程序和操作系统.
  • 由于实现了具有独立读取接口的多个程序Flash模块, 该架构支持进一步的实时处理.
  • AURIX专为性能,安全性和安全性而设计,具有并行执行流程,锁步核心和进一步增强的硬件安全机制的功能

二、FreeRTOS SMP介绍

FreeRTOS SMP是FreeRTOS为了适应多核MCU的开发而推出的SMP对称多核调度方式,一般应用在具有相同架构的对称多核上,通过统一的调度管理来实现多核的调度。

FreeRTOS 内核中的 SMP 支持可使一个 FreeRTOS内核实例在多个相同的处理器内核上调度任务。 这些内核架构必须相同,并共享相同的内存。

FreeRTOS SMP多核代码位于github上的

FreeRTOS/FreeRTOS-Kernel at smp (github.com)icon-default.png?t=N7T8https://github.com/FreeRTOS/FreeRTOS-Kernel/tree/smp本专栏主要通过移植FreeRTOS SMP来实现TC397的多核调度 。

三、FreeRTOS SMP与FreeRTOS的区别

FreeRTOS API 单核和 SMP 版本之间基本上保持相同,除了以下SMP中特定的API。因此,为尽量减少或不增加任何工作量,应同时编译 FreeRTOS 单核 与 SMP 版本。但是, 一些适用于单核应用的设计假设可能不适用于多核应用, 因此可能会存在一些功能问题。

(1)SMP 特定 API

以下附加 API 可用于 FreeRTOS-SMP 内核:

xTaskCreateAffinitySet
  1. BaseType_t xTaskCreateAffinitySet( TaskFunction_t pxTaskCode,
  2. const char * const pcName,
  3. const configSTACK_DEPTH_TYPE usStackDepth,
  4. void * const pvParameters,
  5. UBaseType_t uxPriority,
  6. UBaseType_t uxCoreAffinityMask,
  7. TaskHandle_t * const pxCreatedTask );

此函数是 xTaskCreate 的扩展, 用于创建一个带有关联掩码的新任务,并将其添加到准备运行的任务列表中。 configUSE_CORE_AFFINITY 必须定义为 1,才可使用此函数。

参数:

  • uxCoreAffinityMask - 一个按位值,指示可以 运行任务的内核。内核编号从 0 到 (configNUMBER_OF_CORES - 1)。例如, 为确保任务能在内核 0 和内核 1 上运行,请将 uxCoreAffinityMask 设置为 0x03。

xTaskCreateStaticAffinitySet
  1. TaskHandle_t xTaskCreateStaticAffinitySet( TaskFunction_t pxTaskCode,
  2. const char * const pcName,
  3. const uint32_t ulStackDepth,
  4. void * const pvParameters,
  5. UBaseType_t uxPriority,
  6. StackType_t * const puxStackBuffer,
  7. StaticTask_t * const pxTaskBuffer,
  8. UBaseType_t uxCoreAffinityMask );

此函数是 xTaskCreateStatic 的扩展, 用于创建一个带有关联掩码的新任务,并将其添加到准备运行的任务列表中。 configUSE_CORE_AFFINITY 必须定义为 1,才可使用此函数。

参数:

  • uxCoreAffinityMask - 一个按位值,指示可以 运行任务的内核。内核编号从 0 到 (configNUMBER_OF_CORES - 1)。例如, 可以在内核 0 和内核 1 上运行,请将 uxCoreAffinityMask 设置

vTaskCoreAffinitySet
void vTaskCoreAffinitySet( const TaskHandle_t xTask, UBaseType_t uxCoreAffinityMask ); 

configUSE_CORE_AFFINITY 必须定义为 1,才可使用此函数。

设置任务的内核关联掩码,即可以运行任务的内核

参数:

  • xTask:内核关联掩码所要执行任务的句柄。传递 NULL 将为 调用任务设置内核关联掩码。

  • uxCoreAffinityMask - 一个按位值,指示可以 运行任务的内核。内核的编号范围为 0 到 (configNUMBER_OF_CORES - 1)。例如,为确保任务 可以在内核 0 和内核 1 上运行,请将 uxCoreAffinityMask 设置 为 0x03

用法示例:

  1. /* The function that creates task. */
  2. void vAFunction( void )
  3. {
  4. TaskHandle_t xHandle;
  5. UBaseType_t uxCoreAffinityMask;
  6. /* Create a task, storing the handle. */
  7. xTaskCreate( vTaskCode, "NAME", STACK_SIZE, NULL, tskIDLE_PRIORITY, &( xHandle ) );
  8. /* Define the core affinity mask such that this task can only run on core 0
  9. * and core 2. */
  10. uxCoreAffinityMask = ( ( 1 << 0 ) | ( 1 << 2 ) );
  11. /* Set the core affinity mask for the task. */
  12. vTaskCoreAffinitySet( xHandle, uxCoreAffinityMask );
  13. }
 vTaskCoreAffinityGet
UBaseType_t vTaskCoreAffinityGet( const TaskHandle_t xTask ); 

configUSE_CORE_AFFINITY 必须定义为 1,才可使用此函数。

设置任务的内核关联掩码,即可以运行任务的内核。

参数:

  • xTask:内核关联掩码所要执行任务的句柄。传递 NULL 将为 调用任务设置内核关联掩码。

返回:

  • 内核关联掩码,指示可以 运行任务的内核的按位值。内核编号从 0 到 (configNUMBER_OF_CORES - 1)。 例如,如果任务可以在内核 0 和内核 1 上运行,则内核关联掩码 为 0x03

用法示例:

  1. /* Task handle of the networking task - it is populated elsewhere. */
  2. TaskHandle_t xNetworkingTaskHandle;
  3. void vAFunction( void )
  4. {
  5. TaskHandle_t xHandle;
  6. UBaseType_t uxNetworkingCoreAffinityMask;
  7. /* Create a task, storing the handle. */
  8. xTaskCreate( vTaskCode, "NAME", STACK_SIZE, NULL, tskIDLE_PRIORITY, &( xHandle ) );
  9. /* Get the core affinity mask for the networking task. */
  10. uxNetworkingCoreAffinityMask = vTaskCoreAffinityGet( xNetworkingTaskHandle );
  11. /* Here is a hypothetical scenario, just for the example. Assume that we
  12. * have 2 cores - Core 0 and core 1. We want to pin the application task to
  13. * the core that is not the networking task core to ensure that the
  14. * application task does not interfere with networking. */
  15. if( ( uxNetworkingCoreAffinityMask & ( 1 << 0 ) ) != 0 )
  16. {
  17. /* The networking task can run on core 0, pin our task to core 1. */
  18. vTaskCoreAffinitySet( xHandle, ( 1 << 1 ) );
  19. }
  20. else
  21. {
  22. /* Otherwise, pin our task to core 0. */
  23. vTaskCoreAffinitySet( xHandle, ( 1 << 0 ) );
  24. }
  25. }

(2)SMP 特定配置选项

以下附加配置选项可用于 FreeRTOS-SMP 内核:

configNUMBER_OF_CORES

设置可用的处理器内核数。

configRUN_MULTIPLE_PRIORITIES

在单核 FreeRTOS 应用程序中, 如果存在能够运行的较高优先级任务,则较低优先级任务永远不会运行。 在 SMP FreeRTOS应用程序中, RTOS 内核将运行与可用内核数量一样多的任务,在一个内核上运行优先级较低的任务的同时, 较高优先级的任务可能同时 在另一个内核上运行。 如果您的应用程序或库 为单核环境编写,并且因此对任务执行的顺序做出假设, 这可能会导致问题。 因此,提供 configRUN_MULTIPLE_PRIORITIES 来 控制此行为。

如果 configRUN_MULTIPLE_PRIORITIES 定义 0,则多个任务 只有在具有相同优先级时才能同时运行, 保持这样的范例:如果存在能够运行的较高优先级任务,则较低优先级任务永远不会运行。 如果 configRUN_MULTIPLE_PRIORITIES 定义 1, 则具有不同优先级的多个任务可以同时运行,因此,较高和较低优先级的任务可以同时 在不同的内核上运行。

configUSE_CORE_AFFINITY

应用程序编写者可以控制任务在哪些内核上运行。 如果 configUSE_CORE_AFFINITY 定义为 1,则 vTaskCoreAffinitySet 可 控制任务在哪些内核上运行,vTaskCoreAffinityGet 可 用于查询任务在哪些内核上运行。 如果 configUSE_CORE_AFFINITY 为 0, 则 FreeRTOS 调度器可以在任何可用内核上自由运行任何任务。

configUSE_TASK_PREEMPTION_DISABLE

在单核 FreeRTOS 应用程序中,可以将 FreeRTOS 调度器配置为 抢占式或合作式——请参阅 configUSE_PREEMPTION 的定义。 在 SMP FreeRTOS 应用程序中,如果 configUSE_TASK_PREEMPTION_DISABLE 定义为 1, 则可使用 vTaskPreemptionDisable 和 vTaskPreemptionEnable API 函数将单个任务设置为抢占式或协作式。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/weixin_40725706/article/detail/144458
推荐阅读
相关标签
  

闽ICP备14008679号