| Latest News Items: FAT file system released - Tick suppression demo'ed SAM4L MCU & RX100 MCUs - embTCP released for PIC32 |
|
|||||||||||||||
Atmel SAM3 RTOS Demo
|
|||||||||||||||
| FreeRTOS is now an integral part of the Atmel ASF driver library for SAM. Click to view the application note. |
Two pre-configured projects are provided. One runs on the SAM3S-EK2 evaluation kit, and the other on the SAM3X-EK evaluation kit.
See also the FAQ My application does not run, what could be wrong?
The Atmel Studio 6 Solution file for the ATSAM3S-EK2 demo application is called RTOSDemo.atsln, and is located in the FreeRTOS/Demo/CORTEX_ATSAM3S-EK2_Atmel_Studio directory.
The Atmel Studio 6 Solution file for the ATSAM3X-EK demo application is also called RTOSDemo.atsln, and is located in the FreeRTOS/Demo/CORTEX_ATSAM3X_Atmel_Studio directory.
The Preparing the Project Directory Structure section of this page contains important information on preparing this directory.
It should be noted that the com test tasks are included to demonstrate queues being used to communicate between tasks and interrupts, and to demonstrate a context switch being performed from inside an interrupt service routine. The serial driver used is not intended to represent an efficient implementation. Real applications should make use of the USART's peripheral DMA channel (PDC).
main_blinky() creates one queue, and two tasks. It then starts the RTOS scheduler.
The queue send task is implemented by the prvQueueSendTask() function in main_blinky.c. It sends the value 100 to the queue every 200 milliseconds.
The queue receive task is implemented by the prvQueueReceiveTask() function in main_blinky.c. It repeatedly reads from the queue with a block time specified, causing it to enter the Blocked state if the queue is empty. The task toggles the red/orange LED each time the value 100 is received from the queue, therefore, because the queue send task sends to the queue every 200 milliseconds, the queue receive task should exit the Blocked state and toggle the red/orange LED every 200 milliseconds.
main() creates 34 tasks and three software timers before starting the RTOS scheduler. The demo then dynamically and continuously creates and deletes a further two tasks while it is running.
A 'check' software timer is created that periodically inspects the standard demo tasks to ensure all the tasks are functioning as expected. The check software timer's callback function toggles the green LED on the SAM3 development hardware. This gives a visual feedback of the system health. If the green LED is toggling every 3 seconds, then the check software timer has not discovered any problems. If the LED is toggling every 200 milliseconds, then the check software timer has discovered a problem in one or more tasks. This mechanism can be tested by removing the loopback connector, and in so doing, deliberately causing the com test tasks to fail.
Preparing the Project Directory Structure
Atmel Studio requires all the source files built by a project to be located in, or
in subdirectories of, the directory that contains the Atmel Studio project itself.
It is therefore necessary to copy the FreeRTOS
and standard demo source files used by the demo applications from
their locations in the standard FreeRTOS directory structure into the demo
directory. A batch file called CreateProjectDirectoryStructure.bat is provided
for this purpose.
CreateProjectDirectoryStructure.bat is located in the same directory that contains the Atmel Studio 6 solution, and must be executed before the demo application can be built successfully.
This sets the frequency of the RTOS tick interrupt. The supplied value of 1000Hz is useful for testing the RTOS kernel functionality but is faster than most applications need. Lowering the frequency will improve efficiency.
See the RTOS kernel configuration documentation for full information on these configuration constants.
Whereas configKERNEL_INTERRUPT_PRIORITY and configMAX_SYSCALL_INTERRUPT_PRIORITY are full eight bit shifted values, defined to be used as raw numbers directly in the ARM Cortex-M3 NVIC registers, configLIBRARY_LOWEST_INTERRUPT_PRIORITY and configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY are equivalents that are defined using just the 4 priority bits implemented in the SAM3 NVIC. These values are provided because the CMSIS library function NVIC_SetPriority() requires the un-shifted 4 bit format.
Attention please!: See the page dedicated to setting interrupt priorities on ARM Cortex-M devices. Remember that ARM Cortex-M cores use numerically low priority numbers to represent HIGH priority interrupts. This can seem counter-intuitive and is easy to forget! If you wish to assign an interrupt a low priority do NOT assign it a priority of 0 (or other low numeric value) as this will result in the interrupt actually having the highest priority in the system - and therefore potentially make your system crash if this priority is above configMAX_SYSCALL_INTERRUPT_PRIORITY. Also, do not leave interrupt priorities unassigned, as by default they will have a priority of 0 and therefore the highest priority possible.
The lowest priority on a ARM Cortex-M core is in fact 255 - however different ARM Cortex-M microcontroller manufacturers implement a different number of priority bits and supply library functions that expect priorities to be specified in different ways. For example, on Atmel SAM3 ARM Cortex-M3 microcontrollers, the lowest priority you can specify is in fact 15 - this is defined by the constant configLIBRARY_LOWEST_INTERRUPT_PRIORITY in FreeRTOSConfig.h. The highest priority that can be assigned is always zero.
It is also recommended to ensure that all priority bits are assigned as being preemption priority bits, and none as sub priority bits, as they are in the provided demo.
Each port #defines 'portBASE_TYPE' to equal the most efficient data type for that processor. This port defines portBASE_TYPE to be of type long.
Note that portEND_SWITCHING_ISR() will leave interrupts enabled.
The following source code snippet is provided as an example. The interrupt uses a semaphore to synchronise with a task (not shown), and calls portEND_SWITCHING_ISR to ensure the interrupt returns directly to the task. See the function USART1_Handler() in the file serial.c included in this demo project for another example.
void Dummy_IRQHandler(void)
{
long lHigherPriorityTaskWoken = pdFALSE;
/* Clear the interrupt if necessary. */
Dummy_ClearITPendingBit();
/* This interrupt does nothing more than demonstrate how to synchronise a
task with an interrupt. A semaphore is used for this purpose. Note
lHigherPriorityTaskWoken is initialised to zero. */
xSemaphoreGiveFromISR( xTestSemaphore, &lHigherPriorityTaskWoken );
/* If there was a task that was blocked on the semaphore, and giving the
semaphore caused the task to unblock, and the unblocked task has a priority
higher than the current Running state task (the task that this interrupt
interrupted), then lHigherPriorityTaskWoken will have been set to pdTRUE
internally within xSemaphoreGiveFromISR(). Passing pdTRUE into the
portEND_SWITCHING_ISR() macro will result in a context switch being pended to
ensure this interrupt returns directly to the unblocked, higher priority,
task. Passing pdFALSE into portEND_SWITCHING_ISR() has no effect. */
portEND_SWITCHING_ISR( lHigherPriorityTaskWoken );
}
Only FreeRTOS API functions that end in "FromISR" can be called from an interrupt service routine - and then only if the priority of the interrupt is less than or equal to that set by the configMAX_SYSCALL_INTERRUPT_PRIORITY configuration constant (or configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY).