mirror of
https://github.com/FreeRTOS/FreeRTOS-Kernel.git
synced 2025-10-24 05:37:50 -04:00
Add the common demo files to the MicroBlaze project - this directory will be removed prior to release.
This commit is contained in:
parent
53d52c0cd3
commit
be9b7e82c8
23 changed files with 4668 additions and 0 deletions
|
@ -0,0 +1,308 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Creates six tasks that operate on three queues as follows:
|
||||||
|
*
|
||||||
|
* The first two tasks send and receive an incrementing number to/from a queue.
|
||||||
|
* One task acts as a producer and the other as the consumer. The consumer is a
|
||||||
|
* higher priority than the producer and is set to block on queue reads. The queue
|
||||||
|
* only has space for one item - as soon as the producer posts a message on the
|
||||||
|
* queue the consumer will unblock, pre-empt the producer, and remove the item.
|
||||||
|
*
|
||||||
|
* The second two tasks work the other way around. Again the queue used only has
|
||||||
|
* enough space for one item. This time the consumer has a lower priority than the
|
||||||
|
* producer. The producer will try to post on the queue blocking when the queue is
|
||||||
|
* full. When the consumer wakes it will remove the item from the queue, causing
|
||||||
|
* the producer to unblock, pre-empt the consumer, and immediately re-fill the
|
||||||
|
* queue.
|
||||||
|
*
|
||||||
|
* The last two tasks use the same queue producer and consumer functions. This time the queue has
|
||||||
|
* enough space for lots of items and the tasks operate at the same priority. The
|
||||||
|
* producer will execute, placing items into the queue. The consumer will start
|
||||||
|
* executing when either the queue becomes full (causing the producer to block) or
|
||||||
|
* a context switch occurs (tasks of the same priority will time slice).
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
Changes from V4.1.1
|
||||||
|
|
||||||
|
+ The second set of tasks were created the wrong way around. This has been
|
||||||
|
corrected.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
#include "queue.h"
|
||||||
|
|
||||||
|
/* Demo program include files. */
|
||||||
|
#include "BlockQ.h"
|
||||||
|
|
||||||
|
#define blckqSTACK_SIZE configMINIMAL_STACK_SIZE
|
||||||
|
#define blckqNUM_TASK_SETS ( 3 )
|
||||||
|
|
||||||
|
/* Structure used to pass parameters to the blocking queue tasks. */
|
||||||
|
typedef struct BLOCKING_QUEUE_PARAMETERS
|
||||||
|
{
|
||||||
|
xQueueHandle xQueue; /*< The queue to be used by the task. */
|
||||||
|
portTickType xBlockTime; /*< The block time to use on queue reads/writes. */
|
||||||
|
volatile short *psCheckVariable; /*< Incremented on each successful cycle to check the task is still running. */
|
||||||
|
} xBlockingQueueParameters;
|
||||||
|
|
||||||
|
/* Task function that creates an incrementing number and posts it on a queue. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vBlockingQueueProducer, pvParameters );
|
||||||
|
|
||||||
|
/* Task function that removes the incrementing number from a queue and checks that
|
||||||
|
it is the expected number. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vBlockingQueueConsumer, pvParameters );
|
||||||
|
|
||||||
|
/* Variables which are incremented each time an item is removed from a queue, and
|
||||||
|
found to be the expected value.
|
||||||
|
These are used to check that the tasks are still running. */
|
||||||
|
static volatile short sBlockingConsumerCount[ blckqNUM_TASK_SETS ] = { ( unsigned short ) 0, ( unsigned short ) 0, ( unsigned short ) 0 };
|
||||||
|
|
||||||
|
/* Variable which are incremented each time an item is posted on a queue. These
|
||||||
|
are used to check that the tasks are still running. */
|
||||||
|
static volatile short sBlockingProducerCount[ blckqNUM_TASK_SETS ] = { ( unsigned short ) 0, ( unsigned short ) 0, ( unsigned short ) 0 };
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vStartBlockingQueueTasks( unsigned portBASE_TYPE uxPriority )
|
||||||
|
{
|
||||||
|
xBlockingQueueParameters *pxQueueParameters1, *pxQueueParameters2;
|
||||||
|
xBlockingQueueParameters *pxQueueParameters3, *pxQueueParameters4;
|
||||||
|
xBlockingQueueParameters *pxQueueParameters5, *pxQueueParameters6;
|
||||||
|
const unsigned portBASE_TYPE uxQueueSize1 = 1, uxQueueSize5 = 5;
|
||||||
|
const portTickType xBlockTime = ( portTickType ) 1000 / portTICK_RATE_MS;
|
||||||
|
const portTickType xDontBlock = ( portTickType ) 0;
|
||||||
|
|
||||||
|
/* Create the first two tasks as described at the top of the file. */
|
||||||
|
|
||||||
|
/* First create the structure used to pass parameters to the consumer tasks. */
|
||||||
|
pxQueueParameters1 = ( xBlockingQueueParameters * ) pvPortMalloc( sizeof( xBlockingQueueParameters ) );
|
||||||
|
|
||||||
|
/* Create the queue used by the first two tasks to pass the incrementing number.
|
||||||
|
Pass a pointer to the queue in the parameter structure. */
|
||||||
|
pxQueueParameters1->xQueue = xQueueCreate( uxQueueSize1, ( unsigned portBASE_TYPE ) sizeof( unsigned short ) );
|
||||||
|
|
||||||
|
/* The consumer is created first so gets a block time as described above. */
|
||||||
|
pxQueueParameters1->xBlockTime = xBlockTime;
|
||||||
|
|
||||||
|
/* Pass in the variable that this task is going to increment so we can check it
|
||||||
|
is still running. */
|
||||||
|
pxQueueParameters1->psCheckVariable = &( sBlockingConsumerCount[ 0 ] );
|
||||||
|
|
||||||
|
/* Create the structure used to pass parameters to the producer task. */
|
||||||
|
pxQueueParameters2 = ( xBlockingQueueParameters * ) pvPortMalloc( sizeof( xBlockingQueueParameters ) );
|
||||||
|
|
||||||
|
/* Pass the queue to this task also, using the parameter structure. */
|
||||||
|
pxQueueParameters2->xQueue = pxQueueParameters1->xQueue;
|
||||||
|
|
||||||
|
/* The producer is not going to block - as soon as it posts the consumer will
|
||||||
|
wake and remove the item so the producer should always have room to post. */
|
||||||
|
pxQueueParameters2->xBlockTime = xDontBlock;
|
||||||
|
|
||||||
|
/* Pass in the variable that this task is going to increment so we can check
|
||||||
|
it is still running. */
|
||||||
|
pxQueueParameters2->psCheckVariable = &( sBlockingProducerCount[ 0 ] );
|
||||||
|
|
||||||
|
|
||||||
|
/* Note the producer has a lower priority than the consumer when the tasks are
|
||||||
|
spawned. */
|
||||||
|
xTaskCreate( vBlockingQueueConsumer, ( signed char * ) "QConsB1", blckqSTACK_SIZE, ( void * ) pxQueueParameters1, uxPriority, NULL );
|
||||||
|
xTaskCreate( vBlockingQueueProducer, ( signed char * ) "QProdB2", blckqSTACK_SIZE, ( void * ) pxQueueParameters2, tskIDLE_PRIORITY, NULL );
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Create the second two tasks as described at the top of the file. This uses
|
||||||
|
the same mechanism but reverses the task priorities. */
|
||||||
|
|
||||||
|
pxQueueParameters3 = ( xBlockingQueueParameters * ) pvPortMalloc( sizeof( xBlockingQueueParameters ) );
|
||||||
|
pxQueueParameters3->xQueue = xQueueCreate( uxQueueSize1, ( unsigned portBASE_TYPE ) sizeof( unsigned short ) );
|
||||||
|
pxQueueParameters3->xBlockTime = xDontBlock;
|
||||||
|
pxQueueParameters3->psCheckVariable = &( sBlockingProducerCount[ 1 ] );
|
||||||
|
|
||||||
|
pxQueueParameters4 = ( xBlockingQueueParameters * ) pvPortMalloc( sizeof( xBlockingQueueParameters ) );
|
||||||
|
pxQueueParameters4->xQueue = pxQueueParameters3->xQueue;
|
||||||
|
pxQueueParameters4->xBlockTime = xBlockTime;
|
||||||
|
pxQueueParameters4->psCheckVariable = &( sBlockingConsumerCount[ 1 ] );
|
||||||
|
|
||||||
|
xTaskCreate( vBlockingQueueConsumer, ( signed char * ) "QProdB3", blckqSTACK_SIZE, ( void * ) pxQueueParameters3, tskIDLE_PRIORITY, NULL );
|
||||||
|
xTaskCreate( vBlockingQueueProducer, ( signed char * ) "QConsB4", blckqSTACK_SIZE, ( void * ) pxQueueParameters4, uxPriority, NULL );
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Create the last two tasks as described above. The mechanism is again just
|
||||||
|
the same. This time both parameter structures are given a block time. */
|
||||||
|
pxQueueParameters5 = ( xBlockingQueueParameters * ) pvPortMalloc( sizeof( xBlockingQueueParameters ) );
|
||||||
|
pxQueueParameters5->xQueue = xQueueCreate( uxQueueSize5, ( unsigned portBASE_TYPE ) sizeof( unsigned short ) );
|
||||||
|
pxQueueParameters5->xBlockTime = xBlockTime;
|
||||||
|
pxQueueParameters5->psCheckVariable = &( sBlockingProducerCount[ 2 ] );
|
||||||
|
|
||||||
|
pxQueueParameters6 = ( xBlockingQueueParameters * ) pvPortMalloc( sizeof( xBlockingQueueParameters ) );
|
||||||
|
pxQueueParameters6->xQueue = pxQueueParameters5->xQueue;
|
||||||
|
pxQueueParameters6->xBlockTime = xBlockTime;
|
||||||
|
pxQueueParameters6->psCheckVariable = &( sBlockingConsumerCount[ 2 ] );
|
||||||
|
|
||||||
|
xTaskCreate( vBlockingQueueProducer, ( signed char * ) "QProdB5", blckqSTACK_SIZE, ( void * ) pxQueueParameters5, tskIDLE_PRIORITY, NULL );
|
||||||
|
xTaskCreate( vBlockingQueueConsumer, ( signed char * ) "QConsB6", blckqSTACK_SIZE, ( void * ) pxQueueParameters6, tskIDLE_PRIORITY, NULL );
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vBlockingQueueProducer, pvParameters )
|
||||||
|
{
|
||||||
|
unsigned short usValue = 0;
|
||||||
|
xBlockingQueueParameters *pxQueueParameters;
|
||||||
|
short sErrorEverOccurred = pdFALSE;
|
||||||
|
|
||||||
|
pxQueueParameters = ( xBlockingQueueParameters * ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
if( xQueueSend( pxQueueParameters->xQueue, ( void * ) &usValue, pxQueueParameters->xBlockTime ) != pdPASS )
|
||||||
|
{
|
||||||
|
sErrorEverOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* We have successfully posted a message, so increment the variable
|
||||||
|
used to check we are still running. */
|
||||||
|
if( sErrorEverOccurred == pdFALSE )
|
||||||
|
{
|
||||||
|
( *pxQueueParameters->psCheckVariable )++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Increment the variable we are going to post next time round. The
|
||||||
|
consumer will expect the numbers to follow in numerical order. */
|
||||||
|
++usValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vBlockingQueueConsumer, pvParameters )
|
||||||
|
{
|
||||||
|
unsigned short usData, usExpectedValue = 0;
|
||||||
|
xBlockingQueueParameters *pxQueueParameters;
|
||||||
|
short sErrorEverOccurred = pdFALSE;
|
||||||
|
|
||||||
|
pxQueueParameters = ( xBlockingQueueParameters * ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
if( xQueueReceive( pxQueueParameters->xQueue, &usData, pxQueueParameters->xBlockTime ) == pdPASS )
|
||||||
|
{
|
||||||
|
if( usData != usExpectedValue )
|
||||||
|
{
|
||||||
|
/* Catch-up. */
|
||||||
|
usExpectedValue = usData;
|
||||||
|
|
||||||
|
sErrorEverOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* We have successfully received a message, so increment the
|
||||||
|
variable used to check we are still running. */
|
||||||
|
if( sErrorEverOccurred == pdFALSE )
|
||||||
|
{
|
||||||
|
( *pxQueueParameters->psCheckVariable )++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Increment the value we expect to remove from the queue next time
|
||||||
|
round. */
|
||||||
|
++usExpectedValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* This is called to check that all the created tasks are still running. */
|
||||||
|
portBASE_TYPE xAreBlockingQueuesStillRunning( void )
|
||||||
|
{
|
||||||
|
static short sLastBlockingConsumerCount[ blckqNUM_TASK_SETS ] = { ( unsigned short ) 0, ( unsigned short ) 0, ( unsigned short ) 0 };
|
||||||
|
static short sLastBlockingProducerCount[ blckqNUM_TASK_SETS ] = { ( unsigned short ) 0, ( unsigned short ) 0, ( unsigned short ) 0 };
|
||||||
|
portBASE_TYPE xReturn = pdPASS, xTasks;
|
||||||
|
|
||||||
|
/* Not too worried about mutual exclusion on these variables as they are 16
|
||||||
|
bits and we are only reading them. We also only care to see if they have
|
||||||
|
changed or not.
|
||||||
|
|
||||||
|
Loop through each check variable to and return pdFALSE if any are found not
|
||||||
|
to have changed since the last call. */
|
||||||
|
|
||||||
|
for( xTasks = 0; xTasks < blckqNUM_TASK_SETS; xTasks++ )
|
||||||
|
{
|
||||||
|
if( sBlockingConsumerCount[ xTasks ] == sLastBlockingConsumerCount[ xTasks ] )
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
sLastBlockingConsumerCount[ xTasks ] = sBlockingConsumerCount[ xTasks ];
|
||||||
|
|
||||||
|
|
||||||
|
if( sBlockingProducerCount[ xTasks ] == sLastBlockingProducerCount[ xTasks ] )
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
sLastBlockingProducerCount[ xTasks ] = sBlockingProducerCount[ xTasks ];
|
||||||
|
}
|
||||||
|
|
||||||
|
return xReturn;
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,572 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Tests the extra queue functionality introduced in FreeRTOS.org V4.5.0 -
|
||||||
|
* including xQueueSendToFront(), xQueueSendToBack(), xQueuePeek() and
|
||||||
|
* mutex behaviour.
|
||||||
|
*
|
||||||
|
* See the comments above the prvSendFrontAndBackTest() and
|
||||||
|
* prvLowPriorityMutexTask() prototypes below for more information.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
#include "queue.h"
|
||||||
|
#include "semphr.h"
|
||||||
|
|
||||||
|
/* Demo program include files. */
|
||||||
|
#include "GenQTest.h"
|
||||||
|
|
||||||
|
#define genqQUEUE_LENGTH ( 5 )
|
||||||
|
#define genqNO_BLOCK ( 0 )
|
||||||
|
|
||||||
|
#define genqMUTEX_LOW_PRIORITY ( tskIDLE_PRIORITY )
|
||||||
|
#define genqMUTEX_TEST_PRIORITY ( tskIDLE_PRIORITY + 1 )
|
||||||
|
#define genqMUTEX_MEDIUM_PRIORITY ( tskIDLE_PRIORITY + 2 )
|
||||||
|
#define genqMUTEX_HIGH_PRIORITY ( tskIDLE_PRIORITY + 3 )
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Tests the behaviour of the xQueueSendToFront() and xQueueSendToBack()
|
||||||
|
* macros by using both to fill a queue, then reading from the queue to
|
||||||
|
* check the resultant queue order is as expected. Queue data is also
|
||||||
|
* peeked.
|
||||||
|
*/
|
||||||
|
static void prvSendFrontAndBackTest( void *pvParameters );
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The following three tasks are used to demonstrate the mutex behaviour.
|
||||||
|
* Each task is given a different priority to demonstrate the priority
|
||||||
|
* inheritance mechanism.
|
||||||
|
*
|
||||||
|
* The low priority task obtains a mutex. After this a high priority task
|
||||||
|
* attempts to obtain the same mutex, causing its priority to be inherited
|
||||||
|
* by the low priority task. The task with the inherited high priority then
|
||||||
|
* resumes a medium priority task to ensure it is not blocked by the medium
|
||||||
|
* priority task while it holds the inherited high priority. Once the mutex
|
||||||
|
* is returned the task with the inherited priority returns to its original
|
||||||
|
* low priority, and is therefore immediately preempted by first the high
|
||||||
|
* priority task and then the medium prioroity task before it can continue.
|
||||||
|
*/
|
||||||
|
static void prvLowPriorityMutexTask( void *pvParameters );
|
||||||
|
static void prvMediumPriorityMutexTask( void *pvParameters );
|
||||||
|
static void prvHighPriorityMutexTask( void *pvParameters );
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* Flag that will be latched to pdTRUE should any unexpected behaviour be
|
||||||
|
detected in any of the tasks. */
|
||||||
|
static portBASE_TYPE xErrorDetected = pdFALSE;
|
||||||
|
|
||||||
|
/* Counters that are incremented on each cycle of a test. This is used to
|
||||||
|
detect a stalled task - a test that is no longer running. */
|
||||||
|
static volatile unsigned portLONG ulLoopCounter = 0;
|
||||||
|
static volatile unsigned portLONG ulLoopCounter2 = 0;
|
||||||
|
|
||||||
|
/* The variable that is guarded by the mutex in the mutex demo tasks. */
|
||||||
|
static volatile unsigned portLONG ulGuardedVariable = 0;
|
||||||
|
|
||||||
|
/* Handles used in the mutext test to suspend and resume the high and medium
|
||||||
|
priority mutex test tasks. */
|
||||||
|
static xTaskHandle xHighPriorityMutexTask, xMediumPriorityMutexTask;
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vStartGenericQueueTasks( unsigned portBASE_TYPE uxPriority )
|
||||||
|
{
|
||||||
|
xQueueHandle xQueue;
|
||||||
|
xSemaphoreHandle xMutex;
|
||||||
|
|
||||||
|
/* Create the queue that we are going to use for the
|
||||||
|
prvSendFrontAndBackTest demo. */
|
||||||
|
xQueue = xQueueCreate( genqQUEUE_LENGTH, sizeof( unsigned portLONG ) );
|
||||||
|
|
||||||
|
/* vQueueAddToRegistry() adds the queue to the queue registry, if one is
|
||||||
|
in use. The queue registry is provided as a means for kernel aware
|
||||||
|
debuggers to locate queues and has no purpose if a kernel aware debugger
|
||||||
|
is not being used. The call to vQueueAddToRegistry() will be removed
|
||||||
|
by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is
|
||||||
|
defined to be less than 1. */
|
||||||
|
vQueueAddToRegistry( xQueue, ( signed portCHAR * ) "Gen_Queue_Test" );
|
||||||
|
|
||||||
|
/* Create the demo task and pass it the queue just created. We are
|
||||||
|
passing the queue handle by value so it does not matter that it is
|
||||||
|
declared on the stack here. */
|
||||||
|
xTaskCreate( prvSendFrontAndBackTest, ( signed portCHAR * )"GenQ", configMINIMAL_STACK_SIZE, ( void * ) xQueue, uxPriority, NULL );
|
||||||
|
|
||||||
|
/* Create the mutex used by the prvMutexTest task. */
|
||||||
|
xMutex = xSemaphoreCreateMutex();
|
||||||
|
|
||||||
|
/* vQueueAddToRegistry() adds the mutex to the registry, if one is
|
||||||
|
in use. The registry is provided as a means for kernel aware
|
||||||
|
debuggers to locate mutexes and has no purpose if a kernel aware debugger
|
||||||
|
is not being used. The call to vQueueAddToRegistry() will be removed
|
||||||
|
by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is
|
||||||
|
defined to be less than 1. */
|
||||||
|
vQueueAddToRegistry( ( xQueueHandle ) xMutex, ( signed portCHAR * ) "Gen_Queue_Mutex" );
|
||||||
|
|
||||||
|
/* Create the mutex demo tasks and pass it the mutex just created. We are
|
||||||
|
passing the mutex handle by value so it does not matter that it is declared
|
||||||
|
on the stack here. */
|
||||||
|
xTaskCreate( prvLowPriorityMutexTask, ( signed portCHAR * )"MuLow", configMINIMAL_STACK_SIZE, ( void * ) xMutex, genqMUTEX_LOW_PRIORITY, NULL );
|
||||||
|
xTaskCreate( prvMediumPriorityMutexTask, ( signed portCHAR * )"MuMed", configMINIMAL_STACK_SIZE, NULL, genqMUTEX_MEDIUM_PRIORITY, &xMediumPriorityMutexTask );
|
||||||
|
xTaskCreate( prvHighPriorityMutexTask, ( signed portCHAR * )"MuHigh", configMINIMAL_STACK_SIZE, ( void * ) xMutex, genqMUTEX_HIGH_PRIORITY, &xHighPriorityMutexTask );
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvSendFrontAndBackTest( void *pvParameters )
|
||||||
|
{
|
||||||
|
unsigned portLONG ulData, ulData2;
|
||||||
|
xQueueHandle xQueue;
|
||||||
|
|
||||||
|
#ifdef USE_STDIO
|
||||||
|
void vPrintDisplayMessage( const portCHAR * const * ppcMessageToSend );
|
||||||
|
|
||||||
|
const portCHAR * const pcTaskStartMsg = "Queue SendToFront/SendToBack/Peek test started.\r\n";
|
||||||
|
|
||||||
|
/* Queue a message for printing to say the task has started. */
|
||||||
|
vPrintDisplayMessage( &pcTaskStartMsg );
|
||||||
|
#endif
|
||||||
|
|
||||||
|
xQueue = ( xQueueHandle ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* The queue is empty, so sending an item to the back of the queue
|
||||||
|
should have the same efect as sending it to the front of the queue.
|
||||||
|
|
||||||
|
First send to the front and check everything is as expected. */
|
||||||
|
xQueueSendToFront( xQueue, ( void * ) &ulLoopCounter, genqNO_BLOCK );
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 1 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xQueueReceive( xQueue, ( void * ) &ulData, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The data we sent to the queue should equal the data we just received
|
||||||
|
from the queue. */
|
||||||
|
if( ulLoopCounter != ulData )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Then do the same, sending the data to the back, checking everything
|
||||||
|
is as expected. */
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 0 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
xQueueSendToBack( xQueue, ( void * ) &ulLoopCounter, genqNO_BLOCK );
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 1 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xQueueReceive( xQueue, ( void * ) &ulData, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 0 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The data we sent to the queue should equal the data we just received
|
||||||
|
from the queue. */
|
||||||
|
if( ulLoopCounter != ulData )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Place 2, 3, 4 into the queue, adding items to the back of the queue. */
|
||||||
|
for( ulData = 2; ulData < 5; ulData++ )
|
||||||
|
{
|
||||||
|
xQueueSendToBack( xQueue, ( void * ) &ulData, genqNO_BLOCK );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now the order in the queue should be 2, 3, 4, with 2 being the first
|
||||||
|
thing to be read out. Now add 1 then 0 to the front of the queue. */
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 3 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
ulData = 1;
|
||||||
|
xQueueSendToFront( xQueue, ( void * ) &ulData, genqNO_BLOCK );
|
||||||
|
ulData = 0;
|
||||||
|
xQueueSendToFront( xQueue, ( void * ) &ulData, genqNO_BLOCK );
|
||||||
|
|
||||||
|
/* Now the queue should be full, and when we read the data out we
|
||||||
|
should receive 0, 1, 2, 3, 4. */
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 5 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xQueueSendToFront( xQueue, ( void * ) &ulData, genqNO_BLOCK ) != errQUEUE_FULL )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xQueueSendToBack( xQueue, ( void * ) &ulData, genqNO_BLOCK ) != errQUEUE_FULL )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Check the data we read out is in the expected order. */
|
||||||
|
for( ulData = 0; ulData < genqQUEUE_LENGTH; ulData++ )
|
||||||
|
{
|
||||||
|
/* Try peeking the data first. */
|
||||||
|
if( xQueuePeek( xQueue, &ulData2, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ulData != ulData2 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Now try receiving the data for real. The value should be the
|
||||||
|
same. Clobber the value first so we know we really received it. */
|
||||||
|
ulData2 = ~ulData2;
|
||||||
|
if( xQueueReceive( xQueue, &ulData2, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ulData != ulData2 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The queue should now be empty again. */
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 0 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/* Our queue is empty once more, add 10, 11 to the back. */
|
||||||
|
ulData = 10;
|
||||||
|
if( xQueueSend( xQueue, &ulData, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
ulData = 11;
|
||||||
|
if( xQueueSend( xQueue, &ulData, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 2 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now we should have 10, 11 in the queue. Add 7, 8, 9 to the
|
||||||
|
front. */
|
||||||
|
for( ulData = 9; ulData >= 7; ulData-- )
|
||||||
|
{
|
||||||
|
if( xQueueSendToFront( xQueue, ( void * ) &ulData, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now check that the queue is full, and that receiving data provides
|
||||||
|
the expected sequence of 7, 8, 9, 10, 11. */
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 5 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xQueueSendToFront( xQueue, ( void * ) &ulData, genqNO_BLOCK ) != errQUEUE_FULL )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xQueueSendToBack( xQueue, ( void * ) &ulData, genqNO_BLOCK ) != errQUEUE_FULL )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Check the data we read out is in the expected order. */
|
||||||
|
for( ulData = 7; ulData < ( 7 + genqQUEUE_LENGTH ); ulData++ )
|
||||||
|
{
|
||||||
|
if( xQueueReceive( xQueue, &ulData2, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ulData != ulData2 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 0 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
ulLoopCounter++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvLowPriorityMutexTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
xSemaphoreHandle xMutex = ( xSemaphoreHandle ) pvParameters;
|
||||||
|
|
||||||
|
#ifdef USE_STDIO
|
||||||
|
void vPrintDisplayMessage( const portCHAR * const * ppcMessageToSend );
|
||||||
|
|
||||||
|
const portCHAR * const pcTaskStartMsg = "Mutex with priority inheritance test started.\r\n";
|
||||||
|
|
||||||
|
/* Queue a message for printing to say the task has started. */
|
||||||
|
vPrintDisplayMessage( &pcTaskStartMsg );
|
||||||
|
#endif
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Take the mutex. It should be available now. */
|
||||||
|
if( xSemaphoreTake( xMutex, genqNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set our guarded variable to a known start value. */
|
||||||
|
ulGuardedVariable = 0;
|
||||||
|
|
||||||
|
/* Our priority should be as per that assigned when the task was
|
||||||
|
created. */
|
||||||
|
if( uxTaskPriorityGet( NULL ) != genqMUTEX_LOW_PRIORITY )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now unsuspend the high priority task. This will attempt to take the
|
||||||
|
mutex, and block when it finds it cannot obtain it. */
|
||||||
|
vTaskResume( xHighPriorityMutexTask );
|
||||||
|
|
||||||
|
/* We should now have inherited the prioritoy of the high priority task,
|
||||||
|
as by now it will have attempted to get the mutex. */
|
||||||
|
if( uxTaskPriorityGet( NULL ) != genqMUTEX_HIGH_PRIORITY )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We can attempt to set our priority to the test priority - between the
|
||||||
|
idle priority and the medium/high test priorities, but our actual
|
||||||
|
prioroity should remain at the high priority. */
|
||||||
|
vTaskPrioritySet( NULL, genqMUTEX_TEST_PRIORITY );
|
||||||
|
if( uxTaskPriorityGet( NULL ) != genqMUTEX_HIGH_PRIORITY )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now unsuspend the medium priority task. This should not run as our
|
||||||
|
inherited priority is above that of the medium priority task. */
|
||||||
|
vTaskResume( xMediumPriorityMutexTask );
|
||||||
|
|
||||||
|
/* If the did run then it will have incremented our guarded variable. */
|
||||||
|
if( ulGuardedVariable != 0 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When we give back the semaphore our priority should be disinherited
|
||||||
|
back to the priority to which we attempted to set ourselves. This means
|
||||||
|
that when the high priority task next blocks, the medium priority task
|
||||||
|
should execute and increment the guarded variable. When we next run
|
||||||
|
both the high and medium priority tasks will have been suspended again. */
|
||||||
|
if( xSemaphoreGive( xMutex ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check that the guarded variable did indeed increment... */
|
||||||
|
if( ulGuardedVariable != 1 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ... and that our priority has been disinherited to
|
||||||
|
genqMUTEX_TEST_PRIORITY. */
|
||||||
|
if( uxTaskPriorityGet( NULL ) != genqMUTEX_TEST_PRIORITY )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set our priority back to our original priority ready for the next
|
||||||
|
loop around this test. */
|
||||||
|
vTaskPrioritySet( NULL, genqMUTEX_LOW_PRIORITY );
|
||||||
|
|
||||||
|
/* Just to show we are still running. */
|
||||||
|
ulLoopCounter2++;
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvMediumPriorityMutexTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* The medium priority task starts by suspending itself. The low
|
||||||
|
priority task will unsuspend this task when required. */
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
|
||||||
|
/* When this task unsuspends all it does is increment the guarded
|
||||||
|
variable, this is so the low priority task knows that it has
|
||||||
|
executed. */
|
||||||
|
ulGuardedVariable++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvHighPriorityMutexTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
xSemaphoreHandle xMutex = ( xSemaphoreHandle ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* The high priority task starts by suspending itself. The low
|
||||||
|
priority task will unsuspend this task when required. */
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
|
||||||
|
/* When this task unsuspends all it does is attempt to obtain
|
||||||
|
the mutex. It should find the mutex is not available so a
|
||||||
|
block time is specified. */
|
||||||
|
if( xSemaphoreTake( xMutex, portMAX_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When we eventually obtain the mutex we just give it back then
|
||||||
|
return to suspend ready for the next test. */
|
||||||
|
if( xSemaphoreGive( xMutex ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* This is called to check that all the created tasks are still running. */
|
||||||
|
portBASE_TYPE xAreGenericQueueTasksStillRunning( void )
|
||||||
|
{
|
||||||
|
static unsigned portLONG ulLastLoopCounter = 0, ulLastLoopCounter2 = 0;
|
||||||
|
|
||||||
|
/* If the demo task is still running then we expect the loopcounters to
|
||||||
|
have incremented since this function was last called. */
|
||||||
|
if( ulLastLoopCounter == ulLoopCounter )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ulLastLoopCounter2 == ulLoopCounter2 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
ulLastLoopCounter = ulLoopCounter;
|
||||||
|
ulLastLoopCounter2 = ulLoopCounter2;
|
||||||
|
|
||||||
|
/* Errors detected in the task itself will have latched xErrorDetected
|
||||||
|
to true. */
|
||||||
|
|
||||||
|
return !xErrorDetected;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,246 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This version of PollQ. c is for use on systems that have limited stack
|
||||||
|
* space and no display facilities. The complete version can be found in
|
||||||
|
* the Demo/Common/Full directory.
|
||||||
|
*
|
||||||
|
* Creates two tasks that communicate over a single queue. One task acts as a
|
||||||
|
* producer, the other a consumer.
|
||||||
|
*
|
||||||
|
* The producer loops for three iteration, posting an incrementing number onto the
|
||||||
|
* queue each cycle. It then delays for a fixed period before doing exactly the
|
||||||
|
* same again.
|
||||||
|
*
|
||||||
|
* The consumer loops emptying the queue. Each item removed from the queue is
|
||||||
|
* checked to ensure it contains the expected value. When the queue is empty it
|
||||||
|
* blocks for a fixed period, then does the same again.
|
||||||
|
*
|
||||||
|
* All queue access is performed without blocking. The consumer completely empties
|
||||||
|
* the queue each time it runs so the producer should never find the queue full.
|
||||||
|
*
|
||||||
|
* An error is flagged if the consumer obtains an unexpected value or the producer
|
||||||
|
* find the queue is full.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Changes from V2.0.0
|
||||||
|
|
||||||
|
+ Delay periods are now specified using variables and constants of
|
||||||
|
portTickType rather than unsigned long.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
#include "queue.h"
|
||||||
|
|
||||||
|
/* Demo program include files. */
|
||||||
|
#include "PollQ.h"
|
||||||
|
|
||||||
|
#define pollqSTACK_SIZE configMINIMAL_STACK_SIZE
|
||||||
|
#define pollqQUEUE_SIZE ( 10 )
|
||||||
|
#define pollqPRODUCER_DELAY ( ( portTickType ) 200 / portTICK_RATE_MS )
|
||||||
|
#define pollqCONSUMER_DELAY ( pollqPRODUCER_DELAY - ( portTickType ) ( 20 / portTICK_RATE_MS ) )
|
||||||
|
#define pollqNO_DELAY ( ( portTickType ) 0 )
|
||||||
|
#define pollqVALUES_TO_PRODUCE ( ( signed portBASE_TYPE ) 3 )
|
||||||
|
#define pollqINITIAL_VALUE ( ( signed portBASE_TYPE ) 0 )
|
||||||
|
|
||||||
|
/* The task that posts the incrementing number onto the queue. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vPolledQueueProducer, pvParameters );
|
||||||
|
|
||||||
|
/* The task that empties the queue. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vPolledQueueConsumer, pvParameters );
|
||||||
|
|
||||||
|
/* Variables that are used to check that the tasks are still running with no
|
||||||
|
errors. */
|
||||||
|
static volatile signed portBASE_TYPE xPollingConsumerCount = pollqINITIAL_VALUE, xPollingProducerCount = pollqINITIAL_VALUE;
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vStartPolledQueueTasks( unsigned portBASE_TYPE uxPriority )
|
||||||
|
{
|
||||||
|
static xQueueHandle xPolledQueue;
|
||||||
|
|
||||||
|
/* Create the queue used by the producer and consumer. */
|
||||||
|
xPolledQueue = xQueueCreate( pollqQUEUE_SIZE, ( unsigned portBASE_TYPE ) sizeof( unsigned short ) );
|
||||||
|
|
||||||
|
/* vQueueAddToRegistry() adds the queue to the queue registry, if one is
|
||||||
|
in use. The queue registry is provided as a means for kernel aware
|
||||||
|
debuggers to locate queues and has no purpose if a kernel aware debugger
|
||||||
|
is not being used. The call to vQueueAddToRegistry() will be removed
|
||||||
|
by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is
|
||||||
|
defined to be less than 1. */
|
||||||
|
vQueueAddToRegistry( xPolledQueue, ( signed char * ) "Poll_Test_Queue" );
|
||||||
|
|
||||||
|
/* Spawn the producer and consumer. */
|
||||||
|
xTaskCreate( vPolledQueueConsumer, ( signed char * ) "QConsNB", pollqSTACK_SIZE, ( void * ) &xPolledQueue, uxPriority, ( xTaskHandle * ) NULL );
|
||||||
|
xTaskCreate( vPolledQueueProducer, ( signed char * ) "QProdNB", pollqSTACK_SIZE, ( void * ) &xPolledQueue, uxPriority, ( xTaskHandle * ) NULL );
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vPolledQueueProducer, pvParameters )
|
||||||
|
{
|
||||||
|
unsigned short usValue = ( unsigned short ) 0;
|
||||||
|
signed portBASE_TYPE xError = pdFALSE, xLoop;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
for( xLoop = 0; xLoop < pollqVALUES_TO_PRODUCE; xLoop++ )
|
||||||
|
{
|
||||||
|
/* Send an incrementing number on the queue without blocking. */
|
||||||
|
if( xQueueSend( *( ( xQueueHandle * ) pvParameters ), ( void * ) &usValue, pollqNO_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We should never find the queue full so if we get here there
|
||||||
|
has been an error. */
|
||||||
|
xError = pdTRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if( xError == pdFALSE )
|
||||||
|
{
|
||||||
|
/* If an error has ever been recorded we stop incrementing the
|
||||||
|
check variable. */
|
||||||
|
portENTER_CRITICAL();
|
||||||
|
xPollingProducerCount++;
|
||||||
|
portEXIT_CRITICAL();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Update the value we are going to post next time around. */
|
||||||
|
usValue++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Wait before we start posting again to ensure the consumer runs and
|
||||||
|
empties the queue. */
|
||||||
|
vTaskDelay( pollqPRODUCER_DELAY );
|
||||||
|
}
|
||||||
|
} /*lint !e818 Function prototype must conform to API. */
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vPolledQueueConsumer, pvParameters )
|
||||||
|
{
|
||||||
|
unsigned short usData, usExpectedValue = ( unsigned short ) 0;
|
||||||
|
signed portBASE_TYPE xError = pdFALSE;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Loop until the queue is empty. */
|
||||||
|
while( uxQueueMessagesWaiting( *( ( xQueueHandle * ) pvParameters ) ) )
|
||||||
|
{
|
||||||
|
if( xQueueReceive( *( ( xQueueHandle * ) pvParameters ), &usData, pollqNO_DELAY ) == pdPASS )
|
||||||
|
{
|
||||||
|
if( usData != usExpectedValue )
|
||||||
|
{
|
||||||
|
/* This is not what we expected to receive so an error has
|
||||||
|
occurred. */
|
||||||
|
xError = pdTRUE;
|
||||||
|
|
||||||
|
/* Catch-up to the value we received so our next expected
|
||||||
|
value should again be correct. */
|
||||||
|
usExpectedValue = usData;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if( xError == pdFALSE )
|
||||||
|
{
|
||||||
|
/* Only increment the check variable if no errors have
|
||||||
|
occurred. */
|
||||||
|
portENTER_CRITICAL();
|
||||||
|
xPollingConsumerCount++;
|
||||||
|
portEXIT_CRITICAL();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Next time round we would expect the number to be one higher. */
|
||||||
|
usExpectedValue++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now the queue is empty we block, allowing the producer to place more
|
||||||
|
items in the queue. */
|
||||||
|
vTaskDelay( pollqCONSUMER_DELAY );
|
||||||
|
}
|
||||||
|
} /*lint !e818 Function prototype must conform to API. */
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* This is called to check that all the created tasks are still running with no errors. */
|
||||||
|
portBASE_TYPE xArePollingQueuesStillRunning( void )
|
||||||
|
{
|
||||||
|
portBASE_TYPE xReturn;
|
||||||
|
|
||||||
|
/* Check both the consumer and producer poll count to check they have both
|
||||||
|
been changed since out last trip round. We do not need a critical section
|
||||||
|
around the check variables as this is called from a higher priority than
|
||||||
|
the other tasks that access the same variables. */
|
||||||
|
if( ( xPollingConsumerCount == pollqINITIAL_VALUE ) ||
|
||||||
|
( xPollingProducerCount == pollqINITIAL_VALUE )
|
||||||
|
)
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
xReturn = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set the check variables back down so we know if they have been
|
||||||
|
incremented the next time around. */
|
||||||
|
xPollingConsumerCount = pollqINITIAL_VALUE;
|
||||||
|
xPollingProducerCount = pollqINITIAL_VALUE;
|
||||||
|
|
||||||
|
return xReturn;
|
||||||
|
}
|
|
@ -0,0 +1,446 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Tests the behaviour when data is peeked from a queue when there are
|
||||||
|
* multiple tasks blocked on the queue.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
#include "queue.h"
|
||||||
|
#include "semphr.h"
|
||||||
|
|
||||||
|
/* Demo program include files. */
|
||||||
|
#include "QPeek.h"
|
||||||
|
|
||||||
|
#define qpeekQUEUE_LENGTH ( 5 )
|
||||||
|
#define qpeekNO_BLOCK ( 0 )
|
||||||
|
#define qpeekSHORT_DELAY ( 10 )
|
||||||
|
|
||||||
|
#define qpeekLOW_PRIORITY ( tskIDLE_PRIORITY + 0 )
|
||||||
|
#define qpeekMEDIUM_PRIORITY ( tskIDLE_PRIORITY + 1 )
|
||||||
|
#define qpeekHIGH_PRIORITY ( tskIDLE_PRIORITY + 2 )
|
||||||
|
#define qpeekHIGHEST_PRIORITY ( tskIDLE_PRIORITY + 3 )
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The following three tasks are used to demonstrate the peeking behaviour.
|
||||||
|
* Each task is given a different priority to demonstrate the order in which
|
||||||
|
* tasks are woken as data is peeked from a queue.
|
||||||
|
*/
|
||||||
|
static void prvLowPriorityPeekTask( void *pvParameters );
|
||||||
|
static void prvMediumPriorityPeekTask( void *pvParameters );
|
||||||
|
static void prvHighPriorityPeekTask( void *pvParameters );
|
||||||
|
static void prvHighestPriorityPeekTask( void *pvParameters );
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* Flag that will be latched to pdTRUE should any unexpected behaviour be
|
||||||
|
detected in any of the tasks. */
|
||||||
|
static volatile portBASE_TYPE xErrorDetected = pdFALSE;
|
||||||
|
|
||||||
|
/* Counter that is incremented on each cycle of a test. This is used to
|
||||||
|
detect a stalled task - a test that is no longer running. */
|
||||||
|
static volatile unsigned portLONG ulLoopCounter = 0;
|
||||||
|
|
||||||
|
/* Handles to the test tasks. */
|
||||||
|
xTaskHandle xMediumPriorityTask, xHighPriorityTask, xHighestPriorityTask;
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vStartQueuePeekTasks( void )
|
||||||
|
{
|
||||||
|
xQueueHandle xQueue;
|
||||||
|
|
||||||
|
/* Create the queue that we are going to use for the test/demo. */
|
||||||
|
xQueue = xQueueCreate( qpeekQUEUE_LENGTH, sizeof( unsigned portLONG ) );
|
||||||
|
|
||||||
|
/* vQueueAddToRegistry() adds the queue to the queue registry, if one is
|
||||||
|
in use. The queue registry is provided as a means for kernel aware
|
||||||
|
debuggers to locate queues and has no purpose if a kernel aware debugger
|
||||||
|
is not being used. The call to vQueueAddToRegistry() will be removed
|
||||||
|
by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is
|
||||||
|
defined to be less than 1. */
|
||||||
|
vQueueAddToRegistry( xQueue, ( signed portCHAR * ) "QPeek_Test_Queue" );
|
||||||
|
|
||||||
|
/* Create the demo tasks and pass it the queue just created. We are
|
||||||
|
passing the queue handle by value so it does not matter that it is declared
|
||||||
|
on the stack here. */
|
||||||
|
xTaskCreate( prvLowPriorityPeekTask, ( signed portCHAR * )"PeekL", configMINIMAL_STACK_SIZE, ( void * ) xQueue, qpeekLOW_PRIORITY, NULL );
|
||||||
|
xTaskCreate( prvMediumPriorityPeekTask, ( signed portCHAR * )"PeekM", configMINIMAL_STACK_SIZE, ( void * ) xQueue, qpeekMEDIUM_PRIORITY, &xMediumPriorityTask );
|
||||||
|
xTaskCreate( prvHighPriorityPeekTask, ( signed portCHAR * )"PeekH1", configMINIMAL_STACK_SIZE, ( void * ) xQueue, qpeekHIGH_PRIORITY, &xHighPriorityTask );
|
||||||
|
xTaskCreate( prvHighestPriorityPeekTask, ( signed portCHAR * )"PeekH2", configMINIMAL_STACK_SIZE, ( void * ) xQueue, qpeekHIGHEST_PRIORITY, &xHighestPriorityTask );
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvHighestPriorityPeekTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
xQueueHandle xQueue = ( xQueueHandle ) pvParameters;
|
||||||
|
unsigned portLONG ulValue;
|
||||||
|
|
||||||
|
#ifdef USE_STDIO
|
||||||
|
{
|
||||||
|
void vPrintDisplayMessage( const portCHAR * const * ppcMessageToSend );
|
||||||
|
|
||||||
|
const portCHAR * const pcTaskStartMsg = "Queue peek test started.\r\n";
|
||||||
|
|
||||||
|
/* Queue a message for printing to say the task has started. */
|
||||||
|
vPrintDisplayMessage( &pcTaskStartMsg );
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Try peeking from the queue. The queue should be empty so we will
|
||||||
|
block, allowing the high priority task to execute. */
|
||||||
|
if( xQueuePeek( xQueue, &ulValue, portMAX_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We expected to have received something by the time we unblock. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When we reach here the high and medium priority tasks should still
|
||||||
|
be blocked on the queue. We unblocked because the low priority task
|
||||||
|
wrote a value to the queue, which we should have peeked. Peeking the
|
||||||
|
data (rather than receiving it) will leave the data on the queue, so
|
||||||
|
the high priority task should then have also been unblocked, but not
|
||||||
|
yet executed. */
|
||||||
|
if( ulValue != 0x11223344 )
|
||||||
|
{
|
||||||
|
/* We did not receive the expected value. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 1 )
|
||||||
|
{
|
||||||
|
/* The message should have been left on the queue. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now we are going to actually receive the data, so when the high
|
||||||
|
priority task runs it will find the queue empty and return to the
|
||||||
|
blocked state. */
|
||||||
|
ulValue = 0;
|
||||||
|
if( xQueueReceive( xQueue, &ulValue, qpeekNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We expected to receive the value. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ulValue != 0x11223344 )
|
||||||
|
{
|
||||||
|
/* We did not receive the expected value - which should have been
|
||||||
|
the same value as was peeked. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now we will block again as the queue is once more empty. The low
|
||||||
|
priority task can then execute again. */
|
||||||
|
if( xQueuePeek( xQueue, &ulValue, portMAX_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We expected to have received something by the time we unblock. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When we get here the low priority task should have again written to the
|
||||||
|
queue. */
|
||||||
|
if( ulValue != 0x01234567 )
|
||||||
|
{
|
||||||
|
/* We did not receive the expected value. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 1 )
|
||||||
|
{
|
||||||
|
/* The message should have been left on the queue. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We only peeked the data, so suspending ourselves now should enable
|
||||||
|
the high priority task to also peek the data. The high priority task
|
||||||
|
will have been unblocked when we peeked the data as we left the data
|
||||||
|
in the queue. */
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* This time we are going to do the same as the above test, but the
|
||||||
|
high priority task is going to receive the data, rather than peek it.
|
||||||
|
This means that the medium priority task should never peek the value. */
|
||||||
|
if( xQueuePeek( xQueue, &ulValue, portMAX_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ulValue != 0xaabbaabb )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvHighPriorityPeekTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
xQueueHandle xQueue = ( xQueueHandle ) pvParameters;
|
||||||
|
unsigned portLONG ulValue;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Try peeking from the queue. The queue should be empty so we will
|
||||||
|
block, allowing the medium priority task to execute. Both the high
|
||||||
|
and highest priority tasks will then be blocked on the queue. */
|
||||||
|
if( xQueuePeek( xQueue, &ulValue, portMAX_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We expected to have received something by the time we unblock. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When we get here the highest priority task should have peeked the data
|
||||||
|
(unblocking this task) then suspended (allowing this task to also peek
|
||||||
|
the data). */
|
||||||
|
if( ulValue != 0x01234567 )
|
||||||
|
{
|
||||||
|
/* We did not receive the expected value. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 1 )
|
||||||
|
{
|
||||||
|
/* The message should have been left on the queue. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We only peeked the data, so suspending ourselves now should enable
|
||||||
|
the medium priority task to also peek the data. The medium priority task
|
||||||
|
will have been unblocked when we peeked the data as we left the data
|
||||||
|
in the queue. */
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
|
||||||
|
|
||||||
|
/* This time we are going actually receive the value, so the medium
|
||||||
|
priority task will never peek the data - we removed it from the queue. */
|
||||||
|
if( xQueueReceive( xQueue, &ulValue, portMAX_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ulValue != 0xaabbaabb )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvMediumPriorityPeekTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
xQueueHandle xQueue = ( xQueueHandle ) pvParameters;
|
||||||
|
unsigned portLONG ulValue;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Try peeking from the queue. The queue should be empty so we will
|
||||||
|
block, allowing the low priority task to execute. The highest, high
|
||||||
|
and medium priority tasks will then all be blocked on the queue. */
|
||||||
|
if( xQueuePeek( xQueue, &ulValue, portMAX_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We expected to have received something by the time we unblock. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When we get here the high priority task should have peeked the data
|
||||||
|
(unblocking this task) then suspended (allowing this task to also peek
|
||||||
|
the data). */
|
||||||
|
if( ulValue != 0x01234567 )
|
||||||
|
{
|
||||||
|
/* We did not receive the expected value. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 1 )
|
||||||
|
{
|
||||||
|
/* The message should have been left on the queue. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Just so we know the test is still running. */
|
||||||
|
ulLoopCounter++;
|
||||||
|
|
||||||
|
/* Now we can suspend ourselves so the low priority task can execute
|
||||||
|
again. */
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvLowPriorityPeekTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
xQueueHandle xQueue = ( xQueueHandle ) pvParameters;
|
||||||
|
unsigned portLONG ulValue;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Write some data to the queue. This should unblock the highest
|
||||||
|
priority task that is waiting to peek data from the queue. */
|
||||||
|
ulValue = 0x11223344;
|
||||||
|
if( xQueueSendToBack( xQueue, &ulValue, qpeekNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We were expecting the queue to be empty so we should not of
|
||||||
|
had a problem writing to the queue. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* By the time we get here the data should have been removed from
|
||||||
|
the queue. */
|
||||||
|
if( uxQueueMessagesWaiting( xQueue ) != 0 )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Write another value to the queue, again waking the highest priority
|
||||||
|
task that is blocked on the queue. */
|
||||||
|
ulValue = 0x01234567;
|
||||||
|
if( xQueueSendToBack( xQueue, &ulValue, qpeekNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We were expecting the queue to be empty so we should not of
|
||||||
|
had a problem writing to the queue. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* All the other tasks should now have successfully peeked the data.
|
||||||
|
The data is still in the queue so we should be able to receive it. */
|
||||||
|
ulValue = 0;
|
||||||
|
if( xQueueReceive( xQueue, &ulValue, qpeekNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We expected to receive the data. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ulValue != 0x01234567 )
|
||||||
|
{
|
||||||
|
/* We did not receive the expected value. */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Lets just delay a while as this is an intensive test as we don't
|
||||||
|
want to starve other tests of processing time. */
|
||||||
|
vTaskDelay( qpeekSHORT_DELAY );
|
||||||
|
|
||||||
|
/* Unsuspend the other tasks so we can repeat the test - this time
|
||||||
|
however not all the other tasks will peek the data as the high
|
||||||
|
priority task is actually going to remove it from the queue. Send
|
||||||
|
to front is used just to be different. As the queue is empty it
|
||||||
|
makes no difference to the result. */
|
||||||
|
vTaskResume( xMediumPriorityTask );
|
||||||
|
vTaskResume( xHighPriorityTask );
|
||||||
|
vTaskResume( xHighestPriorityTask );
|
||||||
|
|
||||||
|
ulValue = 0xaabbaabb;
|
||||||
|
if( xQueueSendToFront( xQueue, &ulValue, qpeekNO_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
/* We were expecting the queue to be empty so we should not of
|
||||||
|
had a problem writing to the queue. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This time we should find that the queue is empty. The high priority
|
||||||
|
task actually removed the data rather than just peeking it. */
|
||||||
|
if( xQueuePeek( xQueue, &ulValue, qpeekNO_BLOCK ) != errQUEUE_EMPTY )
|
||||||
|
{
|
||||||
|
/* We expected to receive the data. */
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Unsuspend the highest and high priority tasks so we can go back
|
||||||
|
and repeat the whole thing. The medium priority task should not be
|
||||||
|
suspended as it was not able to peek the data in this last case. */
|
||||||
|
vTaskResume( xHighPriorityTask );
|
||||||
|
vTaskResume( xHighestPriorityTask );
|
||||||
|
|
||||||
|
/* Lets just delay a while as this is an intensive test as we don't
|
||||||
|
want to starve other tests of processing time. */
|
||||||
|
vTaskDelay( qpeekSHORT_DELAY );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* This is called to check that all the created tasks are still running. */
|
||||||
|
portBASE_TYPE xAreQueuePeekTasksStillRunning( void )
|
||||||
|
{
|
||||||
|
static unsigned portLONG ulLastLoopCounter = 0;
|
||||||
|
|
||||||
|
/* If the demo task is still running then we expect the loopcounter to
|
||||||
|
have incremented since this function was last called. */
|
||||||
|
if( ulLastLoopCounter == ulLoopCounter )
|
||||||
|
{
|
||||||
|
xErrorDetected = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
ulLastLoopCounter = ulLoopCounter;
|
||||||
|
|
||||||
|
/* Errors detected in the task itself will have latched xErrorDetected
|
||||||
|
to true. */
|
||||||
|
|
||||||
|
return !xErrorDetected;
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,493 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This file contains some test scenarios that ensure tasks do not exit queue
|
||||||
|
* send or receive functions prematurely. A description of the tests is
|
||||||
|
* included within the code.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Kernel includes. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
#include "queue.h"
|
||||||
|
|
||||||
|
/* Demo includes. */
|
||||||
|
#include "blocktim.h"
|
||||||
|
|
||||||
|
/* Task priorities. Allow these to be overridden. */
|
||||||
|
#ifndef bktPRIMARY_PRIORITY
|
||||||
|
#define bktPRIMARY_PRIORITY ( configMAX_PRIORITIES - 3 )
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef bktSECONDARY_PRIORITY
|
||||||
|
#define bktSECONDARY_PRIORITY ( configMAX_PRIORITIES - 4 )
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Task behaviour. */
|
||||||
|
#define bktQUEUE_LENGTH ( 5 )
|
||||||
|
#define bktSHORT_WAIT ( ( ( portTickType ) 20 ) / portTICK_RATE_MS )
|
||||||
|
#define bktPRIMARY_BLOCK_TIME ( 10 )
|
||||||
|
#define bktALLOWABLE_MARGIN ( 15 )
|
||||||
|
#define bktTIME_TO_BLOCK ( 175 )
|
||||||
|
#define bktDONT_BLOCK ( ( portTickType ) 0 )
|
||||||
|
#define bktRUN_INDICATOR ( ( unsigned portBASE_TYPE ) 0x55 )
|
||||||
|
|
||||||
|
/* The queue on which the tasks block. */
|
||||||
|
static xQueueHandle xTestQueue;
|
||||||
|
|
||||||
|
/* Handle to the secondary task is required by the primary task for calls
|
||||||
|
to vTaskSuspend/Resume(). */
|
||||||
|
static xTaskHandle xSecondary;
|
||||||
|
|
||||||
|
/* Used to ensure that tasks are still executing without error. */
|
||||||
|
static volatile portBASE_TYPE xPrimaryCycles = 0, xSecondaryCycles = 0;
|
||||||
|
static volatile portBASE_TYPE xErrorOccurred = pdFALSE;
|
||||||
|
|
||||||
|
/* Provides a simple mechanism for the primary task to know when the
|
||||||
|
secondary task has executed. */
|
||||||
|
static volatile unsigned portBASE_TYPE xRunIndicator;
|
||||||
|
|
||||||
|
/* The two test tasks. Their behaviour is commented within the files. */
|
||||||
|
static void vPrimaryBlockTimeTestTask( void *pvParameters );
|
||||||
|
static void vSecondaryBlockTimeTestTask( void *pvParameters );
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vCreateBlockTimeTasks( void )
|
||||||
|
{
|
||||||
|
/* Create the queue on which the two tasks block. */
|
||||||
|
xTestQueue = xQueueCreate( bktQUEUE_LENGTH, sizeof( portBASE_TYPE ) );
|
||||||
|
|
||||||
|
/* vQueueAddToRegistry() adds the queue to the queue registry, if one is
|
||||||
|
in use. The queue registry is provided as a means for kernel aware
|
||||||
|
debuggers to locate queues and has no purpose if a kernel aware debugger
|
||||||
|
is not being used. The call to vQueueAddToRegistry() will be removed
|
||||||
|
by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is
|
||||||
|
defined to be less than 1. */
|
||||||
|
vQueueAddToRegistry( xTestQueue, ( signed char * ) "Block_Time_Queue" );
|
||||||
|
|
||||||
|
/* Create the two test tasks. */
|
||||||
|
xTaskCreate( vPrimaryBlockTimeTestTask, ( signed char * )"BTest1", configMINIMAL_STACK_SIZE, NULL, bktPRIMARY_PRIORITY, NULL );
|
||||||
|
xTaskCreate( vSecondaryBlockTimeTestTask, ( signed char * )"BTest2", configMINIMAL_STACK_SIZE, NULL, bktSECONDARY_PRIORITY, &xSecondary );
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void vPrimaryBlockTimeTestTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
portBASE_TYPE xItem, xData;
|
||||||
|
portTickType xTimeWhenBlocking;
|
||||||
|
portTickType xTimeToBlock, xBlockedTime;
|
||||||
|
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/*********************************************************************
|
||||||
|
Test 1
|
||||||
|
|
||||||
|
Simple block time wakeup test on queue receives. */
|
||||||
|
for( xItem = 0; xItem < bktQUEUE_LENGTH; xItem++ )
|
||||||
|
{
|
||||||
|
/* The queue is empty. Attempt to read from the queue using a block
|
||||||
|
time. When we wake, ensure the delta in time is as expected. */
|
||||||
|
xTimeToBlock = bktPRIMARY_BLOCK_TIME << xItem;
|
||||||
|
|
||||||
|
xTimeWhenBlocking = xTaskGetTickCount();
|
||||||
|
|
||||||
|
/* We should unblock after xTimeToBlock having not received
|
||||||
|
anything on the queue. */
|
||||||
|
if( xQueueReceive( xTestQueue, &xData, xTimeToBlock ) != errQUEUE_EMPTY )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* How long were we blocked for? */
|
||||||
|
xBlockedTime = xTaskGetTickCount() - xTimeWhenBlocking;
|
||||||
|
|
||||||
|
if( xBlockedTime < xTimeToBlock )
|
||||||
|
{
|
||||||
|
/* Should not have blocked for less than we requested. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xBlockedTime > ( xTimeToBlock + bktALLOWABLE_MARGIN ) )
|
||||||
|
{
|
||||||
|
/* Should not have blocked for longer than we requested,
|
||||||
|
although we would not necessarily run as soon as we were
|
||||||
|
unblocked so a margin is allowed. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*********************************************************************
|
||||||
|
Test 2
|
||||||
|
|
||||||
|
Simple block time wakeup test on queue sends.
|
||||||
|
|
||||||
|
First fill the queue. It should be empty so all sends should pass. */
|
||||||
|
for( xItem = 0; xItem < bktQUEUE_LENGTH; xItem++ )
|
||||||
|
{
|
||||||
|
if( xQueueSend( xTestQueue, &xItem, bktDONT_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
for( xItem = 0; xItem < bktQUEUE_LENGTH; xItem++ )
|
||||||
|
{
|
||||||
|
/* The queue is full. Attempt to write to the queue using a block
|
||||||
|
time. When we wake, ensure the delta in time is as expected. */
|
||||||
|
xTimeToBlock = bktPRIMARY_BLOCK_TIME << xItem;
|
||||||
|
|
||||||
|
xTimeWhenBlocking = xTaskGetTickCount();
|
||||||
|
|
||||||
|
/* We should unblock after xTimeToBlock having not received
|
||||||
|
anything on the queue. */
|
||||||
|
if( xQueueSend( xTestQueue, &xItem, xTimeToBlock ) != errQUEUE_FULL )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* How long were we blocked for? */
|
||||||
|
xBlockedTime = xTaskGetTickCount() - xTimeWhenBlocking;
|
||||||
|
|
||||||
|
if( xBlockedTime < xTimeToBlock )
|
||||||
|
{
|
||||||
|
/* Should not have blocked for less than we requested. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xBlockedTime > ( xTimeToBlock + bktALLOWABLE_MARGIN ) )
|
||||||
|
{
|
||||||
|
/* Should not have blocked for longer than we requested,
|
||||||
|
although we would not necessarily run as soon as we were
|
||||||
|
unblocked so a margin is allowed. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*********************************************************************
|
||||||
|
Test 3
|
||||||
|
|
||||||
|
Wake the other task, it will block attempting to post to the queue.
|
||||||
|
When we read from the queue the other task will wake, but before it
|
||||||
|
can run we will post to the queue again. When the other task runs it
|
||||||
|
will find the queue still full, even though it was woken. It should
|
||||||
|
recognise that its block time has not expired and return to block for
|
||||||
|
the remains of its block time.
|
||||||
|
|
||||||
|
Wake the other task so it blocks attempting to post to the already
|
||||||
|
full queue. */
|
||||||
|
xRunIndicator = 0;
|
||||||
|
vTaskResume( xSecondary );
|
||||||
|
|
||||||
|
/* We need to wait a little to ensure the other task executes. */
|
||||||
|
while( xRunIndicator != bktRUN_INDICATOR )
|
||||||
|
{
|
||||||
|
/* The other task has not yet executed. */
|
||||||
|
vTaskDelay( bktSHORT_WAIT );
|
||||||
|
}
|
||||||
|
/* Make sure the other task is blocked on the queue. */
|
||||||
|
vTaskDelay( bktSHORT_WAIT );
|
||||||
|
xRunIndicator = 0;
|
||||||
|
|
||||||
|
for( xItem = 0; xItem < bktQUEUE_LENGTH; xItem++ )
|
||||||
|
{
|
||||||
|
/* Now when we make space on the queue the other task should wake
|
||||||
|
but not execute as this task has higher priority. */
|
||||||
|
if( xQueueReceive( xTestQueue, &xData, bktDONT_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now fill the queue again before the other task gets a chance to
|
||||||
|
execute. If the other task had executed we would find the queue
|
||||||
|
full ourselves, and the other task have set xRunIndicator. */
|
||||||
|
if( xQueueSend( xTestQueue, &xItem, bktDONT_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xRunIndicator == bktRUN_INDICATOR )
|
||||||
|
{
|
||||||
|
/* The other task should not have executed. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Raise the priority of the other task so it executes and blocks
|
||||||
|
on the queue again. */
|
||||||
|
vTaskPrioritySet( xSecondary, bktPRIMARY_PRIORITY + 2 );
|
||||||
|
|
||||||
|
/* The other task should now have re-blocked without exiting the
|
||||||
|
queue function. */
|
||||||
|
if( xRunIndicator == bktRUN_INDICATOR )
|
||||||
|
{
|
||||||
|
/* The other task should not have executed outside of the
|
||||||
|
queue function. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set the priority back down. */
|
||||||
|
vTaskPrioritySet( xSecondary, bktSECONDARY_PRIORITY );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Let the other task timeout. When it unblockes it will check that it
|
||||||
|
unblocked at the correct time, then suspend itself. */
|
||||||
|
while( xRunIndicator != bktRUN_INDICATOR )
|
||||||
|
{
|
||||||
|
vTaskDelay( bktSHORT_WAIT );
|
||||||
|
}
|
||||||
|
vTaskDelay( bktSHORT_WAIT );
|
||||||
|
xRunIndicator = 0;
|
||||||
|
|
||||||
|
|
||||||
|
/*********************************************************************
|
||||||
|
Test 4
|
||||||
|
|
||||||
|
As per test 3 - but with the send and receive the other way around.
|
||||||
|
The other task blocks attempting to read from the queue.
|
||||||
|
|
||||||
|
Empty the queue. We should find that it is full. */
|
||||||
|
for( xItem = 0; xItem < bktQUEUE_LENGTH; xItem++ )
|
||||||
|
{
|
||||||
|
if( xQueueReceive( xTestQueue, &xData, bktDONT_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Wake the other task so it blocks attempting to read from the
|
||||||
|
already empty queue. */
|
||||||
|
vTaskResume( xSecondary );
|
||||||
|
|
||||||
|
/* We need to wait a little to ensure the other task executes. */
|
||||||
|
while( xRunIndicator != bktRUN_INDICATOR )
|
||||||
|
{
|
||||||
|
vTaskDelay( bktSHORT_WAIT );
|
||||||
|
}
|
||||||
|
vTaskDelay( bktSHORT_WAIT );
|
||||||
|
xRunIndicator = 0;
|
||||||
|
|
||||||
|
for( xItem = 0; xItem < bktQUEUE_LENGTH; xItem++ )
|
||||||
|
{
|
||||||
|
/* Now when we place an item on the queue the other task should
|
||||||
|
wake but not execute as this task has higher priority. */
|
||||||
|
if( xQueueSend( xTestQueue, &xItem, bktDONT_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now empty the queue again before the other task gets a chance to
|
||||||
|
execute. If the other task had executed we would find the queue
|
||||||
|
empty ourselves, and the other task would be suspended. */
|
||||||
|
if( xQueueReceive( xTestQueue, &xData, bktDONT_BLOCK ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xRunIndicator == bktRUN_INDICATOR )
|
||||||
|
{
|
||||||
|
/* The other task should not have executed. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Raise the priority of the other task so it executes and blocks
|
||||||
|
on the queue again. */
|
||||||
|
vTaskPrioritySet( xSecondary, bktPRIMARY_PRIORITY + 2 );
|
||||||
|
|
||||||
|
/* The other task should now have re-blocked without exiting the
|
||||||
|
queue function. */
|
||||||
|
if( xRunIndicator == bktRUN_INDICATOR )
|
||||||
|
{
|
||||||
|
/* The other task should not have executed outside of the
|
||||||
|
queue function. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
vTaskPrioritySet( xSecondary, bktSECONDARY_PRIORITY );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Let the other task timeout. When it unblockes it will check that it
|
||||||
|
unblocked at the correct time, then suspend itself. */
|
||||||
|
while( xRunIndicator != bktRUN_INDICATOR )
|
||||||
|
{
|
||||||
|
vTaskDelay( bktSHORT_WAIT );
|
||||||
|
}
|
||||||
|
vTaskDelay( bktSHORT_WAIT );
|
||||||
|
|
||||||
|
xPrimaryCycles++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void vSecondaryBlockTimeTestTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
portTickType xTimeWhenBlocking, xBlockedTime;
|
||||||
|
portBASE_TYPE xData;
|
||||||
|
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/*********************************************************************
|
||||||
|
Test 1 and 2
|
||||||
|
|
||||||
|
This task does does not participate in these tests. */
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
|
||||||
|
/*********************************************************************
|
||||||
|
Test 3
|
||||||
|
|
||||||
|
The first thing we do is attempt to read from the queue. It should be
|
||||||
|
full so we block. Note the time before we block so we can check the
|
||||||
|
wake time is as per that expected. */
|
||||||
|
xTimeWhenBlocking = xTaskGetTickCount();
|
||||||
|
|
||||||
|
/* We should unblock after bktTIME_TO_BLOCK having not sent
|
||||||
|
anything to the queue. */
|
||||||
|
xData = 0;
|
||||||
|
xRunIndicator = bktRUN_INDICATOR;
|
||||||
|
if( xQueueSend( xTestQueue, &xData, bktTIME_TO_BLOCK ) != errQUEUE_FULL )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* How long were we inside the send function? */
|
||||||
|
xBlockedTime = xTaskGetTickCount() - xTimeWhenBlocking;
|
||||||
|
|
||||||
|
/* We should not have blocked for less time than bktTIME_TO_BLOCK. */
|
||||||
|
if( xBlockedTime < bktTIME_TO_BLOCK )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We should of not blocked for much longer than bktALLOWABLE_MARGIN
|
||||||
|
either. A margin is permitted as we would not necessarily run as
|
||||||
|
soon as we unblocked. */
|
||||||
|
if( xBlockedTime > ( bktTIME_TO_BLOCK + bktALLOWABLE_MARGIN ) )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Suspend ready for test 3. */
|
||||||
|
xRunIndicator = bktRUN_INDICATOR;
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
|
||||||
|
/*********************************************************************
|
||||||
|
Test 4
|
||||||
|
|
||||||
|
As per test three, but with the send and receive reversed. */
|
||||||
|
xTimeWhenBlocking = xTaskGetTickCount();
|
||||||
|
|
||||||
|
/* We should unblock after bktTIME_TO_BLOCK having not received
|
||||||
|
anything on the queue. */
|
||||||
|
xRunIndicator = bktRUN_INDICATOR;
|
||||||
|
if( xQueueReceive( xTestQueue, &xData, bktTIME_TO_BLOCK ) != errQUEUE_EMPTY )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
xBlockedTime = xTaskGetTickCount() - xTimeWhenBlocking;
|
||||||
|
|
||||||
|
/* We should not have blocked for less time than bktTIME_TO_BLOCK. */
|
||||||
|
if( xBlockedTime < bktTIME_TO_BLOCK )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We should of not blocked for much longer than bktALLOWABLE_MARGIN
|
||||||
|
either. A margin is permitted as we would not necessarily run as soon
|
||||||
|
as we unblocked. */
|
||||||
|
if( xBlockedTime > ( bktTIME_TO_BLOCK + bktALLOWABLE_MARGIN ) )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
xRunIndicator = bktRUN_INDICATOR;
|
||||||
|
|
||||||
|
xSecondaryCycles++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
portBASE_TYPE xAreBlockTimeTestTasksStillRunning( void )
|
||||||
|
{
|
||||||
|
static portBASE_TYPE xLastPrimaryCycleCount = 0, xLastSecondaryCycleCount = 0;
|
||||||
|
portBASE_TYPE xReturn = pdPASS;
|
||||||
|
|
||||||
|
/* Have both tasks performed at least one cycle since this function was
|
||||||
|
last called? */
|
||||||
|
if( xPrimaryCycles == xLastPrimaryCycleCount )
|
||||||
|
{
|
||||||
|
xReturn = pdFAIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xSecondaryCycles == xLastSecondaryCycleCount )
|
||||||
|
{
|
||||||
|
xReturn = pdFAIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xErrorOccurred == pdTRUE )
|
||||||
|
{
|
||||||
|
xReturn = pdFAIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
xLastSecondaryCycleCount = xSecondaryCycles;
|
||||||
|
xLastPrimaryCycleCount = xPrimaryCycles;
|
||||||
|
|
||||||
|
return xReturn;
|
||||||
|
}
|
|
@ -0,0 +1,255 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a single persistent task which periodically dynamically creates another
|
||||||
|
* two tasks. The original task is called the creator task, the two tasks it
|
||||||
|
* creates are called suicidal tasks.
|
||||||
|
*
|
||||||
|
* One of the created suicidal tasks kill one other suicidal task before killing
|
||||||
|
* itself - leaving just the original task remaining.
|
||||||
|
*
|
||||||
|
* The creator task must be spawned after all of the other demo application tasks
|
||||||
|
* as it keeps a check on the number of tasks under the scheduler control. The
|
||||||
|
* number of tasks it expects to see running should never be greater than the
|
||||||
|
* number of tasks that were in existence when the creator task was spawned, plus
|
||||||
|
* one set of four suicidal tasks. If this number is exceeded an error is flagged.
|
||||||
|
*
|
||||||
|
* \page DeathC death.c
|
||||||
|
* \ingroup DemoFiles
|
||||||
|
* <HR>
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Changes from V3.0.0
|
||||||
|
+ CreationCount sizes changed from unsigned portBASE_TYPE to
|
||||||
|
unsigned short to minimize the risk of overflowing.
|
||||||
|
|
||||||
|
+ Reset of usLastCreationCount added
|
||||||
|
|
||||||
|
Changes from V3.1.0
|
||||||
|
+ Changed the dummy calculation to use variables of type long, rather than
|
||||||
|
float. This allows the file to be used with ports that do not support
|
||||||
|
floating point.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
|
||||||
|
/* Demo program include files. */
|
||||||
|
#include "death.h"
|
||||||
|
|
||||||
|
#define deathSTACK_SIZE ( configMINIMAL_STACK_SIZE + 60 )
|
||||||
|
|
||||||
|
/* The task originally created which is responsible for periodically dynamically
|
||||||
|
creating another four tasks. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vCreateTasks, pvParameters );
|
||||||
|
|
||||||
|
/* The task function of the dynamically created tasks. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vSuicidalTask, pvParameters );
|
||||||
|
|
||||||
|
/* A variable which is incremented every time the dynamic tasks are created. This
|
||||||
|
is used to check that the task is still running. */
|
||||||
|
static volatile unsigned short usCreationCount = 0;
|
||||||
|
|
||||||
|
/* Used to store the number of tasks that were originally running so the creator
|
||||||
|
task can tell if any of the suicidal tasks have failed to die.
|
||||||
|
*/
|
||||||
|
static volatile unsigned portBASE_TYPE uxTasksRunningAtStart = 0;
|
||||||
|
|
||||||
|
/* Tasks are deleted by the idle task. Under heavy load the idle task might
|
||||||
|
not get much processing time, so it would be legitimate for several tasks to
|
||||||
|
remain undeleted for a short period. */
|
||||||
|
static const unsigned portBASE_TYPE uxMaxNumberOfExtraTasksRunning = 2;
|
||||||
|
|
||||||
|
/* Used to store a handle to the task that should be killed by a suicidal task,
|
||||||
|
before it kills itself. */
|
||||||
|
xTaskHandle xCreatedTask;
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vCreateSuicidalTasks( unsigned portBASE_TYPE uxPriority )
|
||||||
|
{
|
||||||
|
unsigned portBASE_TYPE *puxPriority;
|
||||||
|
|
||||||
|
/* Create the Creator tasks - passing in as a parameter the priority at which
|
||||||
|
the suicidal tasks should be created. */
|
||||||
|
puxPriority = ( unsigned portBASE_TYPE * ) pvPortMalloc( sizeof( unsigned portBASE_TYPE ) );
|
||||||
|
*puxPriority = uxPriority;
|
||||||
|
|
||||||
|
xTaskCreate( vCreateTasks, ( signed char * ) "CREATOR", deathSTACK_SIZE, ( void * ) puxPriority, uxPriority, NULL );
|
||||||
|
|
||||||
|
/* Record the number of tasks that are running now so we know if any of the
|
||||||
|
suicidal tasks have failed to be killed. */
|
||||||
|
uxTasksRunningAtStart = ( unsigned portBASE_TYPE ) uxTaskGetNumberOfTasks();
|
||||||
|
|
||||||
|
/* FreeRTOS.org versions before V3.0 started the idle-task as the very
|
||||||
|
first task. The idle task was then already included in uxTasksRunningAtStart.
|
||||||
|
From FreeRTOS V3.0 on, the idle task is started when the scheduler is
|
||||||
|
started. Therefore the idle task is not yet accounted for. We correct
|
||||||
|
this by increasing uxTasksRunningAtStart by 1. */
|
||||||
|
uxTasksRunningAtStart++;
|
||||||
|
|
||||||
|
/* From FreeRTOS version 7.0.0 can optionally create a timer service task.
|
||||||
|
If this is done, then uxTasksRunningAtStart needs incrementing again as that
|
||||||
|
too is created when the scheduler is started. */
|
||||||
|
#if configUSE_TIMERS == 1
|
||||||
|
uxTasksRunningAtStart++;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vSuicidalTask, pvParameters )
|
||||||
|
{
|
||||||
|
volatile long l1, l2;
|
||||||
|
xTaskHandle xTaskToKill;
|
||||||
|
const portTickType xDelay = ( portTickType ) 200 / portTICK_RATE_MS;
|
||||||
|
|
||||||
|
if( pvParameters != NULL )
|
||||||
|
{
|
||||||
|
/* This task is periodically created four times. Two created tasks are
|
||||||
|
passed a handle to the other task so it can kill it before killing itself.
|
||||||
|
The other task is passed in null. */
|
||||||
|
xTaskToKill = *( xTaskHandle* )pvParameters;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
xTaskToKill = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Do something random just to use some stack and registers. */
|
||||||
|
l1 = 2;
|
||||||
|
l2 = 89;
|
||||||
|
l2 *= l1;
|
||||||
|
vTaskDelay( xDelay );
|
||||||
|
|
||||||
|
if( xTaskToKill != NULL )
|
||||||
|
{
|
||||||
|
/* Make sure the other task has a go before we delete it. */
|
||||||
|
vTaskDelay( ( portTickType ) 0 );
|
||||||
|
|
||||||
|
/* Kill the other task that was created by vCreateTasks(). */
|
||||||
|
vTaskDelete( xTaskToKill );
|
||||||
|
|
||||||
|
/* Kill ourselves. */
|
||||||
|
vTaskDelete( NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}/*lint !e818 !e550 Function prototype must be as per standard for task functions. */
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vCreateTasks, pvParameters )
|
||||||
|
{
|
||||||
|
const portTickType xDelay = ( portTickType ) 1000 / portTICK_RATE_MS;
|
||||||
|
unsigned portBASE_TYPE uxPriority;
|
||||||
|
|
||||||
|
uxPriority = *( unsigned portBASE_TYPE * ) pvParameters;
|
||||||
|
vPortFree( pvParameters );
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Just loop round, delaying then creating the four suicidal tasks. */
|
||||||
|
vTaskDelay( xDelay );
|
||||||
|
|
||||||
|
xCreatedTask = NULL;
|
||||||
|
|
||||||
|
xTaskCreate( vSuicidalTask, ( signed char * ) "SUICID1", configMINIMAL_STACK_SIZE, NULL, uxPriority, &xCreatedTask );
|
||||||
|
xTaskCreate( vSuicidalTask, ( signed char * ) "SUICID2", configMINIMAL_STACK_SIZE, &xCreatedTask, uxPriority, NULL );
|
||||||
|
|
||||||
|
++usCreationCount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* This is called to check that the creator task is still running and that there
|
||||||
|
are not any more than four extra tasks. */
|
||||||
|
portBASE_TYPE xIsCreateTaskStillRunning( void )
|
||||||
|
{
|
||||||
|
static unsigned short usLastCreationCount = 0xfff;
|
||||||
|
portBASE_TYPE xReturn = pdTRUE;
|
||||||
|
static unsigned portBASE_TYPE uxTasksRunningNow;
|
||||||
|
|
||||||
|
if( usLastCreationCount == usCreationCount )
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
usLastCreationCount = usCreationCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
uxTasksRunningNow = ( unsigned portBASE_TYPE ) uxTaskGetNumberOfTasks();
|
||||||
|
|
||||||
|
if( uxTasksRunningNow < uxTasksRunningAtStart )
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
else if( ( uxTasksRunningNow - uxTasksRunningAtStart ) > uxMaxNumberOfExtraTasksRunning )
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Everything is okay. */
|
||||||
|
}
|
||||||
|
|
||||||
|
return xReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,427 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The first test creates three tasks - two counter tasks (one continuous count
|
||||||
|
* and one limited count) and one controller. A "count" variable is shared
|
||||||
|
* between all three tasks. The two counter tasks should never be in a "ready"
|
||||||
|
* state at the same time. The controller task runs at the same priority as
|
||||||
|
* the continuous count task, and at a lower priority than the limited count
|
||||||
|
* task.
|
||||||
|
*
|
||||||
|
* One counter task loops indefinitely, incrementing the shared count variable
|
||||||
|
* on each iteration. To ensure it has exclusive access to the variable it
|
||||||
|
* raises it's priority above that of the controller task before each
|
||||||
|
* increment, lowering it again to it's original priority before starting the
|
||||||
|
* next iteration.
|
||||||
|
*
|
||||||
|
* The other counter task increments the shared count variable on each
|
||||||
|
* iteration of it's loop until the count has reached a limit of 0xff - at
|
||||||
|
* which point it suspends itself. It will not start a new loop until the
|
||||||
|
* controller task has made it "ready" again by calling vTaskResume ().
|
||||||
|
* This second counter task operates at a higher priority than controller
|
||||||
|
* task so does not need to worry about mutual exclusion of the counter
|
||||||
|
* variable.
|
||||||
|
*
|
||||||
|
* The controller task is in two sections. The first section controls and
|
||||||
|
* monitors the continuous count task. When this section is operational the
|
||||||
|
* limited count task is suspended. Likewise, the second section controls
|
||||||
|
* and monitors the limited count task. When this section is operational the
|
||||||
|
* continuous count task is suspended.
|
||||||
|
*
|
||||||
|
* In the first section the controller task first takes a copy of the shared
|
||||||
|
* count variable. To ensure mutual exclusion on the count variable it
|
||||||
|
* suspends the continuous count task, resuming it again when the copy has been
|
||||||
|
* taken. The controller task then sleeps for a fixed period - during which
|
||||||
|
* the continuous count task will execute and increment the shared variable.
|
||||||
|
* When the controller task wakes it checks that the continuous count task
|
||||||
|
* has executed by comparing the copy of the shared variable with its current
|
||||||
|
* value. This time, to ensure mutual exclusion, the scheduler itself is
|
||||||
|
* suspended with a call to vTaskSuspendAll (). This is for demonstration
|
||||||
|
* purposes only and is not a recommended technique due to its inefficiency.
|
||||||
|
*
|
||||||
|
* After a fixed number of iterations the controller task suspends the
|
||||||
|
* continuous count task, and moves on to its second section.
|
||||||
|
*
|
||||||
|
* At the start of the second section the shared variable is cleared to zero.
|
||||||
|
* The limited count task is then woken from it's suspension by a call to
|
||||||
|
* vTaskResume (). As this counter task operates at a higher priority than
|
||||||
|
* the controller task the controller task should not run again until the
|
||||||
|
* shared variable has been counted up to the limited value causing the counter
|
||||||
|
* task to suspend itself. The next line after vTaskResume () is therefore
|
||||||
|
* a check on the shared variable to ensure everything is as expected.
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* The second test consists of a couple of very simple tasks that post onto a
|
||||||
|
* queue while the scheduler is suspended. This test was added to test parts
|
||||||
|
* of the scheduler not exercised by the first test.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
#include "semphr.h"
|
||||||
|
|
||||||
|
/* Demo app include files. */
|
||||||
|
#include "dynamic.h"
|
||||||
|
|
||||||
|
/* Function that implements the "limited count" task as described above. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vLimitedIncrementTask, pvParameters );
|
||||||
|
|
||||||
|
/* Function that implements the "continuous count" task as described above. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vContinuousIncrementTask, pvParameters );
|
||||||
|
|
||||||
|
/* Function that implements the controller task as described above. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vCounterControlTask, pvParameters );
|
||||||
|
|
||||||
|
static portTASK_FUNCTION_PROTO( vQueueReceiveWhenSuspendedTask, pvParameters );
|
||||||
|
static portTASK_FUNCTION_PROTO( vQueueSendWhenSuspendedTask, pvParameters );
|
||||||
|
|
||||||
|
/* Demo task specific constants. */
|
||||||
|
#define priSTACK_SIZE ( configMINIMAL_STACK_SIZE )
|
||||||
|
#define priSLEEP_TIME ( ( portTickType ) 128 / portTICK_RATE_MS )
|
||||||
|
#define priLOOPS ( 5 )
|
||||||
|
#define priMAX_COUNT ( ( unsigned long ) 0xff )
|
||||||
|
#define priNO_BLOCK ( ( portTickType ) 0 )
|
||||||
|
#define priSUSPENDED_QUEUE_LENGTH ( 1 )
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* Handles to the two counter tasks. These could be passed in as parameters
|
||||||
|
to the controller task to prevent them having to be file scope. */
|
||||||
|
static xTaskHandle xContinousIncrementHandle, xLimitedIncrementHandle;
|
||||||
|
|
||||||
|
/* The shared counter variable. This is passed in as a parameter to the two
|
||||||
|
counter variables for demonstration purposes. */
|
||||||
|
static unsigned long ulCounter;
|
||||||
|
|
||||||
|
/* Variables used to check that the tasks are still operating without error.
|
||||||
|
Each complete iteration of the controller task increments this variable
|
||||||
|
provided no errors have been found. The variable maintaining the same value
|
||||||
|
is therefore indication of an error. */
|
||||||
|
static volatile unsigned short usCheckVariable = ( unsigned short ) 0;
|
||||||
|
static volatile portBASE_TYPE xSuspendedQueueSendError = pdFALSE;
|
||||||
|
static volatile portBASE_TYPE xSuspendedQueueReceiveError = pdFALSE;
|
||||||
|
|
||||||
|
/* Queue used by the second test. */
|
||||||
|
xQueueHandle xSuspendedTestQueue;
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
/*
|
||||||
|
* Start the three tasks as described at the top of the file.
|
||||||
|
* Note that the limited count task is given a higher priority.
|
||||||
|
*/
|
||||||
|
void vStartDynamicPriorityTasks( void )
|
||||||
|
{
|
||||||
|
xSuspendedTestQueue = xQueueCreate( priSUSPENDED_QUEUE_LENGTH, sizeof( unsigned long ) );
|
||||||
|
|
||||||
|
/* vQueueAddToRegistry() adds the queue to the queue registry, if one is
|
||||||
|
in use. The queue registry is provided as a means for kernel aware
|
||||||
|
debuggers to locate queues and has no purpose if a kernel aware debugger
|
||||||
|
is not being used. The call to vQueueAddToRegistry() will be removed
|
||||||
|
by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is
|
||||||
|
defined to be less than 1. */
|
||||||
|
vQueueAddToRegistry( xSuspendedTestQueue, ( signed char * ) "Suspended_Test_Queue" );
|
||||||
|
|
||||||
|
xTaskCreate( vContinuousIncrementTask, ( signed char * ) "CNT_INC", priSTACK_SIZE, ( void * ) &ulCounter, tskIDLE_PRIORITY, &xContinousIncrementHandle );
|
||||||
|
xTaskCreate( vLimitedIncrementTask, ( signed char * ) "LIM_INC", priSTACK_SIZE, ( void * ) &ulCounter, tskIDLE_PRIORITY + 1, &xLimitedIncrementHandle );
|
||||||
|
xTaskCreate( vCounterControlTask, ( signed char * ) "C_CTRL", priSTACK_SIZE, NULL, tskIDLE_PRIORITY, NULL );
|
||||||
|
xTaskCreate( vQueueSendWhenSuspendedTask, ( signed char * ) "SUSP_TX", priSTACK_SIZE, NULL, tskIDLE_PRIORITY, NULL );
|
||||||
|
xTaskCreate( vQueueReceiveWhenSuspendedTask, ( signed char * ) "SUSP_RX", priSTACK_SIZE, NULL, tskIDLE_PRIORITY, NULL );
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Just loops around incrementing the shared variable until the limit has been
|
||||||
|
* reached. Once the limit has been reached it suspends itself.
|
||||||
|
*/
|
||||||
|
static portTASK_FUNCTION( vLimitedIncrementTask, pvParameters )
|
||||||
|
{
|
||||||
|
unsigned long *pulCounter;
|
||||||
|
|
||||||
|
/* Take a pointer to the shared variable from the parameters passed into
|
||||||
|
the task. */
|
||||||
|
pulCounter = ( unsigned long * ) pvParameters;
|
||||||
|
|
||||||
|
/* This will run before the control task, so the first thing it does is
|
||||||
|
suspend - the control task will resume it when ready. */
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Just count up to a value then suspend. */
|
||||||
|
( *pulCounter )++;
|
||||||
|
|
||||||
|
if( *pulCounter >= priMAX_COUNT )
|
||||||
|
{
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Just keep counting the shared variable up. The control task will suspend
|
||||||
|
* this task when it wants.
|
||||||
|
*/
|
||||||
|
static portTASK_FUNCTION( vContinuousIncrementTask, pvParameters )
|
||||||
|
{
|
||||||
|
unsigned long *pulCounter;
|
||||||
|
unsigned portBASE_TYPE uxOurPriority;
|
||||||
|
|
||||||
|
/* Take a pointer to the shared variable from the parameters passed into
|
||||||
|
the task. */
|
||||||
|
pulCounter = ( unsigned long * ) pvParameters;
|
||||||
|
|
||||||
|
/* Query our priority so we can raise it when exclusive access to the
|
||||||
|
shared variable is required. */
|
||||||
|
uxOurPriority = uxTaskPriorityGet( NULL );
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Raise our priority above the controller task to ensure a context
|
||||||
|
switch does not occur while we are accessing this variable. */
|
||||||
|
vTaskPrioritySet( NULL, uxOurPriority + 1 );
|
||||||
|
( *pulCounter )++;
|
||||||
|
vTaskPrioritySet( NULL, uxOurPriority );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Controller task as described above.
|
||||||
|
*/
|
||||||
|
static portTASK_FUNCTION( vCounterControlTask, pvParameters )
|
||||||
|
{
|
||||||
|
unsigned long ulLastCounter;
|
||||||
|
short sLoops;
|
||||||
|
short sError = pdFALSE;
|
||||||
|
|
||||||
|
/* Just to stop warning messages. */
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Start with the counter at zero. */
|
||||||
|
ulCounter = ( unsigned long ) 0;
|
||||||
|
|
||||||
|
/* First section : */
|
||||||
|
|
||||||
|
/* Check the continuous count task is running. */
|
||||||
|
for( sLoops = 0; sLoops < priLOOPS; sLoops++ )
|
||||||
|
{
|
||||||
|
/* Suspend the continuous count task so we can take a mirror of the
|
||||||
|
shared variable without risk of corruption. */
|
||||||
|
vTaskSuspend( xContinousIncrementHandle );
|
||||||
|
ulLastCounter = ulCounter;
|
||||||
|
vTaskResume( xContinousIncrementHandle );
|
||||||
|
|
||||||
|
/* Now delay to ensure the other task has processor time. */
|
||||||
|
vTaskDelay( priSLEEP_TIME );
|
||||||
|
|
||||||
|
/* Check the shared variable again. This time to ensure mutual
|
||||||
|
exclusion the whole scheduler will be locked. This is just for
|
||||||
|
demo purposes! */
|
||||||
|
vTaskSuspendAll();
|
||||||
|
{
|
||||||
|
if( ulLastCounter == ulCounter )
|
||||||
|
{
|
||||||
|
/* The shared variable has not changed. There is a problem
|
||||||
|
with the continuous count task so flag an error. */
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
xTaskResumeAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Second section: */
|
||||||
|
|
||||||
|
/* Suspend the continuous counter task so it stops accessing the shared variable. */
|
||||||
|
vTaskSuspend( xContinousIncrementHandle );
|
||||||
|
|
||||||
|
/* Reset the variable. */
|
||||||
|
ulCounter = ( unsigned long ) 0;
|
||||||
|
|
||||||
|
/* Resume the limited count task which has a higher priority than us.
|
||||||
|
We should therefore not return from this call until the limited count
|
||||||
|
task has suspended itself with a known value in the counter variable. */
|
||||||
|
vTaskResume( xLimitedIncrementHandle );
|
||||||
|
|
||||||
|
/* Does the counter variable have the expected value? */
|
||||||
|
if( ulCounter != priMAX_COUNT )
|
||||||
|
{
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( sError == pdFALSE )
|
||||||
|
{
|
||||||
|
/* If no errors have occurred then increment the check variable. */
|
||||||
|
portENTER_CRITICAL();
|
||||||
|
usCheckVariable++;
|
||||||
|
portEXIT_CRITICAL();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Resume the continuous count task and do it all again. */
|
||||||
|
vTaskResume( xContinousIncrementHandle );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vQueueSendWhenSuspendedTask, pvParameters )
|
||||||
|
{
|
||||||
|
static unsigned long ulValueToSend = ( unsigned long ) 0;
|
||||||
|
|
||||||
|
/* Just to stop warning messages. */
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
vTaskSuspendAll();
|
||||||
|
{
|
||||||
|
/* We must not block while the scheduler is suspended! */
|
||||||
|
if( xQueueSend( xSuspendedTestQueue, ( void * ) &ulValueToSend, priNO_BLOCK ) != pdTRUE )
|
||||||
|
{
|
||||||
|
xSuspendedQueueSendError = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
xTaskResumeAll();
|
||||||
|
|
||||||
|
vTaskDelay( priSLEEP_TIME );
|
||||||
|
|
||||||
|
++ulValueToSend;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vQueueReceiveWhenSuspendedTask, pvParameters )
|
||||||
|
{
|
||||||
|
static unsigned long ulExpectedValue = ( unsigned long ) 0, ulReceivedValue;
|
||||||
|
portBASE_TYPE xGotValue;
|
||||||
|
|
||||||
|
/* Just to stop warning messages. */
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
/* Suspending the scheduler here is fairly pointless and
|
||||||
|
undesirable for a normal application. It is done here purely
|
||||||
|
to test the scheduler. The inner xTaskResumeAll() should
|
||||||
|
never return pdTRUE as the scheduler is still locked by the
|
||||||
|
outer call. */
|
||||||
|
vTaskSuspendAll();
|
||||||
|
{
|
||||||
|
vTaskSuspendAll();
|
||||||
|
{
|
||||||
|
xGotValue = xQueueReceive( xSuspendedTestQueue, ( void * ) &ulReceivedValue, priNO_BLOCK );
|
||||||
|
}
|
||||||
|
if( xTaskResumeAll() )
|
||||||
|
{
|
||||||
|
xSuspendedQueueReceiveError = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
xTaskResumeAll();
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
{
|
||||||
|
taskYIELD();
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
} while( xGotValue == pdFALSE );
|
||||||
|
|
||||||
|
if( ulReceivedValue != ulExpectedValue )
|
||||||
|
{
|
||||||
|
xSuspendedQueueReceiveError = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
++ulExpectedValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* Called to check that all the created tasks are still running without error. */
|
||||||
|
portBASE_TYPE xAreDynamicPriorityTasksStillRunning( void )
|
||||||
|
{
|
||||||
|
/* Keep a history of the check variables so we know if it has been incremented
|
||||||
|
since the last call. */
|
||||||
|
static unsigned short usLastTaskCheck = ( unsigned short ) 0;
|
||||||
|
portBASE_TYPE xReturn = pdTRUE;
|
||||||
|
|
||||||
|
/* Check the tasks are still running by ensuring the check variable
|
||||||
|
is still incrementing. */
|
||||||
|
|
||||||
|
if( usCheckVariable == usLastTaskCheck )
|
||||||
|
{
|
||||||
|
/* The check has not incremented so an error exists. */
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xSuspendedQueueSendError == pdTRUE )
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xSuspendedQueueReceiveError == pdTRUE )
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
usLastTaskCheck = usCheckVariable;
|
||||||
|
return xReturn;
|
||||||
|
}
|
|
@ -0,0 +1,145 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This version of flash .c is for use on systems that have limited stack space
|
||||||
|
* and no display facilities. The complete version can be found in the
|
||||||
|
* Demo/Common/Full directory.
|
||||||
|
*
|
||||||
|
* Three tasks are created, each of which flash an LED at a different rate. The first
|
||||||
|
* LED flashes every 200ms, the second every 400ms, the third every 600ms.
|
||||||
|
*
|
||||||
|
* The LED flash tasks provide instant visual feedback. They show that the scheduler
|
||||||
|
* is still operational.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
|
||||||
|
/* Demo program include files. */
|
||||||
|
#include "partest.h"
|
||||||
|
#include "flash.h"
|
||||||
|
|
||||||
|
#define ledSTACK_SIZE configMINIMAL_STACK_SIZE
|
||||||
|
#define ledNUMBER_OF_LEDS ( 3 )
|
||||||
|
#define ledFLASH_RATE_BASE ( ( portTickType ) 333 )
|
||||||
|
|
||||||
|
/* Variable used by the created tasks to calculate the LED number to use, and
|
||||||
|
the rate at which they should flash the LED. */
|
||||||
|
static volatile unsigned portBASE_TYPE uxFlashTaskNumber = 0;
|
||||||
|
|
||||||
|
/* The task that is created three times. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vLEDFlashTask, pvParameters );
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vStartLEDFlashTasks( unsigned portBASE_TYPE uxPriority )
|
||||||
|
{
|
||||||
|
signed portBASE_TYPE xLEDTask;
|
||||||
|
|
||||||
|
/* Create the three tasks. */
|
||||||
|
for( xLEDTask = 0; xLEDTask < ledNUMBER_OF_LEDS; ++xLEDTask )
|
||||||
|
{
|
||||||
|
/* Spawn the task. */
|
||||||
|
xTaskCreate( vLEDFlashTask, ( signed char * ) "LEDx", ledSTACK_SIZE, NULL, uxPriority, ( xTaskHandle * ) NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vLEDFlashTask, pvParameters )
|
||||||
|
{
|
||||||
|
portTickType xFlashRate, xLastFlashTime;
|
||||||
|
unsigned portBASE_TYPE uxLED;
|
||||||
|
|
||||||
|
/* The parameters are not used. */
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
/* Calculate the LED and flash rate. */
|
||||||
|
portENTER_CRITICAL();
|
||||||
|
{
|
||||||
|
/* See which of the eight LED's we should use. */
|
||||||
|
uxLED = uxFlashTaskNumber;
|
||||||
|
|
||||||
|
/* Update so the next task uses the next LED. */
|
||||||
|
uxFlashTaskNumber++;
|
||||||
|
}
|
||||||
|
portEXIT_CRITICAL();
|
||||||
|
|
||||||
|
xFlashRate = ledFLASH_RATE_BASE + ( ledFLASH_RATE_BASE * ( portTickType ) uxLED );
|
||||||
|
xFlashRate /= portTICK_RATE_MS;
|
||||||
|
|
||||||
|
/* We will turn the LED on and off again in the delay period, so each
|
||||||
|
delay is only half the total period. */
|
||||||
|
xFlashRate /= ( portTickType ) 2;
|
||||||
|
|
||||||
|
/* We need to initialise xLastFlashTime prior to the first call to
|
||||||
|
vTaskDelayUntil(). */
|
||||||
|
xLastFlashTime = xTaskGetTickCount();
|
||||||
|
|
||||||
|
for(;;)
|
||||||
|
{
|
||||||
|
/* Delay for half the flash period then turn the LED on. */
|
||||||
|
vTaskDelayUntil( &xLastFlashTime, xFlashRate );
|
||||||
|
vParTestToggleLED( uxLED );
|
||||||
|
|
||||||
|
/* Delay for half the flash period then turn the LED off. */
|
||||||
|
vTaskDelayUntil( &xLastFlashTime, xFlashRate );
|
||||||
|
vParTestToggleLED( uxLED );
|
||||||
|
}
|
||||||
|
} /*lint !e715 !e818 !e830 Function definition must be standard for task creation. */
|
||||||
|
|
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BLOCK_Q_H
|
||||||
|
#define BLOCK_Q_H
|
||||||
|
|
||||||
|
void vStartBlockingQueueTasks( unsigned portBASE_TYPE uxPriority );
|
||||||
|
portBASE_TYPE xAreBlockingQueuesStillRunning( void );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,63 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GEN_Q_TEST_H
|
||||||
|
#define GEN_Q_TEST_H
|
||||||
|
|
||||||
|
void vStartGenericQueueTasks( unsigned portBASE_TYPE uxPriority );
|
||||||
|
portBASE_TYPE xAreGenericQueueTasksStillRunning( void );
|
||||||
|
|
||||||
|
#endif /* GEN_Q_TEST_H */
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef POLLED_Q_H
|
||||||
|
#define POLLED_Q_H
|
||||||
|
|
||||||
|
void vStartPolledQueueTasks( unsigned portBASE_TYPE uxPriority );
|
||||||
|
portBASE_TYPE xArePollingQueuesStillRunning( void );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,63 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef Q_PEEK_TEST_H
|
||||||
|
#define Q_PEEK_TEST_H
|
||||||
|
|
||||||
|
void vStartQueuePeekTasks( void );
|
||||||
|
portBASE_TYPE xAreQueuePeekTasksStillRunning( void );
|
||||||
|
|
||||||
|
#endif /* Q_PEEK_TEST_H */
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BLOCK_TIME_TEST_H
|
||||||
|
#define BLOCK_TIME_TEST_H
|
||||||
|
|
||||||
|
void vCreateBlockTimeTasks( void );
|
||||||
|
portBASE_TYPE xAreBlockTimeTestTasksStillRunning( void );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef SUICIDE_TASK_H
|
||||||
|
#define SUICIDE_TASK_H
|
||||||
|
|
||||||
|
void vCreateSuicidalTasks( unsigned portBASE_TYPE uxPriority );
|
||||||
|
portBASE_TYPE xIsCreateTaskStillRunning( void );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef DYNAMIC_MANIPULATION_H
|
||||||
|
#define DYNAMIC_MANIPULATION_H
|
||||||
|
|
||||||
|
void vStartDynamicPriorityTasks( void );
|
||||||
|
portBASE_TYPE xAreDynamicPriorityTasksStillRunning( void );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,60 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef FLASH_LED_H
|
||||||
|
#define FLASH_LED_H
|
||||||
|
|
||||||
|
void vStartLEDFlashTasks( unsigned portBASE_TYPE uxPriority );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef FLOP_TASKS_H
|
||||||
|
#define FLOP_TASKS_H
|
||||||
|
|
||||||
|
void vStartMathTasks( unsigned portBASE_TYPE uxPriority );
|
||||||
|
portBASE_TYPE xAreMathsTaskStillRunning( void );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,64 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PARTEST_H
|
||||||
|
#define PARTEST_H
|
||||||
|
|
||||||
|
#define partstDEFAULT_PORT_ADDRESS ( ( unsigned short ) 0x378 )
|
||||||
|
|
||||||
|
void vParTestInitialise( void );
|
||||||
|
void vParTestSetLED( unsigned portBASE_TYPE uxLED, signed portBASE_TYPE xValue );
|
||||||
|
void vParTestToggleLED( unsigned portBASE_TYPE uxLED );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -0,0 +1,61 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef RECURSIVE_MUTEX_TEST_H
|
||||||
|
#define RECURSIVE_MUTEX_TEST_H
|
||||||
|
|
||||||
|
void vStartRecursiveMutexTasks( void );
|
||||||
|
portBASE_TYPE xAreRecursiveMutexTasksStillRunning( void );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -0,0 +1,61 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef SEMAPHORE_TEST_H
|
||||||
|
#define SEMAPHORE_TEST_H
|
||||||
|
|
||||||
|
void vStartSemaphoreTasks( unsigned portBASE_TYPE uxPriority );
|
||||||
|
portBASE_TYPE xAreSemaphoreTasksStillRunning( void );
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -0,0 +1,395 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
The tasks defined on this page demonstrate the use of recursive mutexes.
|
||||||
|
|
||||||
|
For recursive mutex functionality the created mutex should be created using
|
||||||
|
xSemaphoreCreateRecursiveMutex(), then be manipulated
|
||||||
|
using the xSemaphoreTakeRecursive() and xSemaphoreGiveRecursive() API
|
||||||
|
functions.
|
||||||
|
|
||||||
|
This demo creates three tasks all of which access the same recursive mutex:
|
||||||
|
|
||||||
|
prvRecursiveMutexControllingTask() has the highest priority so executes
|
||||||
|
first and grabs the mutex. It then performs some recursive accesses -
|
||||||
|
between each of which it sleeps for a short period to let the lower
|
||||||
|
priority tasks execute. When it has completed its demo functionality
|
||||||
|
it gives the mutex back before suspending itself.
|
||||||
|
|
||||||
|
prvRecursiveMutexBlockingTask() attempts to access the mutex by performing
|
||||||
|
a blocking 'take'. The blocking task has a lower priority than the
|
||||||
|
controlling task so by the time it executes the mutex has already been
|
||||||
|
taken by the controlling task, causing the blocking task to block. It
|
||||||
|
does not unblock until the controlling task has given the mutex back,
|
||||||
|
and it does not actually run until the controlling task has suspended
|
||||||
|
itself (due to the relative priorities). When it eventually does obtain
|
||||||
|
the mutex all it does is give the mutex back prior to also suspending
|
||||||
|
itself. At this point both the controlling task and the blocking task are
|
||||||
|
suspended.
|
||||||
|
|
||||||
|
prvRecursiveMutexPollingTask() runs at the idle priority. It spins round
|
||||||
|
a tight loop attempting to obtain the mutex with a non-blocking call. As
|
||||||
|
the lowest priority task it will not successfully obtain the mutex until
|
||||||
|
both the controlling and blocking tasks are suspended. Once it eventually
|
||||||
|
does obtain the mutex it first unsuspends both the controlling task and
|
||||||
|
blocking task prior to giving the mutex back - resulting in the polling
|
||||||
|
task temporarily inheriting the controlling tasks priority.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
#include "semphr.h"
|
||||||
|
|
||||||
|
/* Demo app include files. */
|
||||||
|
#include "recmutex.h"
|
||||||
|
|
||||||
|
/* Priorities assigned to the three tasks. */
|
||||||
|
#define recmuCONTROLLING_TASK_PRIORITY ( tskIDLE_PRIORITY + 2 )
|
||||||
|
#define recmuBLOCKING_TASK_PRIORITY ( tskIDLE_PRIORITY + 1 )
|
||||||
|
#define recmuPOLLING_TASK_PRIORITY ( tskIDLE_PRIORITY + 0 )
|
||||||
|
|
||||||
|
/* The recursive call depth. */
|
||||||
|
#define recmuMAX_COUNT ( 10 )
|
||||||
|
|
||||||
|
/* Misc. */
|
||||||
|
#define recmuSHORT_DELAY ( 20 / portTICK_RATE_MS )
|
||||||
|
#define recmuNO_DELAY ( ( portTickType ) 0 )
|
||||||
|
#define recmuTWO_TICK_DELAY ( ( portTickType ) 2 )
|
||||||
|
|
||||||
|
/* The three tasks as described at the top of this file. */
|
||||||
|
static void prvRecursiveMutexControllingTask( void *pvParameters );
|
||||||
|
static void prvRecursiveMutexBlockingTask( void *pvParameters );
|
||||||
|
static void prvRecursiveMutexPollingTask( void *pvParameters );
|
||||||
|
|
||||||
|
/* The mutex used by the demo. */
|
||||||
|
static xSemaphoreHandle xMutex;
|
||||||
|
|
||||||
|
/* Variables used to detect and latch errors. */
|
||||||
|
static volatile portBASE_TYPE xErrorOccurred = pdFALSE, xControllingIsSuspended = pdFALSE, xBlockingIsSuspended = pdFALSE;
|
||||||
|
static volatile unsigned portBASE_TYPE uxControllingCycles = 0, uxBlockingCycles = 0, uxPollingCycles = 0;
|
||||||
|
|
||||||
|
/* Handles of the two higher priority tasks, required so they can be resumed
|
||||||
|
(unsuspended). */
|
||||||
|
static xTaskHandle xControllingTaskHandle, xBlockingTaskHandle;
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vStartRecursiveMutexTasks( void )
|
||||||
|
{
|
||||||
|
/* Just creates the mutex and the three tasks. */
|
||||||
|
|
||||||
|
xMutex = xSemaphoreCreateRecursiveMutex();
|
||||||
|
|
||||||
|
/* vQueueAddToRegistry() adds the mutex to the registry, if one is
|
||||||
|
in use. The registry is provided as a means for kernel aware
|
||||||
|
debuggers to locate mutex and has no purpose if a kernel aware debugger
|
||||||
|
is not being used. The call to vQueueAddToRegistry() will be removed
|
||||||
|
by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is
|
||||||
|
defined to be less than 1. */
|
||||||
|
vQueueAddToRegistry( ( xQueueHandle ) xMutex, ( signed portCHAR * ) "Recursive_Mutex" );
|
||||||
|
|
||||||
|
|
||||||
|
if( xMutex != NULL )
|
||||||
|
{
|
||||||
|
xTaskCreate( prvRecursiveMutexControllingTask, ( signed portCHAR * ) "Rec1", configMINIMAL_STACK_SIZE, NULL, recmuCONTROLLING_TASK_PRIORITY, &xControllingTaskHandle );
|
||||||
|
xTaskCreate( prvRecursiveMutexBlockingTask, ( signed portCHAR * ) "Rec2", configMINIMAL_STACK_SIZE, NULL, recmuBLOCKING_TASK_PRIORITY, &xBlockingTaskHandle );
|
||||||
|
xTaskCreate( prvRecursiveMutexPollingTask, ( signed portCHAR * ) "Rec3", configMINIMAL_STACK_SIZE, NULL, recmuPOLLING_TASK_PRIORITY, NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvRecursiveMutexControllingTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
unsigned portBASE_TYPE ux;
|
||||||
|
|
||||||
|
/* Just to remove compiler warning. */
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Should not be able to 'give' the mutex, as we have not yet 'taken'
|
||||||
|
it. The first time through, the mutex will not have been used yet,
|
||||||
|
subsequent times through, at this point the mutex will be held by the
|
||||||
|
polling task. */
|
||||||
|
if( xSemaphoreGiveRecursive( xMutex ) == pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
for( ux = 0; ux < recmuMAX_COUNT; ux++ )
|
||||||
|
{
|
||||||
|
/* We should now be able to take the mutex as many times as
|
||||||
|
we like.
|
||||||
|
|
||||||
|
The first time through the mutex will be immediately available, on
|
||||||
|
subsequent times through the mutex will be held by the polling task
|
||||||
|
at this point and this Take will cause the polling task to inherit
|
||||||
|
the priority of this task. In this case the block time must be
|
||||||
|
long enough to ensure the polling task will execute again before the
|
||||||
|
block time expires. If the block time does expire then the error
|
||||||
|
flag will be set here. */
|
||||||
|
if( xSemaphoreTakeRecursive( xMutex, recmuTWO_TICK_DELAY ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ensure the other task attempting to access the mutex (and the
|
||||||
|
other demo tasks) are able to execute to ensure they either block
|
||||||
|
(where a block time is specified) or return an error (where no
|
||||||
|
block time is specified) as the mutex is held by this task. */
|
||||||
|
vTaskDelay( recmuSHORT_DELAY );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* For each time we took the mutex, give it back. */
|
||||||
|
for( ux = 0; ux < recmuMAX_COUNT; ux++ )
|
||||||
|
{
|
||||||
|
/* Ensure the other task attempting to access the mutex (and the
|
||||||
|
other demo tasks) are able to execute. */
|
||||||
|
vTaskDelay( recmuSHORT_DELAY );
|
||||||
|
|
||||||
|
/* We should now be able to give the mutex as many times as we
|
||||||
|
took it. When the mutex is available again the Blocking task
|
||||||
|
should be unblocked but not run because it has a lower priority
|
||||||
|
than this task. The polling task should also not run at this point
|
||||||
|
as it too has a lower priority than this task. */
|
||||||
|
if( xSemaphoreGiveRecursive( xMutex ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Having given it back the same number of times as it was taken, we
|
||||||
|
should no longer be the mutex owner, so the next give sh ould fail. */
|
||||||
|
if( xSemaphoreGiveRecursive( xMutex ) == pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Keep count of the number of cycles this task has performed so a
|
||||||
|
stall can be detected. */
|
||||||
|
uxControllingCycles++;
|
||||||
|
|
||||||
|
/* Suspend ourselves to the blocking task can execute. */
|
||||||
|
xControllingIsSuspended = pdTRUE;
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
xControllingIsSuspended = pdFALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvRecursiveMutexBlockingTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
/* Just to remove compiler warning. */
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* This task will run while the controlling task is blocked, and the
|
||||||
|
controlling task will block only once it has the mutex - therefore
|
||||||
|
this call should block until the controlling task has given up the
|
||||||
|
mutex, and not actually execute past this call until the controlling
|
||||||
|
task is suspended. */
|
||||||
|
if( xSemaphoreTakeRecursive( xMutex, portMAX_DELAY ) == pdPASS )
|
||||||
|
{
|
||||||
|
if( xControllingIsSuspended != pdTRUE )
|
||||||
|
{
|
||||||
|
/* Did not expect to execute until the controlling task was
|
||||||
|
suspended. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Give the mutex back before suspending ourselves to allow
|
||||||
|
the polling task to obtain the mutex. */
|
||||||
|
if( xSemaphoreGiveRecursive( xMutex ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
xBlockingIsSuspended = pdTRUE;
|
||||||
|
vTaskSuspend( NULL );
|
||||||
|
xBlockingIsSuspended = pdFALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* We should not leave the xSemaphoreTakeRecursive() function
|
||||||
|
until the mutex was obtained. */
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The controlling and blocking tasks should be in lock step. */
|
||||||
|
if( uxControllingCycles != ( uxBlockingCycles + 1 ) )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Keep count of the number of cycles this task has performed so a
|
||||||
|
stall can be detected. */
|
||||||
|
uxBlockingCycles++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static void prvRecursiveMutexPollingTask( void *pvParameters )
|
||||||
|
{
|
||||||
|
/* Just to remove compiler warning. */
|
||||||
|
( void ) pvParameters;
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Keep attempting to obtain the mutex. We should only obtain it when
|
||||||
|
the blocking task has suspended itself, which in turn should only
|
||||||
|
happen when the controlling task is also suspended. */
|
||||||
|
if( xSemaphoreTakeRecursive( xMutex, recmuNO_DELAY ) == pdPASS )
|
||||||
|
{
|
||||||
|
/* Is the blocking task suspended? */
|
||||||
|
if( ( xBlockingIsSuspended != pdTRUE ) || ( xControllingIsSuspended != pdTRUE ) )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Keep count of the number of cycles this task has performed
|
||||||
|
so a stall can be detected. */
|
||||||
|
uxPollingCycles++;
|
||||||
|
|
||||||
|
/* We can resume the other tasks here even though they have a
|
||||||
|
higher priority than the polling task. When they execute they
|
||||||
|
will attempt to obtain the mutex but fail because the polling
|
||||||
|
task is still the mutex holder. The polling task (this task)
|
||||||
|
will then inherit the higher priority. The Blocking task will
|
||||||
|
block indefinitely when it attempts to obtain the mutex, the
|
||||||
|
Controlling task will only block for a fixed period and an
|
||||||
|
error will be latched if the polling task has not returned the
|
||||||
|
mutex by the time this fixed period has expired. */
|
||||||
|
vTaskResume( xBlockingTaskHandle );
|
||||||
|
vTaskResume( xControllingTaskHandle );
|
||||||
|
|
||||||
|
/* The other two tasks should now have executed and no longer
|
||||||
|
be suspended. */
|
||||||
|
if( ( xBlockingIsSuspended == pdTRUE ) || ( xControllingIsSuspended == pdTRUE ) )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Release the mutex, disinheriting the higher priority again. */
|
||||||
|
if( xSemaphoreGiveRecursive( xMutex ) != pdPASS )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
{
|
||||||
|
taskYIELD();
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* This is called to check that all the created tasks are still running. */
|
||||||
|
portBASE_TYPE xAreRecursiveMutexTasksStillRunning( void )
|
||||||
|
{
|
||||||
|
portBASE_TYPE xReturn;
|
||||||
|
static unsigned portBASE_TYPE uxLastControllingCycles = 0, uxLastBlockingCycles = 0, uxLastPollingCycles = 0;
|
||||||
|
|
||||||
|
/* Is the controlling task still cycling? */
|
||||||
|
if( uxLastControllingCycles == uxControllingCycles )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
uxLastControllingCycles = uxControllingCycles;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Is the blocking task still cycling? */
|
||||||
|
if( uxLastBlockingCycles == uxBlockingCycles )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
uxLastBlockingCycles = uxBlockingCycles;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Is the polling task still cycling? */
|
||||||
|
if( uxLastPollingCycles == uxPollingCycles )
|
||||||
|
{
|
||||||
|
xErrorOccurred = pdTRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
uxLastPollingCycles = uxPollingCycles;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( xErrorOccurred == pdTRUE )
|
||||||
|
{
|
||||||
|
xReturn = pdFAIL;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
xReturn = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return xReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,284 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Creates two sets of two tasks. The tasks within a set share a variable, access
|
||||||
|
* to which is guarded by a semaphore.
|
||||||
|
*
|
||||||
|
* Each task starts by attempting to obtain the semaphore. On obtaining a
|
||||||
|
* semaphore a task checks to ensure that the guarded variable has an expected
|
||||||
|
* value. It then clears the variable to zero before counting it back up to the
|
||||||
|
* expected value in increments of 1. After each increment the variable is checked
|
||||||
|
* to ensure it contains the value to which it was just set. When the starting
|
||||||
|
* value is again reached the task releases the semaphore giving the other task in
|
||||||
|
* the set a chance to do exactly the same thing. The starting value is high
|
||||||
|
* enough to ensure that a tick is likely to occur during the incrementing loop.
|
||||||
|
*
|
||||||
|
* An error is flagged if at any time during the process a shared variable is
|
||||||
|
* found to have a value other than that expected. Such an occurrence would
|
||||||
|
* suggest an error in the mutual exclusion mechanism by which access to the
|
||||||
|
* variable is restricted.
|
||||||
|
*
|
||||||
|
* The first set of two tasks poll their semaphore. The second set use blocking
|
||||||
|
* calls.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
#include "semphr.h"
|
||||||
|
|
||||||
|
/* Demo app include files. */
|
||||||
|
#include "semtest.h"
|
||||||
|
|
||||||
|
/* The value to which the shared variables are counted. */
|
||||||
|
#define semtstBLOCKING_EXPECTED_VALUE ( ( unsigned long ) 0xfff )
|
||||||
|
#define semtstNON_BLOCKING_EXPECTED_VALUE ( ( unsigned long ) 0xff )
|
||||||
|
|
||||||
|
#define semtstSTACK_SIZE configMINIMAL_STACK_SIZE
|
||||||
|
|
||||||
|
#define semtstNUM_TASKS ( 4 )
|
||||||
|
|
||||||
|
#define semtstDELAY_FACTOR ( ( portTickType ) 10 )
|
||||||
|
|
||||||
|
/* The task function as described at the top of the file. */
|
||||||
|
static portTASK_FUNCTION_PROTO( prvSemaphoreTest, pvParameters );
|
||||||
|
|
||||||
|
/* Structure used to pass parameters to each task. */
|
||||||
|
typedef struct SEMAPHORE_PARAMETERS
|
||||||
|
{
|
||||||
|
xSemaphoreHandle xSemaphore;
|
||||||
|
volatile unsigned long *pulSharedVariable;
|
||||||
|
portTickType xBlockTime;
|
||||||
|
} xSemaphoreParameters;
|
||||||
|
|
||||||
|
/* Variables used to check that all the tasks are still running without errors. */
|
||||||
|
static volatile short sCheckVariables[ semtstNUM_TASKS ] = { 0 };
|
||||||
|
static volatile short sNextCheckVariable = 0;
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vStartSemaphoreTasks( unsigned portBASE_TYPE uxPriority )
|
||||||
|
{
|
||||||
|
xSemaphoreParameters *pxFirstSemaphoreParameters, *pxSecondSemaphoreParameters;
|
||||||
|
const portTickType xBlockTime = ( portTickType ) 100;
|
||||||
|
|
||||||
|
/* Create the structure used to pass parameters to the first two tasks. */
|
||||||
|
pxFirstSemaphoreParameters = ( xSemaphoreParameters * ) pvPortMalloc( sizeof( xSemaphoreParameters ) );
|
||||||
|
|
||||||
|
if( pxFirstSemaphoreParameters != NULL )
|
||||||
|
{
|
||||||
|
/* Create the semaphore used by the first two tasks. */
|
||||||
|
vSemaphoreCreateBinary( pxFirstSemaphoreParameters->xSemaphore );
|
||||||
|
|
||||||
|
if( pxFirstSemaphoreParameters->xSemaphore != NULL )
|
||||||
|
{
|
||||||
|
/* Create the variable which is to be shared by the first two tasks. */
|
||||||
|
pxFirstSemaphoreParameters->pulSharedVariable = ( unsigned long * ) pvPortMalloc( sizeof( unsigned long ) );
|
||||||
|
|
||||||
|
/* Initialise the share variable to the value the tasks expect. */
|
||||||
|
*( pxFirstSemaphoreParameters->pulSharedVariable ) = semtstNON_BLOCKING_EXPECTED_VALUE;
|
||||||
|
|
||||||
|
/* The first two tasks do not block on semaphore calls. */
|
||||||
|
pxFirstSemaphoreParameters->xBlockTime = ( portTickType ) 0;
|
||||||
|
|
||||||
|
/* Spawn the first two tasks. As they poll they operate at the idle priority. */
|
||||||
|
xTaskCreate( prvSemaphoreTest, ( signed char * ) "PolSEM1", semtstSTACK_SIZE, ( void * ) pxFirstSemaphoreParameters, tskIDLE_PRIORITY, ( xTaskHandle * ) NULL );
|
||||||
|
xTaskCreate( prvSemaphoreTest, ( signed char * ) "PolSEM2", semtstSTACK_SIZE, ( void * ) pxFirstSemaphoreParameters, tskIDLE_PRIORITY, ( xTaskHandle * ) NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Do exactly the same to create the second set of tasks, only this time
|
||||||
|
provide a block time for the semaphore calls. */
|
||||||
|
pxSecondSemaphoreParameters = ( xSemaphoreParameters * ) pvPortMalloc( sizeof( xSemaphoreParameters ) );
|
||||||
|
if( pxSecondSemaphoreParameters != NULL )
|
||||||
|
{
|
||||||
|
vSemaphoreCreateBinary( pxSecondSemaphoreParameters->xSemaphore );
|
||||||
|
|
||||||
|
if( pxSecondSemaphoreParameters->xSemaphore != NULL )
|
||||||
|
{
|
||||||
|
pxSecondSemaphoreParameters->pulSharedVariable = ( unsigned long * ) pvPortMalloc( sizeof( unsigned long ) );
|
||||||
|
*( pxSecondSemaphoreParameters->pulSharedVariable ) = semtstBLOCKING_EXPECTED_VALUE;
|
||||||
|
pxSecondSemaphoreParameters->xBlockTime = xBlockTime / portTICK_RATE_MS;
|
||||||
|
|
||||||
|
xTaskCreate( prvSemaphoreTest, ( signed char * ) "BlkSEM1", semtstSTACK_SIZE, ( void * ) pxSecondSemaphoreParameters, uxPriority, ( xTaskHandle * ) NULL );
|
||||||
|
xTaskCreate( prvSemaphoreTest, ( signed char * ) "BlkSEM2", semtstSTACK_SIZE, ( void * ) pxSecondSemaphoreParameters, uxPriority, ( xTaskHandle * ) NULL );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* vQueueAddToRegistry() adds the semaphore to the registry, if one is
|
||||||
|
in use. The registry is provided as a means for kernel aware
|
||||||
|
debuggers to locate semaphores and has no purpose if a kernel aware debugger
|
||||||
|
is not being used. The call to vQueueAddToRegistry() will be removed
|
||||||
|
by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is
|
||||||
|
defined to be less than 1. */
|
||||||
|
vQueueAddToRegistry( ( xQueueHandle ) pxFirstSemaphoreParameters->xSemaphore, ( signed char * ) "Counting_Sem_1" );
|
||||||
|
vQueueAddToRegistry( ( xQueueHandle ) pxSecondSemaphoreParameters->xSemaphore, ( signed char * ) "Counting_Sem_2" );
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( prvSemaphoreTest, pvParameters )
|
||||||
|
{
|
||||||
|
xSemaphoreParameters *pxParameters;
|
||||||
|
volatile unsigned long *pulSharedVariable, ulExpectedValue;
|
||||||
|
unsigned long ulCounter;
|
||||||
|
short sError = pdFALSE, sCheckVariableToUse;
|
||||||
|
|
||||||
|
/* See which check variable to use. sNextCheckVariable is not semaphore
|
||||||
|
protected! */
|
||||||
|
portENTER_CRITICAL();
|
||||||
|
sCheckVariableToUse = sNextCheckVariable;
|
||||||
|
sNextCheckVariable++;
|
||||||
|
portEXIT_CRITICAL();
|
||||||
|
|
||||||
|
/* A structure is passed in as the parameter. This contains the shared
|
||||||
|
variable being guarded. */
|
||||||
|
pxParameters = ( xSemaphoreParameters * ) pvParameters;
|
||||||
|
pulSharedVariable = pxParameters->pulSharedVariable;
|
||||||
|
|
||||||
|
/* If we are blocking we use a much higher count to ensure loads of context
|
||||||
|
switches occur during the count. */
|
||||||
|
if( pxParameters->xBlockTime > ( portTickType ) 0 )
|
||||||
|
{
|
||||||
|
ulExpectedValue = semtstBLOCKING_EXPECTED_VALUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ulExpectedValue = semtstNON_BLOCKING_EXPECTED_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
/* Try to obtain the semaphore. */
|
||||||
|
if( xSemaphoreTake( pxParameters->xSemaphore, pxParameters->xBlockTime ) == pdPASS )
|
||||||
|
{
|
||||||
|
/* We have the semaphore and so expect any other tasks using the
|
||||||
|
shared variable to have left it in the state we expect to find
|
||||||
|
it. */
|
||||||
|
if( *pulSharedVariable != ulExpectedValue )
|
||||||
|
{
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Clear the variable, then count it back up to the expected value
|
||||||
|
before releasing the semaphore. Would expect a context switch or
|
||||||
|
two during this time. */
|
||||||
|
for( ulCounter = ( unsigned long ) 0; ulCounter <= ulExpectedValue; ulCounter++ )
|
||||||
|
{
|
||||||
|
*pulSharedVariable = ulCounter;
|
||||||
|
if( *pulSharedVariable != ulCounter )
|
||||||
|
{
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Release the semaphore, and if no errors have occurred increment the check
|
||||||
|
variable. */
|
||||||
|
if( xSemaphoreGive( pxParameters->xSemaphore ) == pdFALSE )
|
||||||
|
{
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( sError == pdFALSE )
|
||||||
|
{
|
||||||
|
if( sCheckVariableToUse < semtstNUM_TASKS )
|
||||||
|
{
|
||||||
|
( sCheckVariables[ sCheckVariableToUse ] )++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If we have a block time then we are running at a priority higher
|
||||||
|
than the idle priority. This task takes a long time to complete
|
||||||
|
a cycle (deliberately so to test the guarding) so will be starving
|
||||||
|
out lower priority tasks. Block for some time to allow give lower
|
||||||
|
priority tasks some processor time. */
|
||||||
|
vTaskDelay( pxParameters->xBlockTime * semtstDELAY_FACTOR );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if( pxParameters->xBlockTime == ( portTickType ) 0 )
|
||||||
|
{
|
||||||
|
/* We have not got the semaphore yet, so no point using the
|
||||||
|
processor. We are not blocking when attempting to obtain the
|
||||||
|
semaphore. */
|
||||||
|
taskYIELD();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* This is called to check that all the created tasks are still running. */
|
||||||
|
portBASE_TYPE xAreSemaphoreTasksStillRunning( void )
|
||||||
|
{
|
||||||
|
static short sLastCheckVariables[ semtstNUM_TASKS ] = { 0 };
|
||||||
|
portBASE_TYPE xTask, xReturn = pdTRUE;
|
||||||
|
|
||||||
|
for( xTask = 0; xTask < semtstNUM_TASKS; xTask++ )
|
||||||
|
{
|
||||||
|
if( sLastCheckVariables[ xTask ] == sCheckVariables[ xTask ] )
|
||||||
|
{
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
sLastCheckVariables[ xTask ] = sCheckVariables[ xTask ];
|
||||||
|
}
|
||||||
|
|
||||||
|
return xReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,353 @@
|
||||||
|
/*
|
||||||
|
FreeRTOS V7.0.1 - Copyright (C) 2011 Real Time Engineers Ltd.
|
||||||
|
|
||||||
|
|
||||||
|
***************************************************************************
|
||||||
|
* *
|
||||||
|
* FreeRTOS tutorial books are available in pdf and paperback. *
|
||||||
|
* Complete, revised, and edited pdf reference manuals are also *
|
||||||
|
* available. *
|
||||||
|
* *
|
||||||
|
* Purchasing FreeRTOS documentation will not only help you, by *
|
||||||
|
* ensuring you get running as quickly as possible and with an *
|
||||||
|
* in-depth knowledge of how to use FreeRTOS, it will also help *
|
||||||
|
* the FreeRTOS project to continue with its mission of providing *
|
||||||
|
* professional grade, cross platform, de facto standard solutions *
|
||||||
|
* for microcontrollers - completely free of charge! *
|
||||||
|
* *
|
||||||
|
* >>> See http://www.FreeRTOS.org/Documentation for details. <<< *
|
||||||
|
* *
|
||||||
|
* Thank you for using FreeRTOS, and thank you for your support! *
|
||||||
|
* *
|
||||||
|
***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
This file is part of the FreeRTOS distribution.
|
||||||
|
|
||||||
|
FreeRTOS is free software; you can redistribute it and/or modify it under
|
||||||
|
the terms of the GNU General Public License (version 2) as published by the
|
||||||
|
Free Software Foundation AND MODIFIED BY the FreeRTOS exception.
|
||||||
|
>>>NOTE<<< The modification to the GPL is included to allow you to
|
||||||
|
distribute a combined work that includes FreeRTOS without being obliged to
|
||||||
|
provide the source code for proprietary components outside of the FreeRTOS
|
||||||
|
kernel. FreeRTOS is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
more details. You should have received a copy of the GNU General Public
|
||||||
|
License and the FreeRTOS license exception along with FreeRTOS; if not it
|
||||||
|
can be viewed here: http://www.freertos.org/a00114.html and also obtained
|
||||||
|
by writing to Richard Barry, contact details for whom are available on the
|
||||||
|
FreeRTOS WEB site.
|
||||||
|
|
||||||
|
1 tab == 4 spaces!
|
||||||
|
|
||||||
|
http://www.FreeRTOS.org - Documentation, latest information, license and
|
||||||
|
contact details.
|
||||||
|
|
||||||
|
http://www.SafeRTOS.com - A version that is certified for use in safety
|
||||||
|
critical systems.
|
||||||
|
|
||||||
|
http://www.OpenRTOS.com - Commercial support, development, porting,
|
||||||
|
licensing and training services.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Creates eight tasks, each of which loops continuously performing a floating
|
||||||
|
* point calculation - using single precision variables.
|
||||||
|
*
|
||||||
|
* All the tasks run at the idle priority and never block or yield. This causes
|
||||||
|
* all eight tasks to time slice with the idle task. Running at the idle priority
|
||||||
|
* means that these tasks will get pre-empted any time another task is ready to run
|
||||||
|
* or a time slice occurs. More often than not the pre-emption will occur mid
|
||||||
|
* calculation, creating a good test of the schedulers context switch mechanism - a
|
||||||
|
* calculation producing an unexpected result could be a symptom of a corruption in
|
||||||
|
* the context of a task.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <math.h>
|
||||||
|
|
||||||
|
/* Scheduler include files. */
|
||||||
|
#include "FreeRTOS.h"
|
||||||
|
#include "task.h"
|
||||||
|
|
||||||
|
/* Demo program include files. */
|
||||||
|
#include "flop.h"
|
||||||
|
|
||||||
|
#define mathSTACK_SIZE configMINIMAL_STACK_SIZE
|
||||||
|
#define mathNUMBER_OF_TASKS ( 8 )
|
||||||
|
|
||||||
|
/* Four tasks, each of which performs a different floating point calculation.
|
||||||
|
Each of the four is created twice. */
|
||||||
|
static portTASK_FUNCTION_PROTO( vCompetingMathTask1, pvParameters );
|
||||||
|
static portTASK_FUNCTION_PROTO( vCompetingMathTask2, pvParameters );
|
||||||
|
static portTASK_FUNCTION_PROTO( vCompetingMathTask3, pvParameters );
|
||||||
|
static portTASK_FUNCTION_PROTO( vCompetingMathTask4, pvParameters );
|
||||||
|
|
||||||
|
/* These variables are used to check that all the tasks are still running. If a
|
||||||
|
task gets a calculation wrong it will
|
||||||
|
stop incrementing its check variable. */
|
||||||
|
static volatile unsigned short usTaskCheck[ mathNUMBER_OF_TASKS ] = { ( unsigned short ) 0 };
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
void vStartMathTasks( unsigned portBASE_TYPE uxPriority )
|
||||||
|
{
|
||||||
|
xTaskCreate( vCompetingMathTask1, ( signed char * ) "Math1", mathSTACK_SIZE, ( void * ) &( usTaskCheck[ 0 ] ), uxPriority, NULL );
|
||||||
|
xTaskCreate( vCompetingMathTask2, ( signed char * ) "Math2", mathSTACK_SIZE, ( void * ) &( usTaskCheck[ 1 ] ), uxPriority, NULL );
|
||||||
|
xTaskCreate( vCompetingMathTask3, ( signed char * ) "Math3", mathSTACK_SIZE, ( void * ) &( usTaskCheck[ 2 ] ), uxPriority, NULL );
|
||||||
|
xTaskCreate( vCompetingMathTask4, ( signed char * ) "Math4", mathSTACK_SIZE, ( void * ) &( usTaskCheck[ 3 ] ), uxPriority, NULL );
|
||||||
|
xTaskCreate( vCompetingMathTask1, ( signed char * ) "Math5", mathSTACK_SIZE, ( void * ) &( usTaskCheck[ 4 ] ), uxPriority, NULL );
|
||||||
|
xTaskCreate( vCompetingMathTask2, ( signed char * ) "Math6", mathSTACK_SIZE, ( void * ) &( usTaskCheck[ 5 ] ), uxPriority, NULL );
|
||||||
|
xTaskCreate( vCompetingMathTask3, ( signed char * ) "Math7", mathSTACK_SIZE, ( void * ) &( usTaskCheck[ 6 ] ), uxPriority, NULL );
|
||||||
|
xTaskCreate( vCompetingMathTask4, ( signed char * ) "Math8", mathSTACK_SIZE, ( void * ) &( usTaskCheck[ 7 ] ), uxPriority, NULL );
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vCompetingMathTask1, pvParameters )
|
||||||
|
{
|
||||||
|
volatile float f1, f2, f3, f4;
|
||||||
|
volatile unsigned short *pusTaskCheckVariable;
|
||||||
|
volatile float fAnswer;
|
||||||
|
short sError = pdFALSE;
|
||||||
|
|
||||||
|
f1 = 123.4567F;
|
||||||
|
f2 = 2345.6789F;
|
||||||
|
f3 = -918.222F;
|
||||||
|
|
||||||
|
fAnswer = ( f1 + f2 ) * f3;
|
||||||
|
|
||||||
|
/* The variable this task increments to show it is still running is passed in
|
||||||
|
as the parameter. */
|
||||||
|
pusTaskCheckVariable = ( unsigned short * ) pvParameters;
|
||||||
|
|
||||||
|
/* Keep performing a calculation and checking the result against a constant. */
|
||||||
|
for(;;)
|
||||||
|
{
|
||||||
|
f1 = 123.4567F;
|
||||||
|
f2 = 2345.6789F;
|
||||||
|
f3 = -918.222F;
|
||||||
|
|
||||||
|
f4 = ( f1 + f2 ) * f3;
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* If the calculation does not match the expected constant, stop the
|
||||||
|
increment of the check variable. */
|
||||||
|
if( fabs( f4 - fAnswer ) > 0.001F )
|
||||||
|
{
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( sError == pdFALSE )
|
||||||
|
{
|
||||||
|
/* If the calculation has always been correct, increment the check
|
||||||
|
variable so we know this task is still running okay. */
|
||||||
|
( *pusTaskCheckVariable )++;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vCompetingMathTask2, pvParameters )
|
||||||
|
{
|
||||||
|
volatile float f1, f2, f3, f4;
|
||||||
|
volatile unsigned short *pusTaskCheckVariable;
|
||||||
|
volatile float fAnswer;
|
||||||
|
short sError = pdFALSE;
|
||||||
|
|
||||||
|
f1 = -389.38F;
|
||||||
|
f2 = 32498.2F;
|
||||||
|
f3 = -2.0001F;
|
||||||
|
|
||||||
|
fAnswer = ( f1 / f2 ) * f3;
|
||||||
|
|
||||||
|
|
||||||
|
/* The variable this task increments to show it is still running is passed in
|
||||||
|
as the parameter. */
|
||||||
|
pusTaskCheckVariable = ( unsigned short * ) pvParameters;
|
||||||
|
|
||||||
|
/* Keep performing a calculation and checking the result against a constant. */
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
f1 = -389.38F;
|
||||||
|
f2 = 32498.2F;
|
||||||
|
f3 = -2.0001F;
|
||||||
|
|
||||||
|
f4 = ( f1 / f2 ) * f3;
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* If the calculation does not match the expected constant, stop the
|
||||||
|
increment of the check variable. */
|
||||||
|
if( fabs( f4 - fAnswer ) > 0.001F )
|
||||||
|
{
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( sError == pdFALSE )
|
||||||
|
{
|
||||||
|
/* If the calculation has always been correct, increment the check
|
||||||
|
variable so we know
|
||||||
|
this task is still running okay. */
|
||||||
|
( *pusTaskCheckVariable )++;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vCompetingMathTask3, pvParameters )
|
||||||
|
{
|
||||||
|
volatile float *pfArray, fTotal1, fTotal2, fDifference, fPosition;
|
||||||
|
volatile unsigned short *pusTaskCheckVariable;
|
||||||
|
const size_t xArraySize = 10;
|
||||||
|
size_t xPosition;
|
||||||
|
short sError = pdFALSE;
|
||||||
|
|
||||||
|
/* The variable this task increments to show it is still running is passed in
|
||||||
|
as the parameter. */
|
||||||
|
pusTaskCheckVariable = ( unsigned short * ) pvParameters;
|
||||||
|
|
||||||
|
pfArray = ( float * ) pvPortMalloc( xArraySize * sizeof( float ) );
|
||||||
|
|
||||||
|
/* Keep filling an array, keeping a running total of the values placed in the
|
||||||
|
array. Then run through the array adding up all the values. If the two totals
|
||||||
|
do not match, stop the check variable from incrementing. */
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
fTotal1 = 0.0F;
|
||||||
|
fTotal2 = 0.0F;
|
||||||
|
fPosition = 0.0F;
|
||||||
|
|
||||||
|
for( xPosition = 0; xPosition < xArraySize; xPosition++ )
|
||||||
|
{
|
||||||
|
pfArray[ xPosition ] = fPosition + 5.5F;
|
||||||
|
fTotal1 += fPosition + 5.5F;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
for( xPosition = 0; xPosition < xArraySize; xPosition++ )
|
||||||
|
{
|
||||||
|
fTotal2 += pfArray[ xPosition ];
|
||||||
|
}
|
||||||
|
|
||||||
|
fDifference = fTotal1 - fTotal2;
|
||||||
|
if( fabs( fDifference ) > 0.001F )
|
||||||
|
{
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if( sError == pdFALSE )
|
||||||
|
{
|
||||||
|
/* If the calculation has always been correct, increment the check
|
||||||
|
variable so we know this task is still running okay. */
|
||||||
|
( *pusTaskCheckVariable )++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
static portTASK_FUNCTION( vCompetingMathTask4, pvParameters )
|
||||||
|
{
|
||||||
|
volatile float *pfArray, fTotal1, fTotal2, fDifference, fPosition;
|
||||||
|
volatile unsigned short *pusTaskCheckVariable;
|
||||||
|
const size_t xArraySize = 10;
|
||||||
|
size_t xPosition;
|
||||||
|
short sError = pdFALSE;
|
||||||
|
|
||||||
|
/* The variable this task increments to show it is still running is passed in
|
||||||
|
as the parameter. */
|
||||||
|
pusTaskCheckVariable = ( unsigned short * ) pvParameters;
|
||||||
|
|
||||||
|
pfArray = ( float * ) pvPortMalloc( xArraySize * sizeof( float ) );
|
||||||
|
|
||||||
|
/* Keep filling an array, keeping a running total of the values placed in the
|
||||||
|
array. Then run through the array adding up all the values. If the two totals
|
||||||
|
do not match, stop the check variable from incrementing. */
|
||||||
|
for( ;; )
|
||||||
|
{
|
||||||
|
fTotal1 = 0.0F;
|
||||||
|
fTotal2 = 0.0F;
|
||||||
|
fPosition = 0.0F;
|
||||||
|
|
||||||
|
for( xPosition = 0; xPosition < xArraySize; xPosition++ )
|
||||||
|
{
|
||||||
|
pfArray[ xPosition ] = fPosition * 12.123F;
|
||||||
|
fTotal1 += fPosition * 12.123F;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
for( xPosition = 0; xPosition < xArraySize; xPosition++ )
|
||||||
|
{
|
||||||
|
fTotal2 += pfArray[ xPosition ];
|
||||||
|
}
|
||||||
|
|
||||||
|
fDifference = fTotal1 - fTotal2;
|
||||||
|
if( fabs( fDifference ) > 0.001F )
|
||||||
|
{
|
||||||
|
sError = pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if configUSE_PREEMPTION == 0
|
||||||
|
taskYIELD();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if( sError == pdFALSE )
|
||||||
|
{
|
||||||
|
/* If the calculation has always been correct, increment the check
|
||||||
|
variable so we know this task is still running okay. */
|
||||||
|
( *pusTaskCheckVariable )++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*-----------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* This is called to check that all the created tasks are still running. */
|
||||||
|
portBASE_TYPE xAreMathsTaskStillRunning( void )
|
||||||
|
{
|
||||||
|
/* Keep a history of the check variables so we know if they have been incremented
|
||||||
|
since the last call. */
|
||||||
|
static unsigned short usLastTaskCheck[ mathNUMBER_OF_TASKS ] = { ( unsigned short ) 0 };
|
||||||
|
portBASE_TYPE xReturn = pdTRUE, xTask;
|
||||||
|
|
||||||
|
/* Check the maths tasks are still running by ensuring their check variables
|
||||||
|
are still incrementing. */
|
||||||
|
for( xTask = 0; xTask < mathNUMBER_OF_TASKS; xTask++ )
|
||||||
|
{
|
||||||
|
if( usTaskCheck[ xTask ] == usLastTaskCheck[ xTask ] )
|
||||||
|
{
|
||||||
|
/* The check has not incremented so an error exists. */
|
||||||
|
xReturn = pdFALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
usLastTaskCheck[ xTask ] = usTaskCheck[ xTask ];
|
||||||
|
}
|
||||||
|
|
||||||
|
return xReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue