/* (c) 2018 Microchip Technology Inc. and its subsidiaries. Subject to your compliance with these terms, you may use Microchip software and any derivatives exclusively with Microchip products. It is your responsibility to comply with third party license terms applicable to your use of third party software (including open source software) that may accompany Microchip software. THIS SOFTWARE IS SUPPLIED BY MICROCHIP "AS IS". NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, APPLY TO THIS SOFTWARE, INCLUDING ANY IMPLIED WARRANTIES OF NON-INFRINGEMENT, MERCHANTABILITY, AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT WILL MICROCHIP BE LIABLE FOR ANY INDIRECT, SPECIAL, PUNITIVE, INCIDENTAL OR CONSEQUENTIAL LOSS, DAMAGE, COST OR EXPENSE OF ANY KIND WHATSOEVER RELATED TO THE SOFTWARE, HOWEVER CAUSED, EVEN IF MICROCHIP HAS BEEN ADVISED OF THE POSSIBILITY OR THE DAMAGES ARE FORESEEABLE. TO THE FULLEST EXTENT ALLOWED BY LAW, MICROCHIP'S TOTAL LIABILITY ON ALL CLAIMS IN ANY WAY RELATED TO THIS SOFTWARE WILL NOT EXCEED THE AMOUNT OF FEES, IF ANY, THAT YOU HAVE PAID DIRECTLY TO MICROCHIP FOR THIS SOFTWARE. */ #include /* Scheduler include files. */ #include "FreeRTOS.h" #if ( mainSELECTED_APPLICATION == MINIMAL_DEMO ) #include "task.h" #include "croutine.h" #include "PollQ.h" #include "integer.h" #include "serial.h" #include "comtest.h" #include "crflash.h" #include "partest.h" #include "regtest.h" #include "serial/usart.h" /* Priority definitions for most of the tasks in the demo application. Some tasks just use the idle priority. */ #define mainCOM_TEST_PRIORITY ( tskIDLE_PRIORITY + 2 ) #define mainQUEUE_POLL_PRIORITY ( tskIDLE_PRIORITY + 2 ) #define mainCHECK_TASK_PRIORITY ( tskIDLE_PRIORITY + 3 ) /* Baud rate used by the MINIMAL demo. */ #define mainCOM_TEST_BAUD_RATE ( ( unsigned long ) 9600 ) /* LED used by the serial port tasks. This is toggled on each character Tx, and mainCOM_TEST_LED + 1 is toggles on each character Rx. */ #define mainCOM_TEST_LED ( 7 ) /* LED that is toggled by the check task. The check task periodically checks that all the other tasks are operating without error. If no errors are found the LED is toggled. If an error is found at any time the LED is never toggles again. */ #define mainCHECK_TASK_LED ( 6 ) /* The period between executions of the check task. */ #define mainCHECK_PERIOD ( ( TickType_t ) 1000 / portTICK_PERIOD_MS ) /* An address in the EEPROM used to count resets. This is used to check that the demo application is not unexpectedly resetting. */ #define mainRESET_COUNT_ADDRESS ( 0x1400 ) /* The number of coroutines to create. */ #define mainNUM_FLASH_COROUTINES ( 3 ) /* The task function for the "Check" task. */ static void vErrorChecks( void *pvParameters ); /* Checks the unique counts of other tasks to ensure they are still operational. Flashes an LED if everything is okay. */ static void prvCheckOtherTasksAreStillRunning( void ); /* Called on boot to increment a count stored in the EEPROM. This is used to ensure the CPU does not reset unexpectedly. */ static void prvIncrementResetCount( void ); void main_minimal( void ) { prvIncrementResetCount(); /* Create the standard demo tasks. */ vStartIntegerMathTasks( tskIDLE_PRIORITY ); vAltStartComTestTasks( mainCOM_TEST_PRIORITY, mainCOM_TEST_BAUD_RATE, mainCOM_TEST_LED ); vStartPolledQueueTasks( mainQUEUE_POLL_PRIORITY ); vStartRegTestTasks(); /* Create the tasks defined within this file. */ xTaskCreate( vErrorChecks, "Check", configMINIMAL_STACK_SIZE, NULL, mainCHECK_TASK_PRIORITY, NULL ); /* Create the co-routines that flash the LED's. */ vStartFlashCoRoutines( mainNUM_FLASH_COROUTINES ); /* In this port, to use preemptive scheduler define configUSE_PREEMPTION as 1 in portmacro.h. To use the cooperative scheduler define configUSE_PREEMPTION as 0. */ vTaskStartScheduler(); } /*-----------------------------------------------------------*/ void init_minimal( void ) { USART_cfg_t cfg; USART_initConfigs(&cfg); cfg.BAUD = (uint16_t)USART_BAUD_RATE(mainCOM_TEST_BAUD_RATE); cfg.CTRLA = 1 << USART_LBME_bp | 1 << USART_RXCIE_bp; cfg.CTRLB = 1 << USART_RXEN_bp | 1 << USART_TXEN_bp; cfg.CTRLC = 0x03; USART_setConfigs(cfg); vParTestInitialise(); } /*-----------------------------------------------------------*/ static void vErrorChecks( void *pvParameters ) { static volatile unsigned long ulDummyVariable = 3UL; /* The parameters are not used. */ ( void ) pvParameters; /* Cycle for ever, delaying then checking all the other tasks are still operating without error. */ for( ;; ) { vTaskDelay( mainCHECK_PERIOD ); /* Perform a bit of 32bit maths to ensure the registers used by the integer tasks get some exercise. The result here is not important - see the demo application documentation for more info. */ ulDummyVariable *= 3; prvCheckOtherTasksAreStillRunning(); } } /*-----------------------------------------------------------*/ static void prvCheckOtherTasksAreStillRunning( void ) { static portBASE_TYPE xErrorHasOccurred = pdFALSE; if( xAreIntegerMathsTaskStillRunning() != pdTRUE ) { xErrorHasOccurred = pdTRUE; } if( xAreComTestTasksStillRunning() != pdTRUE ) { xErrorHasOccurred = pdTRUE; } if( xArePollingQueuesStillRunning() != pdTRUE ) { xErrorHasOccurred = pdTRUE; } if( xAreRegTestTasksStillRunning() != pdTRUE ) { xErrorHasOccurred = pdTRUE; } if( xErrorHasOccurred == pdFALSE ) { /* Toggle the LED if everything is okay so we know if an error occurs even if not using console IO. */ vParTestToggleLED( mainCHECK_TASK_LED ); } } /*-----------------------------------------------------------*/ static void prvIncrementResetCount( void ) { unsigned char ucResetCount; eeprom_read_block( &ucResetCount, ( void * ) mainRESET_COUNT_ADDRESS, sizeof( ucResetCount ) ); ucResetCount++; eeprom_write_byte( ( void * ) mainRESET_COUNT_ADDRESS, ucResetCount ); } /*-----------------------------------------------------------*/ void vApplicationIdleHook( void ) { vCoRoutineSchedule(); } #endif