162 lines
5.8 KiB
C
162 lines
5.8 KiB
C
|
/*
|
||
|
* FreeRTOS V202212.00
|
||
|
* Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
||
|
*
|
||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||
|
* this software and associated documentation files (the "Software"), to deal in
|
||
|
* the Software without restriction, including without limitation the rights to
|
||
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||
|
* the Software, and to permit persons to whom the Software is furnished to do so,
|
||
|
* subject to the following conditions:
|
||
|
*
|
||
|
* The above copyright notice and this permission notice shall be included in all
|
||
|
* copies or substantial portions of the Software.
|
||
|
*
|
||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||
|
*
|
||
|
* https://www.FreeRTOS.org
|
||
|
* https://github.com/FreeRTOS
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* Creates one or more tasks that repeatedly perform a set of integer
|
||
|
* calculations. The result of each run-time calculation is compared to the
|
||
|
* known expected result - with a mismatch being indicative of an error in the
|
||
|
* context switch mechanism.
|
||
|
*/
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
/* Scheduler include files. */
|
||
|
#include "FreeRTOS.h"
|
||
|
#include "task.h"
|
||
|
|
||
|
/* Demo program include files. */
|
||
|
#include "integer.h"
|
||
|
|
||
|
/* The constants used in the calculation. */
|
||
|
#define intgCONST1 ( ( long ) 123 )
|
||
|
#define intgCONST2 ( ( long ) 234567 )
|
||
|
#define intgCONST3 ( ( long ) -3 )
|
||
|
#define intgCONST4 ( ( long ) 7 )
|
||
|
#define intgEXPECTED_ANSWER ( ( ( intgCONST1 + intgCONST2 ) * intgCONST3 ) / intgCONST4 )
|
||
|
|
||
|
#define intgSTACK_SIZE configMINIMAL_STACK_SIZE
|
||
|
|
||
|
/* As this is the minimal version, we will only create one task. */
|
||
|
#define intgNUMBER_OF_TASKS ( 1 )
|
||
|
|
||
|
/* The task function. Repeatedly performs a 32 bit calculation, checking the
|
||
|
* result against the expected result. If the result is incorrect then the
|
||
|
* context switch must have caused some corruption. */
|
||
|
static portTASK_FUNCTION_PROTO( vCompeteingIntMathTask, pvParameters );
|
||
|
|
||
|
/* Variables that are set to true within the calculation task to indicate
|
||
|
* that the task is still executing. The check task sets the variable back to
|
||
|
* false, flagging an error if the variable is still false the next time it
|
||
|
* is called. */
|
||
|
static BaseType_t xTaskCheck[ intgNUMBER_OF_TASKS ] = { ( BaseType_t ) pdFALSE };
|
||
|
|
||
|
/*-----------------------------------------------------------*/
|
||
|
|
||
|
void vStartIntegerMathTasks( UBaseType_t uxPriority )
|
||
|
{
|
||
|
short sTask;
|
||
|
|
||
|
for( sTask = 0; sTask < intgNUMBER_OF_TASKS; sTask++ )
|
||
|
{
|
||
|
xTaskCreate( vCompeteingIntMathTask, "IntMath", intgSTACK_SIZE, ( void * ) &( xTaskCheck[ sTask ] ), uxPriority, ( TaskHandle_t * ) NULL );
|
||
|
}
|
||
|
}
|
||
|
/*-----------------------------------------------------------*/
|
||
|
|
||
|
static portTASK_FUNCTION( vCompeteingIntMathTask, pvParameters )
|
||
|
{
|
||
|
/* These variables are all effectively set to constants so they are volatile to
|
||
|
* ensure the compiler does not just get rid of them. */
|
||
|
volatile long lValue;
|
||
|
short sError = pdFALSE;
|
||
|
volatile BaseType_t * pxTaskHasExecuted;
|
||
|
|
||
|
/* Set a pointer to the variable we are going to set to true each
|
||
|
* iteration. This is also a good test of the parameter passing mechanism
|
||
|
* within each port. */
|
||
|
pxTaskHasExecuted = ( volatile BaseType_t * ) pvParameters;
|
||
|
|
||
|
/* Keep performing a calculation and checking the result against a constant. */
|
||
|
for( ; ; )
|
||
|
{
|
||
|
/* Perform the calculation. This will store partial value in
|
||
|
* registers, resulting in a good test of the context switch mechanism. */
|
||
|
lValue = intgCONST1;
|
||
|
lValue += intgCONST2;
|
||
|
|
||
|
/* Yield in case cooperative scheduling is being used. */
|
||
|
#if configUSE_PREEMPTION == 0
|
||
|
{
|
||
|
taskYIELD();
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/* Finish off the calculation. */
|
||
|
lValue *= intgCONST3;
|
||
|
lValue /= intgCONST4;
|
||
|
|
||
|
/* If the calculation is found to be incorrect we stop setting the
|
||
|
* TaskHasExecuted variable so the check task can see an error has
|
||
|
* occurred. */
|
||
|
if( lValue != intgEXPECTED_ANSWER ) /*lint !e774 volatile used to prevent this being optimised out. */
|
||
|
{
|
||
|
sError = pdTRUE;
|
||
|
}
|
||
|
|
||
|
if( sError == pdFALSE )
|
||
|
{
|
||
|
/* We have not encountered any errors, so set the flag that show
|
||
|
* we are still executing. This will be periodically cleared by
|
||
|
* the check task. */
|
||
|
portENTER_CRITICAL();
|
||
|
*pxTaskHasExecuted = pdTRUE;
|
||
|
portEXIT_CRITICAL();
|
||
|
}
|
||
|
|
||
|
/* Yield in case cooperative scheduling is being used. */
|
||
|
#if configUSE_PREEMPTION == 0
|
||
|
{
|
||
|
taskYIELD();
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
}
|
||
|
/*-----------------------------------------------------------*/
|
||
|
|
||
|
/* This is called to check that all the created tasks are still running. */
|
||
|
BaseType_t xAreIntegerMathsTaskStillRunning( void )
|
||
|
{
|
||
|
BaseType_t xReturn = pdTRUE;
|
||
|
short sTask;
|
||
|
|
||
|
/* Check the maths tasks are still running by ensuring their check variables
|
||
|
* are still being set to true. */
|
||
|
for( sTask = 0; sTask < intgNUMBER_OF_TASKS; sTask++ )
|
||
|
{
|
||
|
if( xTaskCheck[ sTask ] == pdFALSE )
|
||
|
{
|
||
|
/* The check has not incremented so an error exists. */
|
||
|
xReturn = pdFALSE;
|
||
|
}
|
||
|
|
||
|
/* Reset the check variable so we can tell if it has been set by
|
||
|
* the next time around. */
|
||
|
xTaskCheck[ sTask ] = pdFALSE;
|
||
|
}
|
||
|
|
||
|
return xReturn;
|
||
|
}
|