You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1049 lines
37 KiB
C
1049 lines
37 KiB
C
/*
|
|
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 of timers. Some timers are created before the scheduler
|
|
* is started, and some after.
|
|
*/
|
|
|
|
/* Standard includes. */
|
|
#include <string.h>
|
|
|
|
/* Scheduler include files. */
|
|
#include "FreeRTOS.h"
|
|
#include "task.h"
|
|
#include "timers.h"
|
|
|
|
/* Demo program include files. */
|
|
#include "TimerDemo.h"
|
|
|
|
#if ( configTIMER_TASK_PRIORITY < 1 )
|
|
#error configTIMER_TASK_PRIORITY must be set to at least 1 for this test/demo to function correctly.
|
|
#endif
|
|
|
|
#define tmrdemoDONT_BLOCK ( ( portTickType ) 0 )
|
|
#define tmrdemoONE_SHOT_TIMER_PERIOD ( xBasePeriod * ( portTickType ) 3 )
|
|
#define trmdemoNUM_TIMER_RESETS ( ( unsigned char ) 10 )
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/* The callback functions used by the timers. These each increment a counter
|
|
to indicate which timer has expired. The auto-reload timers that are used by
|
|
the test task (as opposed to being used from an ISR) all share the same
|
|
prvAutoReloadTimerCallback() callback function, and use the ID of the
|
|
pxExpiredTimer parameter passed into that function to know which counter to
|
|
increment. The other timers all have their own unique callback function and
|
|
simply increment their counters without using the callback function parameter. */
|
|
static void prvAutoReloadTimerCallback( xTimerHandle pxExpiredTimer );
|
|
static void prvOneShotTimerCallback( xTimerHandle pxExpiredTimer );
|
|
static void prvTimerTestTask( void *pvParameters );
|
|
static void prvISRAutoReloadTimerCallback( xTimerHandle pxExpiredTimer );
|
|
static void prvISROneShotTimerCallback( xTimerHandle pxExpiredTimer );
|
|
|
|
/* The test functions used by the timer test task. These manipulate the auto
|
|
reload and one shot timers in various ways, then delay, then inspect the timers
|
|
to ensure they have behaved as expected. */
|
|
static void prvTest1_CreateTimersWithoutSchedulerRunning( void );
|
|
static void prvTest2_CheckTaskAndTimersInitialState( void );
|
|
static void prvTest3_CheckAutoReloadExpireRates( void );
|
|
static void prvTest4_CheckAutoReloadTimersCanBeStopped( void );
|
|
static void prvTest5_CheckBasicOneShotTimerBehaviour( void );
|
|
static void prvTest6_CheckAutoReloadResetBehaviour( void );
|
|
static void prvResetStartConditionsForNextIteration( void );
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/* Flag that will be latched to pdFAIL should any unexpected behaviour be
|
|
detected in any of the demo tests. */
|
|
static volatile portBASE_TYPE xTestStatus = pdPASS;
|
|
|
|
/* 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 long ulLoopCounter = 0;
|
|
|
|
/* A set of auto reload timers - each of which use the same callback function.
|
|
The callback function uses the timer ID to index into, and then increment, a
|
|
counter in the ucAutoReloadTimerCounters[] array. The auto reload timers
|
|
referenced from xAutoReloadTimers[] are used by the prvTimerTestTask task. */
|
|
static xTimerHandle xAutoReloadTimers[ configTIMER_QUEUE_LENGTH + 1 ] = { 0 };
|
|
static unsigned char ucAutoReloadTimerCounters[ configTIMER_QUEUE_LENGTH + 1 ] = { 0 };
|
|
|
|
/* The one shot timer is configured to use a callback function that increments
|
|
ucOneShotTimerCounter each time it gets called. */
|
|
static xTimerHandle xOneShotTimer = NULL;
|
|
static unsigned char ucOneShotTimerCounter = ( unsigned char ) 0;
|
|
|
|
/* The ISR reload timer is controlled from the tick hook to exercise the timer
|
|
API functions that can be used from an ISR. It is configured to increment
|
|
ucISRReloadTimerCounter each time its callback function is executed. */
|
|
static xTimerHandle xISRAutoReloadTimer = NULL;
|
|
static unsigned char ucISRAutoReloadTimerCounter = ( unsigned char ) 0;
|
|
|
|
/* The ISR one shot timer is controlled from the tick hook to exercise the timer
|
|
API functions that can be used from an ISR. It is configured to increment
|
|
ucISRReloadTimerCounter each time its callback function is executed. */
|
|
static xTimerHandle xISROneShotTimer = NULL;
|
|
static unsigned char ucISROneShotTimerCounter = ( unsigned char ) 0;
|
|
|
|
/* The period of all the timers are a multiple of the base period. The base
|
|
period is configured by the parameter to vStartTimerDemoTask(). */
|
|
static portTickType xBasePeriod = 0;
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
void vStartTimerDemoTask( portTickType xBasePeriodIn )
|
|
{
|
|
/* Start with the timer and counter arrays clear - this is only necessary
|
|
where the compiler does not clear them automatically on start up. */
|
|
memset( ucAutoReloadTimerCounters, 0x00, sizeof( ucAutoReloadTimerCounters ) );
|
|
memset( xAutoReloadTimers, 0x00, sizeof( xAutoReloadTimers ) );
|
|
|
|
/* Store the period from which all the timer periods will be generated from
|
|
(multiples of). */
|
|
xBasePeriod = xBasePeriodIn;
|
|
|
|
/* Create a set of timers for use by this demo/test. */
|
|
prvTest1_CreateTimersWithoutSchedulerRunning();
|
|
|
|
/* Create the task that will control and monitor the timers. This is
|
|
created at a lower priority than the timer service task to ensure, as
|
|
far as it is concerned, commands on timers are actioned immediately
|
|
(sending a command to the timer service task will unblock the timer service
|
|
task, which will then preempt this task). */
|
|
if( xTestStatus != pdFAIL )
|
|
{
|
|
xTaskCreate( prvTimerTestTask, ( signed portCHAR * ) "Tmr Tst", configMINIMAL_STACK_SIZE, NULL, configTIMER_TASK_PRIORITY - 1, NULL );
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvTimerTestTask( void *pvParameters )
|
|
{
|
|
( void ) pvParameters;
|
|
|
|
/* Create a one-shot timer for use later on in this test. */
|
|
xOneShotTimer = xTimerCreate( ( const signed char * ) "Oneshot Timer",/* Text name to facilitate debugging. The kernel does not use this itself. */
|
|
tmrdemoONE_SHOT_TIMER_PERIOD, /* The period for the timer. */
|
|
pdFALSE, /* Don't auto-reload - hence a one shot timer. */
|
|
( void * ) 0, /* The timer identifier. In this case this is not used as the timer has its own callback. */
|
|
prvOneShotTimerCallback ); /* The callback to be called when the timer expires. */
|
|
|
|
if( xOneShotTimer == NULL )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
|
|
/* Ensure all the timers are in their expected initial state. This
|
|
depends on the timer service task having a higher priority than this task. */
|
|
prvTest2_CheckTaskAndTimersInitialState();
|
|
|
|
for( ;; )
|
|
{
|
|
/* Check the auto reload timers expire at the expected/correct rates. */
|
|
prvTest3_CheckAutoReloadExpireRates();
|
|
|
|
/* Check the auto reload timers can be stopped correctly, and correctly
|
|
report their state. */
|
|
prvTest4_CheckAutoReloadTimersCanBeStopped();
|
|
|
|
/* Check the one shot timer only calls its callback once after it has been
|
|
started, and that it reports its state correctly. */
|
|
prvTest5_CheckBasicOneShotTimerBehaviour();
|
|
|
|
/* Check timer reset behaviour. */
|
|
prvTest6_CheckAutoReloadResetBehaviour();
|
|
|
|
/* Start the timers again to restart all the tests over again. */
|
|
prvResetStartConditionsForNextIteration();
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/* This is called to check that the created task is still running and has not
|
|
detected any errors. */
|
|
portBASE_TYPE xAreTimerDemoTasksStillRunning( portTickType xCycleFrequency )
|
|
{
|
|
static unsigned long ulLastLoopCounter = 0UL;
|
|
portTickType xMaxBlockTimeUsedByTheseTests, xLoopCounterIncrementTimeMax;
|
|
static portTickType xIterationsWithoutCounterIncrement = ( portTickType ) 0, xLastCycleFrequency;
|
|
|
|
if( xLastCycleFrequency != xCycleFrequency )
|
|
{
|
|
/* The cycle frequency has probably become much faster due to an error
|
|
elsewhere. Start counting Iterations again. */
|
|
xIterationsWithoutCounterIncrement = ( portTickType ) 0;
|
|
xLastCycleFrequency = xCycleFrequency;
|
|
}
|
|
|
|
/* Calculate the maximum number of times that it is permissible for this
|
|
function to be called without ulLoopCounter being incremented. This is
|
|
necessary because the tests in this file block for extended periods, and the
|
|
block period might be longer than the time between calls to this function. */
|
|
xMaxBlockTimeUsedByTheseTests = ( ( portTickType ) configTIMER_QUEUE_LENGTH ) * xBasePeriod;
|
|
xLoopCounterIncrementTimeMax = xMaxBlockTimeUsedByTheseTests / xCycleFrequency;
|
|
|
|
/* If the demo task is still running then we expect the loopcounter to
|
|
have incremented every xLoopCounterIncrementTimeMax calls. */
|
|
if( ulLastLoopCounter == ulLoopCounter )
|
|
{
|
|
xIterationsWithoutCounterIncrement++;
|
|
if( xIterationsWithoutCounterIncrement > xLoopCounterIncrementTimeMax )
|
|
{
|
|
/* The tests appear to be no longer running (stalled). */
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* ulLoopCounter changed, so the count of times this function was called
|
|
without a change can be reset to zero. */
|
|
xIterationsWithoutCounterIncrement = ( portTickType ) 0;
|
|
}
|
|
|
|
ulLastLoopCounter = ulLoopCounter;
|
|
|
|
/* Errors detected in the task itself will have latched xTestStatus
|
|
to pdFAIL. */
|
|
|
|
return xTestStatus;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvTest1_CreateTimersWithoutSchedulerRunning( void )
|
|
{
|
|
unsigned portBASE_TYPE xTimer;
|
|
|
|
for( xTimer = 0; xTimer < configTIMER_QUEUE_LENGTH; xTimer++ )
|
|
{
|
|
/* As the timer queue is not yet full, it should be possible to both create
|
|
and start a timer. These timers are being started before the scheduler has
|
|
been started, so their block times should get set to zero within the timer
|
|
API itself. */
|
|
xAutoReloadTimers[ xTimer ] = xTimerCreate( ( const signed char * )"FR Timer", /* Text name to facilitate debugging. The kernel does not use this itself. */
|
|
( ( xTimer + ( portTickType ) 1 ) * xBasePeriod ),/* The period for the timer. The plus 1 ensures a period of zero is not specified. */
|
|
pdTRUE, /* Auto-reload is set to true. */
|
|
( void * ) xTimer, /* An identifier for the timer as all the auto reload timers use the same callback. */
|
|
prvAutoReloadTimerCallback ); /* The callback to be called when the timer expires. */
|
|
|
|
if( xAutoReloadTimers[ xTimer ] == NULL )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
else
|
|
{
|
|
/* The scheduler has not yet started, so the block period of
|
|
portMAX_DELAY should just get set to zero in xTimerStart(). Also,
|
|
the timer queue is not yet full so xTimerStart() should return
|
|
pdPASS. */
|
|
if( xTimerStart( xAutoReloadTimers[ xTimer ], portMAX_DELAY ) != pdPASS )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
}
|
|
|
|
/* The timers queue should now be full, so it should be possible to create
|
|
another timer, but not possible to start it (the timer queue will not get
|
|
drained until the scheduler has been started. */
|
|
xAutoReloadTimers[ configTIMER_QUEUE_LENGTH ] = xTimerCreate( ( const signed char * ) "FR Timer", /* Text name to facilitate debugging. The kernel does not use this itself. */
|
|
( configTIMER_QUEUE_LENGTH * xBasePeriod ), /* The period for the timer. */
|
|
pdTRUE, /* Auto-reload is set to true. */
|
|
( void * ) xTimer, /* An identifier for the timer as all the auto reload timers use the same callback. */
|
|
prvAutoReloadTimerCallback ); /* The callback executed when the timer expires. */
|
|
|
|
if( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH ] == NULL )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
else
|
|
{
|
|
if( xTimerStart( xAutoReloadTimers[ xTimer ], portMAX_DELAY ) == pdPASS )
|
|
{
|
|
/* This time it would not be expected that the timer could be
|
|
started at this point. */
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
|
|
/* Create the timers that are used from the tick interrupt to test the timer
|
|
API functions that can be called from an ISR. */
|
|
xISRAutoReloadTimer = xTimerCreate( ( const signed char * ) "ISR AR", /* The text name given to the timer. */
|
|
0xffff, /* The timer is not given a period yet - this will be done from the tick hook, but a period of 0 is invalid. */
|
|
pdTRUE, /* This is an auto reload timer. */
|
|
( void * ) NULL, /* The identifier is not required. */
|
|
prvISRAutoReloadTimerCallback ); /* The callback that is executed when the timer expires. */
|
|
|
|
xISROneShotTimer = xTimerCreate( ( const signed char * ) "ISR OS", /* The text name given to the timer. */
|
|
0xffff, /* The timer is not given a period yet - this will be done from the tick hook, but a period of 0 is invalid. */
|
|
pdFALSE, /* This is a one shot timer. */
|
|
( void * ) NULL, /* The identifier is not required. */
|
|
prvISROneShotTimerCallback ); /* The callback that is executed when the timer expires. */
|
|
|
|
if( ( xISRAutoReloadTimer == NULL ) || ( xISROneShotTimer == NULL ) )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvTest2_CheckTaskAndTimersInitialState( void )
|
|
{
|
|
unsigned char ucTimer;
|
|
|
|
/* Ensure all the timers are in their expected initial state. This depends
|
|
on the timer service task having a higher priority than this task.
|
|
|
|
auto reload timers 0 to ( configTIMER_QUEUE_LENGTH - 1 ) should now be active,
|
|
and auto reload timer configTIMER_QUEUE_LENGTH should not yet be active (it
|
|
could not be started prior to the scheduler being started when it was
|
|
created). */
|
|
for( ucTimer = 0; ucTimer < ( unsigned char ) configTIMER_QUEUE_LENGTH; ucTimer++ )
|
|
{
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ ucTimer ] ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH ] ) != pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvTest3_CheckAutoReloadExpireRates( void )
|
|
{
|
|
unsigned char ucMaxAllowableValue, ucMinAllowableValue, ucTimer;
|
|
portTickType xBlockPeriod, xTimerPeriod, xExpectedNumber;
|
|
|
|
/* Check the auto reload timers expire at the expected rates. */
|
|
|
|
|
|
/* Delaying for configTIMER_QUEUE_LENGTH * xBasePeriod ticks should allow
|
|
all the auto reload timers to expire at least once. */
|
|
xBlockPeriod = ( ( portTickType ) configTIMER_QUEUE_LENGTH ) * xBasePeriod;
|
|
vTaskDelay( xBlockPeriod );
|
|
|
|
/* Check that all the auto reload timers have called their callback
|
|
function the expected number of times. */
|
|
for( ucTimer = 0; ucTimer < ( unsigned char ) configTIMER_QUEUE_LENGTH; ucTimer++ )
|
|
{
|
|
/* The expected number of expiries is equal to the block period divided
|
|
by the timer period. */
|
|
xTimerPeriod = ( ( ( portTickType ) ucTimer + ( portTickType ) 1 ) * xBasePeriod );
|
|
xExpectedNumber = xBlockPeriod / xTimerPeriod;
|
|
|
|
ucMaxAllowableValue = ( ( unsigned char ) xExpectedNumber ) ;
|
|
ucMinAllowableValue = ( ( unsigned char ) xExpectedNumber - ( unsigned char ) 1 );
|
|
|
|
if( ( ucAutoReloadTimerCounters[ ucTimer ] < ucMinAllowableValue ) ||
|
|
( ucAutoReloadTimerCounters[ ucTimer ] > ucMaxAllowableValue )
|
|
)
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
|
|
if( xTestStatus == pdPASS )
|
|
{
|
|
/* No errors have been reported so increment the loop counter so the
|
|
check task knows this task is still running. */
|
|
ulLoopCounter++;
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvTest4_CheckAutoReloadTimersCanBeStopped( void )
|
|
{
|
|
unsigned char ucTimer;
|
|
|
|
/* Check the auto reload timers can be stopped correctly, and correctly
|
|
report their state. */
|
|
|
|
/* Stop all the active timers. */
|
|
for( ucTimer = 0; ucTimer < ( unsigned char ) configTIMER_QUEUE_LENGTH; ucTimer++ )
|
|
{
|
|
/* The timer has not been stopped yet! */
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ ucTimer ] ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Now stop the timer. This will appear to happen immediately to
|
|
this task because this task is running at a priority below the
|
|
timer service task. */
|
|
xTimerStop( xAutoReloadTimers[ ucTimer ], tmrdemoDONT_BLOCK );
|
|
|
|
/* The timer should now be inactive. */
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ ucTimer ] ) != pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
|
|
taskENTER_CRITICAL();
|
|
{
|
|
/* The timer in array position configTIMER_QUEUE_LENGTH should not
|
|
be active. The critical section is used to ensure the timer does
|
|
not call its callback between the next line running and the array
|
|
being cleared back to zero, as that would mask an error condition. */
|
|
if( ucAutoReloadTimerCounters[ configTIMER_QUEUE_LENGTH ] != ( unsigned char ) 0 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Clear the timer callback count. */
|
|
memset( ( void * ) ucAutoReloadTimerCounters, 0, sizeof( ucAutoReloadTimerCounters ) );
|
|
}
|
|
taskEXIT_CRITICAL();
|
|
|
|
/* The timers are now all inactive, so this time, after delaying, none
|
|
of the callback counters should have incremented. */
|
|
vTaskDelay( ( ( portTickType ) configTIMER_QUEUE_LENGTH ) * xBasePeriod );
|
|
for( ucTimer = 0; ucTimer < ( unsigned char ) configTIMER_QUEUE_LENGTH; ucTimer++ )
|
|
{
|
|
if( ucAutoReloadTimerCounters[ ucTimer ] != ( unsigned char ) 0 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
|
|
if( xTestStatus == pdPASS )
|
|
{
|
|
/* No errors have been reported so increment the loop counter so
|
|
the check task knows this task is still running. */
|
|
ulLoopCounter++;
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvTest5_CheckBasicOneShotTimerBehaviour( void )
|
|
{
|
|
/* Check the one shot timer only calls its callback once after it has been
|
|
started, and that it reports its state correctly. */
|
|
|
|
/* The one shot timer should not be active yet. */
|
|
if( xTimerIsTimerActive( xOneShotTimer ) != pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucOneShotTimerCounter != ( unsigned char ) 0 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Start the one shot timer and check that it reports its state correctly. */
|
|
xTimerStart( xOneShotTimer, tmrdemoDONT_BLOCK );
|
|
if( xTimerIsTimerActive( xOneShotTimer ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Delay for three times as long as the one shot timer period, then check
|
|
to ensure it has only called its callback once, and is now not in the
|
|
active state. */
|
|
vTaskDelay( tmrdemoONE_SHOT_TIMER_PERIOD * ( portTickType ) 3 );
|
|
|
|
if( xTimerIsTimerActive( xOneShotTimer ) != pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucOneShotTimerCounter != ( unsigned char ) 1 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
else
|
|
{
|
|
/* Reset the one shot timer callback count. */
|
|
ucOneShotTimerCounter = ( unsigned char ) 0;
|
|
}
|
|
|
|
if( xTestStatus == pdPASS )
|
|
{
|
|
/* No errors have been reported so increment the loop counter so the
|
|
check task knows this task is still running. */
|
|
ulLoopCounter++;
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvTest6_CheckAutoReloadResetBehaviour( void )
|
|
{
|
|
unsigned char ucTimer;
|
|
|
|
/* Check timer reset behaviour. */
|
|
|
|
/* Restart the one shot timer and check it reports its status correctly. */
|
|
xTimerStart( xOneShotTimer, tmrdemoDONT_BLOCK );
|
|
if( xTimerIsTimerActive( xOneShotTimer ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Restart one of the auto reload timers and check that it reports its
|
|
status correctly. */
|
|
xTimerStart( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH - 1 ], tmrdemoDONT_BLOCK );
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH - 1 ] ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
for( ucTimer = 0; ucTimer < trmdemoNUM_TIMER_RESETS; ucTimer++ )
|
|
{
|
|
/* Delay for half as long as the one shot timer period, then reset it.
|
|
It should never expire while this is done, so its callback count should
|
|
never increment. */
|
|
vTaskDelay( tmrdemoONE_SHOT_TIMER_PERIOD / 2 );
|
|
|
|
/* Check both running timers are still active, but have not called their
|
|
callback functions. */
|
|
if( xTimerIsTimerActive( xOneShotTimer ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucOneShotTimerCounter != ( unsigned char ) 0 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH - 1 ] ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucAutoReloadTimerCounters[ configTIMER_QUEUE_LENGTH - 1 ] != ( unsigned char ) 0 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Reset both running timers. */
|
|
xTimerReset( xOneShotTimer, tmrdemoDONT_BLOCK );
|
|
xTimerReset( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH - 1 ], tmrdemoDONT_BLOCK );
|
|
|
|
if( xTestStatus == pdPASS )
|
|
{
|
|
/* No errors have been reported so increment the loop counter so
|
|
the check task knows this task is still running. */
|
|
ulLoopCounter++;
|
|
}
|
|
}
|
|
|
|
/* Finally delay long enough for both running timers to expire. */
|
|
vTaskDelay( ( ( portTickType ) configTIMER_QUEUE_LENGTH ) * xBasePeriod );
|
|
|
|
/* The timers were not reset during the above delay period so should now
|
|
both have called their callback functions. */
|
|
if( ucOneShotTimerCounter != ( unsigned char ) 1 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucAutoReloadTimerCounters[ configTIMER_QUEUE_LENGTH - 1 ] == 0 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* The one shot timer should no longer be active, while the auto reload
|
|
timer should still be active. */
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH - 1 ] ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( xTimerIsTimerActive( xOneShotTimer ) == pdTRUE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Stop the auto reload timer again. */
|
|
xTimerStop( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH - 1 ], tmrdemoDONT_BLOCK );
|
|
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ configTIMER_QUEUE_LENGTH - 1 ] ) != pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Clear the timer callback counts, ready for another iteration of these
|
|
tests. */
|
|
ucAutoReloadTimerCounters[ configTIMER_QUEUE_LENGTH - 1 ] = ( unsigned char ) 0;
|
|
ucOneShotTimerCounter = ( unsigned char ) 0;
|
|
|
|
if( xTestStatus == pdPASS )
|
|
{
|
|
/* No errors have been reported so increment the loop counter so the check
|
|
task knows this task is still running. */
|
|
ulLoopCounter++;
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvResetStartConditionsForNextIteration( void )
|
|
{
|
|
unsigned char ucTimer;
|
|
|
|
/* Start the timers again to start all the tests over again. */
|
|
|
|
/* Start the timers again. */
|
|
for( ucTimer = 0; ucTimer < ( unsigned char ) configTIMER_QUEUE_LENGTH; ucTimer++ )
|
|
{
|
|
/* The timer has not been started yet! */
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ ucTimer ] ) != pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Now start the timer. This will appear to happen immediately to
|
|
this task because this task is running at a priority below the timer
|
|
service task. */
|
|
xTimerStart( xAutoReloadTimers[ ucTimer ], tmrdemoDONT_BLOCK );
|
|
|
|
/* The timer should now be active. */
|
|
if( xTimerIsTimerActive( xAutoReloadTimers[ ucTimer ] ) == pdFALSE )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
|
|
if( xTestStatus == pdPASS )
|
|
{
|
|
/* No errors have been reported so increment the loop counter so the
|
|
check task knows this task is still running. */
|
|
ulLoopCounter++;
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
void vTimerPeriodicISRTests( void )
|
|
{
|
|
static portTickType uxTick = ( portTickType ) -1;
|
|
|
|
/* The xHigherPriorityTaskWoken parameter is not used in this case as this
|
|
function is called from the tick hook anyway. However the API required it
|
|
to be present. */
|
|
portBASE_TYPE xHigherPriorityTaskWoken = pdTRUE;
|
|
portTickType xMargin;
|
|
|
|
if( configTIMER_TASK_PRIORITY != ( configMAX_PRIORITIES - 1 ) )
|
|
{
|
|
/* The timer service task is not the highest priority task, so it cannot
|
|
be assumed that timings will be exact. Timers should never call their
|
|
callback before their expiry time, but a margin is permissible for calling
|
|
their callback after their expiry time. If exact timing is required then
|
|
configTIMER_TASK_PRIORITY must be set to ensure the timer service task
|
|
is the highest priority task in the system. */
|
|
xMargin = 5;
|
|
}
|
|
else
|
|
{
|
|
xMargin = 1;
|
|
}
|
|
|
|
/* This test is called from the tick ISR even when the scheduler is suspended.
|
|
Therefore, it is possible for the xTickCount to be temporarily less than the
|
|
uxTicks count maintained in this function. That can result in calculated
|
|
unblock times being too short, as this function is not called as missed ticks
|
|
(ticks that occur while the scheduler is suspended) are unwound to re-instate
|
|
the real tick value. Therefore, if this happens, just abandon the test
|
|
and start again. */
|
|
if( xTaskGetSchedulerState() != taskSCHEDULER_RUNNING )
|
|
{
|
|
uxTick = ( portTickType ) -1;
|
|
}
|
|
else
|
|
{
|
|
uxTick++;
|
|
}
|
|
|
|
if( uxTick == 0 )
|
|
{
|
|
/* The timers will have been created, but not started. Start them
|
|
now by setting their period. */
|
|
ucISRAutoReloadTimerCounter = 0;
|
|
ucISROneShotTimerCounter = 0;
|
|
xTimerChangePeriodFromISR( xISRAutoReloadTimer, xBasePeriod, &xHigherPriorityTaskWoken );
|
|
xTimerChangePeriodFromISR( xISROneShotTimer, xBasePeriod, &xHigherPriorityTaskWoken );
|
|
}
|
|
else if( uxTick == xBasePeriod )
|
|
{
|
|
/* Neither timer should have expired yet. */
|
|
if( ( ucISRAutoReloadTimerCounter != 0 ) || ( ucISROneShotTimerCounter != 0 ) )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else if( uxTick == ( xBasePeriod + xMargin ) )
|
|
{
|
|
/* Both timers should now have expired once. The auto reload timer will
|
|
still be active, but the one shot timer should now have stopped. */
|
|
if( ( ucISRAutoReloadTimerCounter != 1 ) || ( ucISROneShotTimerCounter != 1 ) )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else if( uxTick == ( 2 * xBasePeriod ) )
|
|
{
|
|
/* The auto reload timer will still be active, but the one shot timer
|
|
should now have stopped - however, at this time neither of the timers
|
|
should have expired again since the last test. */
|
|
if( ( ucISRAutoReloadTimerCounter != 1 ) || ( ucISROneShotTimerCounter != 1 ) )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else if( uxTick == ( ( 2 * xBasePeriod ) + xMargin ) )
|
|
{
|
|
/* The auto reload timer will still be active, but the one shot timer
|
|
should now have stopped. At this time the auto reload timer should have
|
|
expired again, but the one shot timer count should not have changed. */
|
|
if( ucISRAutoReloadTimerCounter != 2 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 1 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else if( uxTick == ( ( 2 * xBasePeriod ) + ( xBasePeriod >> ( portTickType ) 2U ) ) )
|
|
{
|
|
/* The auto reload timer will still be active, but the one shot timer
|
|
should now have stopped. Again though, at this time, neither timer call
|
|
back should have been called since the last test. */
|
|
if( ucISRAutoReloadTimerCounter != 2 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 1 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else if( uxTick == ( 3 * xBasePeriod ) )
|
|
{
|
|
/* Start the one shot timer again. */
|
|
xTimerStartFromISR( xISROneShotTimer, &xHigherPriorityTaskWoken );
|
|
}
|
|
else if( uxTick == ( ( 3 * xBasePeriod ) + xMargin ) )
|
|
{
|
|
/* The auto reload timer and one shot timer will be active. At
|
|
this time the auto reload timer should have expired again, but the one
|
|
shot timer count should not have changed yet. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 1 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Now stop the auto reload timer. The one shot timer was started
|
|
a few ticks ago. */
|
|
xTimerStopFromISR( xISRAutoReloadTimer, &xHigherPriorityTaskWoken );
|
|
}
|
|
else if( uxTick == ( 4 * xBasePeriod ) )
|
|
{
|
|
/* The auto reload timer is now stopped, and the one shot timer is
|
|
active, but at this time neither timer should have expired since the
|
|
last test. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 1 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else if( uxTick == ( ( 4 * xBasePeriod ) + xMargin ) )
|
|
{
|
|
/* The auto reload timer is now stopped, and the one shot timer is
|
|
active. The one shot timer should have expired again, but the auto
|
|
reload timer should not have executed its callback. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 2 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else if( uxTick == ( ( 8 * xBasePeriod ) + xMargin ) )
|
|
{
|
|
/* The auto reload timer is now stopped, and the one shot timer has
|
|
already expired and then stopped itself. Both callback counters should
|
|
not have incremented since the last test. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 2 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
/* Now reset the one shot timer. */
|
|
xTimerResetFromISR( xISROneShotTimer, &xHigherPriorityTaskWoken );
|
|
}
|
|
else if( uxTick == ( 9 * xBasePeriod ) )
|
|
{
|
|
/* Only the one shot timer should be running, but it should not have
|
|
expired since the last test. Check the callback counters have not
|
|
incremented, then reset the one shot timer again. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 2 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
xTimerResetFromISR( xISROneShotTimer, &xHigherPriorityTaskWoken );
|
|
}
|
|
else if( uxTick == ( 10 * xBasePeriod ) )
|
|
{
|
|
/* Only the one shot timer should be running, but it should not have
|
|
expired since the last test. Check the callback counters have not
|
|
incremented, then reset the one shot timer again. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 2 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
xTimerResetFromISR( xISROneShotTimer, &xHigherPriorityTaskWoken );
|
|
}
|
|
else if( uxTick == ( 11 * xBasePeriod ) )
|
|
{
|
|
/* Only the one shot timer should be running, but it should not have
|
|
expired since the last test. Check the callback counters have not
|
|
incremented, then reset the one shot timer once again. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 2 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
xTimerResetFromISR( xISROneShotTimer, &xHigherPriorityTaskWoken );
|
|
}
|
|
else if( uxTick == ( ( 12 * xBasePeriod ) + xMargin ) )
|
|
{
|
|
/* Only the one shot timer should have been running and this time it
|
|
should have expired. Check its callback count has been incremented.
|
|
The auto reload timer is still not running so should still have the same
|
|
count value. This time the one shot timer is not reset so should not
|
|
restart from its expiry period again. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
else if( uxTick == ( 15 * xBasePeriod ) )
|
|
{
|
|
/* Neither timer should be running now. Check neither callback count
|
|
has incremented, then go back to the start to run these tests all
|
|
over again. */
|
|
if( ucISRAutoReloadTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
if( ucISROneShotTimerCounter != 3 )
|
|
{
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
|
|
uxTick = ( portTickType ) -1;
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/*** Timer callback functions are defined below here. ***/
|
|
|
|
static void prvAutoReloadTimerCallback( xTimerHandle pxExpiredTimer )
|
|
{
|
|
portBASE_TYPE xTimerID;
|
|
|
|
xTimerID = ( portBASE_TYPE ) pvTimerGetTimerID( pxExpiredTimer );
|
|
if( xTimerID <= ( configTIMER_QUEUE_LENGTH + 1 ) )
|
|
{
|
|
( ucAutoReloadTimerCounters[ xTimerID ] )++;
|
|
}
|
|
else
|
|
{
|
|
/* The timer ID appears to be unexpected (invalid). */
|
|
xTestStatus = pdFAIL;
|
|
configASSERT( xTestStatus );
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvOneShotTimerCallback( xTimerHandle pxExpiredTimer )
|
|
{
|
|
/* The parameter is not used in this case as only one timer uses this
|
|
callback function. */
|
|
( void ) pxExpiredTimer;
|
|
|
|
ucOneShotTimerCounter++;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvISRAutoReloadTimerCallback( xTimerHandle pxExpiredTimer )
|
|
{
|
|
/* The parameter is not used in this case as only one timer uses this
|
|
callback function. */
|
|
( void ) pxExpiredTimer;
|
|
|
|
ucISRAutoReloadTimerCounter++;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
static void prvISROneShotTimerCallback( xTimerHandle pxExpiredTimer )
|
|
{
|
|
/* The parameter is not used in this case as only one timer uses this
|
|
callback function. */
|
|
( void ) pxExpiredTimer;
|
|
|
|
ucISROneShotTimerCounter++;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
|
|
|
|
|