1
0
mirror of https://github.com/azure-rtos/threadx synced 2025-01-16 07:42:57 +08:00

// /* / / Copyright (c) Microsoft Corporation. All rights reserved. / / / / This software is licensed under the Microsoft Software License / / Terms for Microsoft Azure RTOS. Full text of the license can be / / found in the LICENSE file at https://aka.ms/AzureRTOS_EULA / / and in the root directory of this software. / / */ //

Introduction

Welcome to the FreeRTOS adaptation layer for ThreadX documentation. This document will go over configuration, initialization and usage of the adaptation layer as well as important guidelines and limitations.

Files

The adaptation layer is comprised of the following files:

  • tx_freertos.c
  • FreeRTOS.h
  • event_groups.h
  • queue.h
  • semphr.h
  • task.h
  • timers.h

The main source file for the adaptation layer is “tx_freertos.c” as well as “FreeRTOS.h” the other headers are provided for compatibility with FreeRTOS. In addition, the following configuration file must be available within the project. A template of the configuration file is provided in the source distribution.

  • FreeRTOSConfig.h

ThreadX Configuration

A few ThreadX configurations should be looked at prior to using the adaptation layer. Please note that if a configuration is changed within tx_user.h the preprocessor definition TX_INCLUDE_USER_DEFINE_FILE should be defined at the compiler command line. This is to ensure that tx_user.h is properly included from tx_port.h.

Thread Extension:

The adaptation layer requires a custom field within the TX_THREAD data structure to store a reference to the adaptation layer thread-related data. The following preprocessor definitions should be added to tx_user.h. Failure to add this configuration will result in a compilation error.

#define TX_THREAD_USER_EXTENSION VOID *txfr_thread_ptr;

Timer Processing Task:

To better emulate the FreeRTOS timer behaviour it is recommended, but not necessary, to enable processing of ThreadX timers within a task instead of within an ISR. To do so the TX_TIMER_PROCESS_IN_ISR preprocessor definition should NOT be defined within tx_user.h or tx_port.h It is also recommended, but not required to have the timer task priority set at priority 0, which is the highest priority within ThreadX, like this:

#define TX_TIMER_THREAD_PRIORITY 0

If desired to reduce resource usage, timer processing can be done within the timer tick ISR by defining TX_TIMER_PROCESS_IN_ISR within tx_user.h. This wont have any negative side effect but may change the sequencing of timer firing compared to FreeRTOS.

Adaptation Layer Setup and Configuration

To include the adaptation layer in a ThreadX project it is sufficient to add to the makefile or project the tx_freertos.c source file as well as create or copy the FreeRTOSConfig.h configuration file. The configuration can be taken from an existing project but care should be taken to ensure that it contains no extraneous declarations and definitions that may cause compilation errors. A template of the configuration file can be found within the config_template directory. Every uncommented definitions within the template configuration file are understood by the adaptation layer while every other configuration definitions are ignored. The various FreeRTOS headers can be found at the root of the adaptation layer source directory. For simplicity only a selected set of the usual configuration defines are supported by the adaptation layer. All other configurations not explicitly listed are ignored. In addition a few additional definitions can be added to FreeRTOSConfig.h to tune the behaviour of the adaptation layer.

Name Default Value Description
configUSE_16_BIT_TICKS 0 Set to 1 to use 16-bit tick
configSTACK_DEPTH_TYPE uint16_t Use to override the type used to specify stack depth
configTICK_RATE_HZ - Set the kernel tick rate, used by the pdMS_TO_TICKS() macro
configMAX_PRIORITIES - Maximum number of priorities. Must be less than or equal to the configured number of ThreadX priorities.
configMINIMAL_STACK_SIZE 512u Minimum stack size, used as the stack size of the idle task if TX_FREERTOS_IDLE_STACK is not defined.
configTOTAL_HEAP_SIZE - Amount of internal memory allocated to the adaptation layer when creating FreeRTOS objects. Can be set to 0 to disable dynamic allocation.
INCLUDE_vTaskDelete 1 Set to 0 to disable the task delete API. When disabled the adaptation layer will not create the idle task to save resources.
TX_FREERTOS_IDLE_STACK 512u Stack size of the idle task.
TX_FREERTOS_ASSERT_FAIL Define to a macro invoked on internal assertion failures from within the adaptation layer
configASSERT Define to a macro invoked for invalid arguments

Initialization

Unless auto-initialization is used, see below, early initialization should proceed as is usual for any ThreadX application. The adaptation layer should be initialized upon reaching tx_application_define() by calling tx_freertos_init(). Internally tx_freertos_init() will initialize a ThreadX byte pool that will be used by the adaptation layer to allocate and free kernel objects. FreeRTOS tasks and objects can be created from within tx_application_define(). Usually, at least the initial application task should be created.

Heres an example of a minimal initialization of the adaptation layer.

VOID tx_application_define(VOID * first_unused_memory)
{
    BaseType_t error;
    TaskHandle_t task_handle;

    tx_freertos_init();

    error = xTaskCreate(first_thread_entry, "Initial Task", STACK_SIZE, NULL, 10, &task_handle);
    if(error != pdPASS) {
        // Handle error.
    }
}

It is also possible to initialize the FreeRTOS adaptation layer later once ThreadX is started as long as tx_freertos_init() is called before attempting to create any FreeRTOS kernel objects or tasks.

Auto-initialization

The default method of initializing the adaptation layer requires some modifications to the application to initialize ThreadX and the adaptation layer prior to using the FreeRTOS API. An alternative method is available when modifications to the application isn't desirable. To do so the following configuration should be added and set to one in FreeRTOSConfig.h:

#define TX_FREERTOS_AUTO_INIT 1

Additionally the following preprocessor definition should be added to tx_user.h:

#define TX_PORT_SPECIFIC_PRE_SCHEDULER_INITIALIZATION return;

When both of those configurations are done, the adaptation layer will be initialized automatically by the first call to an object create function, no other call is allowed prior to starting the kernel by calling vTaskStartScheduler().

Port Macros

Four port macros to manipulate ISRs are required by the adaptation layer. Default implementations are provided for IAR. They can be defined in FreeRTOSConfig.h if needed.

taskENTER_CRITICAL_FROM_ISR()

taskEXIT_CRITICAL_FROM_ISR

portDISABLE_INTERRUPTS

portENABLE_INTERRUPTS

Usage Guidelines and Limitations

While the adaptation layer attempts to emulate the behaviour and feature set of FreeRTOS it must be understood that some limitations and deviations exist. Every application developer using the adaptation layer should review this document, especially the exhaustive list of supported API presented later in this document along with important deviations in behaviour from FreeRTOS. In addition, the following general guidelines should be followed.

Usage of FreeRTOS API Calls Within a Native ThreadX Thread

The scenario of using FreeRTOS calls within a thread created using tx_thread_create() is not supported. While it is not explicitly disallowed by the adaptation layer some FreeRTOS API may function erratically when they are called from outside a FreeRTOS task.

Usage of ThreadX Native API Calls Within a FreeRTOS Task

Similarly to the above scenario, usage of native ThreadX calls from within a task created using xTaskCreate() or xTaskCreateStatic() is not recommended, with exceptions. It is possible to use any of the ThreadX synchronizations objects, such as Semaphore, Mutexes, Queues and event flags directly from within a FreeRTOS task. It is not supported to use any of the ThreadX thread manipulation functions, however.

Mix of ThreadX Threads and FreeRTOS Tasks

It is possible to mix native ThreadX threads and FreeRTOS threads in the same applications assuming that previous two guidelines are followed.

Idle Task and Idle Priority

ThreadX by its design does not have an idle task. FreeRTOS, however, does have an idle task, and it is responsible for performing the final cleanup and freeing of memory when deleting a task. The adaptation layer creates, during initialization, an idle task to perform the same duty. The idle task has the lowest possible priority allowed by ThreadX namely TX_MAX_PRIORITIES 1. The idle task will only run if there is a deleted thread to cleanup, otherwise it will be waiting on an internal semaphore posted from the adaptation layer vTaskDelete() function.

Task Yielding and Preemption From ISR

When returning from an ISR, ThreadX will automatically switch to the highest priority task ready to run. As such the taskYIELD_FROM_ISR() macro has no effect and yielding will always occur if a higher priority task was made ready to run within and ISR. taskYIELD() however works as expected yielding control to the next ready-to-run task with the same priority as the current task.

Task Deletion and the Idle Task

FreeRTOS uses the Idle task, which is created during initialization, to cleanup deleted threads. The Threadx adaptation layer mimics this behaviour by deleting and freeing any resources allocated to a task within an internal idle task. If the vTaskDelete() call is disabled by setting INCLUDE_vTaskDelete to 0, the idle task will not be created and the task delete functionality wont be available.

Returning From a Task

FreeRTOS does not allow simply returning from a task while this behaviour is permitted within ThreadX. The adaptation layer allows returning from a task when vTaskDelete() is available although for portability it is recommended to always explicitly delete tasks. When vTaskDelete() is disabled by setting INCLUDE_vTaskDelete to 0, returning from a task will cause an assertion failure.

Memory Management and Heap Configuration

The total memory size available to the adaptation layer when creating FreeRTOS kernel objects dynamically is configurable through the configTOTAL_HEAP_SIZE definition located in FreeRTOSConfig.h configuration file. An area of memory of the specified size is created internally and managed using a ThreadX byte pool. Setting configTOTAL_HEAP_SIZE to 0 will effectively disable dynamic allocation of kernel objects.

Tickless Mode

Tickless mode, which can be selected using configUSE_TICKLESS_IDLE is not supported by the adaptation layer or ThreadX.

API Support by Category

This release of the FreeRTOS adaptation layer for ThreadX broadly supports the following API groups:

  • Task creation, control and utilities
  • Semaphores and Mutexes
  • Queues
  • Queue Sets
  • Direct to Task Notifications
  • Software Timers
  • Event Groups

The tables that follow list the individual along with any notable limitations or deviations from the FreeRTOS behaviour of each API or API group.

Task

The task API represents the core of the adaptation layer enabling creation and control of FreeRTOS tasks using underlying ThreadX threads. FreeRTOS priorities are transparently mapped to ThreadX priorities in reverse orders since under FreeRTOS increasing priority values means an increasing task priority which is the reverse of the ThreadX convention.

Macros

Name Notes
taskSCHEDULER_SUSPENDED
taskSCHEDULER_NOT_STARTED
taskSCHEDULER_RUNNING
taskENTER_CRITICAL()
taskEXIT_CRITICAL()
taskENTER_CRITICAL_FROM_ISR()
taskEXIT_CRITICAL_FROM_ISR()
taskDISABLE_INTERRUPTS()
taskENABLE_INTERRUPTS()
tskIDLE_PRIORITY
taskYIELD()
taskYIELD_FROM_ISR() Has no effect, ThreadX will automatically pre-empt when a higher priority task is available to run upon returning from an ISR.

Functions

Names Notes
vTaskStartScheduler() Has no effect if TX_FREERTOS_AUTO_INIT is undefined or set to 0, scheduler is started automatically when returning from tx_application_define(). Otherwise this call will start the scheduler for the first time.
xTaskGetSchedulerState()
vTaskSuspendAll()
xTaskResumeAll() Always return pdFALSE since pre-emption is handled automatically by ThreadX.
xTaskCreateStatic()
xTaskCreate()
uxTaskGetNumberOfTasks() Only returns the number of task created by either xTaskCreate() or xTaskcreateStatic(). Task created internally by ThreadX or by the application using tx_thread_create() are not counted.
vTaskDelete()
vTaskDelay()
vTaskDelayUntil() The implementation of vTaskDelayUntil() cannot perform a wait in an atomic fashion. As such there might be additional jitter when using this function with the adaptation layer. The implementation will, however, not accumulate any drift.
xTaskGetCurrentTaskHandle() This will only work when called from a task created by either xTaskCreate() or xTaskcreateStatic().
vTaskSuspend()
vTaskResume()
xTaskResumeFromISR()
xTaskAbortDelay()
uxTaskPriorityGet()
uxTaskPriorityGetFromISR()
vTaskPrioritySet()
pcTaskGetName()
eTaskGetState()
uxTaskGetStackHighWaterMark() Not implemented.
uxTaskGetStackHighWaterMark2() Not implemented.
xTaskCallApplicationTaskHook() Not implemented.
xTaskGetIdleTaskHandle() Not implemented since the idle task is not a FreeRTOS task.
uxTaskGetSystemState() Not implemented.
vTaskList() Not implemented.
vTaskGetRunTimeStats() Not implemented.
xTaskGetIdleRunTimeCounter() Not implemented since the idle task is not free-running but waiting for delete events.

Task Notification

Task notifications are fully implemented.

Name Notes
xTaskNotifyGive()
vTaskNotifyGiveFromISR()
ulTaskNotifyTake()
xTaskNotifyWait()
xTaskNotify()
xTaskNotifyFromISR()
xTaskNotifyAndQuery()
xTaskGenericNotify()
xTaskNotifyAndQueryFromISR()
xTaskGenericNotifyFromISR()
xTaskNotifyStateClear()
ulTaskNotifyValueClear()

Semaphore and Mutex

Semaphores, either counting or binary as well as Mutexes are fully implemented. Mutexes under the adaptation layer cannot be taken or given from an ISR as this is not allowed in ThreadX as well as recent version of FreeRTOS. Due to differences between ThreadX and FreeRTOS it is possible that the ordering task wakeup may slightly differ.

Name Notes
xSemaphoreCreateCounting()
xSemaphoreCreateCountingStatic()
xSemaphoreCreateBinary()
xSemaphoreCreateBinaryStatic()
xSemaphoreCreateMutex()
xSemaphoreCreateMutexStatic()
xSemaphoreCreateRecursiveMutex()
xSemaphoreCreateRecursiveMutexStatic()
vSemaphoreDelete()
xSemaphoreTake()
xSemaphoreTakeFromISR() Its not possible to take a mutex from an ISR.
xSemaphoreTakeRecursive()
xSemaphoreGive()
xSemaphoreGiveFromISR()
xSemaphoreGiveRecursive() Its not possible to give a mutex from an ISR.
uxSemaphoreGetCount()
xSemaphoreGetMutexHolder()
xSemaphoreGetMutexHolderFromISR()
vSemaphoreCreateBinary() Not implemented since its marked as deprecated in the FreeRTOS documentation.

Queue

The FreeRTOS queue API is implemented with the help of ThreadX semaphores and is designed to mimic the behaviour of FreeRTOS queues. Due to differences between ThreadX and FreeRTOS it is possible that the ordering task wakeup may slightly differ.

Name Notes
xQueueCreate()
xQueueCreateStatic()
vQueueDelete()
xQueueSend()
xQueueSendFromISR()
xQueueSendToBack()
xQueueSendToBackFromISR()
xQueueSendToFront()
xQueueSendToFrontFromISR()
xQueueReceive()
xQueueReceiveFromISR()
xQueuePeek()
xQueuePeekFromISR()
uxQueueMessagesWaiting()
uxQueueMessagesWaitingFromISR()
uxQueueSpacesAvailable()
xQueueIsQueueEmptyFromISR()
xQueueIsQueueFullFromISR()
xQueueReset()
xQueueOverwrite()
xQueueOverwriteFromISR()
pcQueueGetName() Not implemented.

Queue Sets

Queue sets are implemented with support for adding queues, semaphores and mutexes to a set. Due to the way ThreadX deliver messages, it is possible that the order of events returned by xQueueSelectFromSet() and xQueueSelectFromSetFromISR() differs from the order they would be returned by FreeRTOS.

Name Notes
xQueueCreateSet()
xQueueAddToSet()
xQueueRemoveFromSet()
xQueueSelectFromSet()
xQueueSelectFromSetFromISR()

Event Group

Event groups are implemented using ThreadXs event flags. It is important to note however that xEventGroupSync() is not atomic.

Name Notes
xEventGroupCreate()
xEventGroupCreateStatic()
vEventGroupDelete()
xEventGroupWaitBits()
xEventGroupSetBits()
xEventGroupSetBitsFromISR()
xEventGroupClearBits()
xEventGroupClearBitsFromISR()
xEventGroupGetBits()
xEventGroupGetBitsFromISR()
xEventGroupSync() Not atomic.

Timer

The timer API is fully implemented except for the pend function all functionality provided by xTimerPendFunctionCall() and xTimerPendFunctionCallFromISR(). Also since the timer handling thread is not a FreeRTOS task, xTimerGetTimerDaemonTaskHandle() is not supported as well.

Name Notes
xTimerCreate()
xTimerCreateStatic()
xTimerDelete()
xTimerIsTimerActive()
xTimerStart()
xTimerStop()
xTimerChangePeriod()
xTimerReset()
xTimerStartFromISR()
xTimerStopFromISR()
xTimerChangePeriodFromISR()
xTimerResetFromISR()
pvTimerGetTimerID()
vTimerSetTimerID()
vTimerSetReloadMode()
pcTimerGetName()
xTimerGetPeriod()
xTimerGetExpiryTime()
uxTimerGetReloadMode()
xTimerPendFunctionCall() Not implemented.
xTimerPendFunctionCallFromISR() Not implemented.
xTimerGetTimerDaemonTaskHandle() Not implemented.

Document Revision History

Version Release Notes
1 2020-09-30 - Initial release.