3 * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved.
5 * Permission is hereby granted, free of charge, to any person obtaining a copy of
6 * this software and associated documentation files (the "Software"), to deal in
7 * the Software without restriction, including without limitation the rights to
8 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
9 * the Software, and to permit persons to whom the Software is furnished to do so,
10 * subject to the following conditions:
12 * The above copyright notice and this permission notice shall be included in all
13 * copies or substantial portions of the Software.
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
17 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
18 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
19 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
20 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
22 * http://www.FreeRTOS.org
23 * http://aws.amazon.com/freertos
29 * Creates the demo application tasks, then starts the scheduler. The WEB
30 * documentation provides more details of the demo application tasks.
32 * Main. c also creates four other tasks:
35 * This only executes every few seconds but has the highest priority so is
36 * guaranteed to get processor time. Its main function is to check that all
37 * the standard demo application tasks are still operational and have not
38 * experienced any errors. vErrorChecks() will toggle the on board LED
39 * every mainNO_ERROR_FLASH_PERIOD milliseconds if none of the demo application
40 * tasks have reported an error. Should any task report an error at any time
41 * the rate at which the on board LED is toggled is increased to
42 * mainERROR_FLASH_PERIOD - providing visual feedback that something has gone
46 * This is a very simple task that checks that all the registers are always
47 * in their expected state. The task only makes use of the A register, so
48 * all the other registers should always contain their initial values.
49 * An incorrect value indicates an error in the context switch mechanism.
50 * The task operates at the idle priority so will be preempted regularly.
51 * Any error will cause the toggle rate of the on board LED to increase to
52 * mainERROR_FLASH_PERIOD milliseconds.
54 * 3 and 4) vFLOPCheck1() and vFLOPCheck2()
55 * These are very basic versions of the standard FLOP tasks. They are good
56 * at detecting errors in the context switch mechanism, and also check that
57 * the floating point libraries are correctly built to be re-enterant. The
58 * stack restrictions of the 8051 prevent the use of the standard FLOP demo
62 /* Standard includes. */
65 /* Scheduler includes. */
69 /* Demo application includes. */
77 /* Demo task priorities. */
78 #define mainLED_TASK_PRIORITY ( tskIDLE_PRIORITY + 1 )
79 #define mainQUEUE_POLL_PRIORITY ( tskIDLE_PRIORITY + 2 )
80 #define mainCOM_TEST_PRIORITY ( tskIDLE_PRIORITY + 2 )
81 #define mainCHECK_TASK_PRIORITY ( tskIDLE_PRIORITY + 3 )
82 #define mainSEM_TEST_PRIORITY ( tskIDLE_PRIORITY + 2 )
83 #define mainINTEGER_PRIORITY tskIDLE_PRIORITY
85 /* Constants required to disable the watchdog. */
86 #define mainDISABLE_BYTE_1 ( ( unsigned char ) 0xde )
87 #define mainDISABLE_BYTE_2 ( ( unsigned char ) 0xad )
89 /* Constants to setup and use the on board LED. */
90 #define ucLED_BIT ( ( unsigned char ) 0x40 )
91 #define mainPORT_1_BIT_6 ( ( unsigned char ) 0x40 )
92 #define mainENABLE_CROSS_BAR ( ( unsigned char ) 0x40 )
94 /* Constants to set the clock frequency. */
95 #define mainSELECT_INTERNAL_OSC ( ( unsigned char ) 0x80 )
96 #define mainDIVIDE_CLOCK_BY_1 ( ( unsigned char ) 0x03 )
97 #define mainPLL_USES_INTERNAL_OSC ( ( unsigned char ) 0x04 )
98 #define mainFLASH_READ_TIMING ( ( unsigned char ) 0x30 )
99 #define mainPLL_POWER_ON ( ( unsigned char ) 0x01 )
100 #define mainPLL_NO_PREDIVIDE ( ( unsigned char ) 0x01 )
101 #define mainPLL_FILTER ( ( unsigned char ) 0x01 )
102 #define mainPLL_MULTIPLICATION ( ( unsigned char ) 0x04 )
103 #define mainENABLE_PLL ( ( unsigned char ) 0x02 )
104 #define mainPLL_LOCKED ( ( unsigned char ) 0x10 )
105 #define mainSELECT_PLL_AS_SOURCE ( ( unsigned char ) 0x02 )
107 /* Toggle rate for the on board LED - which is dependent on whether or not
108 an error has been detected. */
109 #define mainNO_ERROR_FLASH_PERIOD ( ( TickType_t ) 5000 )
110 #define mainERROR_FLASH_PERIOD ( ( TickType_t ) 250 )
112 /* Baud rate used by the serial port tasks. */
113 #define mainCOM_TEST_BAUD_RATE ( ( unsigned long ) 115200 )
115 /* Pass an invalid LED number to the COM test task as we don't want it to flash
116 an LED. There are only 8 LEDs (excluding the on board LED) wired in and these
117 are all used by the flash tasks. */
118 #define mainCOM_TEST_LED ( 200 )
120 /* We want the Cygnal to act as much as possible as a standard 8052. */
121 #define mainAUTO_SFR_OFF ( ( unsigned char ) 0 )
123 /* Constants required to setup the IO pins for serial comms. */
124 #define mainENABLE_COMS ( ( unsigned char ) 0x04 )
125 #define mainCOMS_LINES_TO_PUSH_PULL ( ( unsigned char ) 0x03 )
127 /* Pointer passed as a parameter to vRegisterCheck() just so it has some know
128 values to check for in the DPH, DPL and B registers. */
129 #define mainDUMMY_POINTER ( ( xdata void * ) 0xabcd )
131 /* Macro that lets vErrorChecks() know that one of the tasks defined in
132 main. c has detected an error. A critical region is used around xLatchError
133 as it is accessed from vErrorChecks(), which has a higher priority. */
134 #define mainLATCH_ERROR() \
136 portENTER_CRITICAL(); \
137 xLatchedError = pdTRUE; \
138 portEXIT_CRITICAL(); \
142 * Setup the Cygnal microcontroller for its fastest operation.
144 static void prvSetupSystemClock( void );
147 * Setup the peripherals, including the on board LED.
149 static void prvSetupHardware( void );
152 * Toggle the state of the on board LED.
154 static void prvToggleOnBoardLED( void );
157 * See comments at the top of the file for details.
159 static void vErrorChecks( void *pvParameters );
162 * See comments at the top of the file for details.
164 static void vRegisterCheck( void *pvParameters );
167 * See comments at the top of the file for details.
169 static void vFLOPCheck1( void *pvParameters );
172 * See comments at the top of the file for details.
174 static void vFLOPCheck2( void *pvParameters );
176 /* File scope variable used to communicate the occurrence of an error between
178 static portBASE_TYPE xLatchedError = pdFALSE;
180 /*-----------------------------------------------------------*/
183 * Starts all the other tasks, then starts the scheduler.
187 /* Initialise the hardware including the system clock and on board
191 /* Initialise the port that controls the external LED's utilized by the
193 vParTestInitialise();
195 /* Start the used standard demo tasks. */
196 vStartLEDFlashTasks( mainLED_TASK_PRIORITY );
197 vStartPolledQueueTasks( mainQUEUE_POLL_PRIORITY );
198 vStartIntegerMathTasks( mainINTEGER_PRIORITY );
199 vAltStartComTestTasks( mainCOM_TEST_PRIORITY, mainCOM_TEST_BAUD_RATE, mainCOM_TEST_LED );
200 vStartSemaphoreTasks( mainSEM_TEST_PRIORITY );
202 /* Start the tasks defined in this file. The first three never block so
203 must not be used with the co-operative scheduler. */
204 #if configUSE_PREEMPTION == 1
206 xTaskCreate( vRegisterCheck, "RegChck", configMINIMAL_STACK_SIZE, mainDUMMY_POINTER, tskIDLE_PRIORITY, ( TaskHandle_t * ) NULL );
207 xTaskCreate( vFLOPCheck1, "FLOP", configMINIMAL_STACK_SIZE, NULL, tskIDLE_PRIORITY, ( TaskHandle_t * ) NULL );
208 xTaskCreate( vFLOPCheck2, "FLOP", configMINIMAL_STACK_SIZE, NULL, tskIDLE_PRIORITY, ( TaskHandle_t * ) NULL );
212 xTaskCreate( vErrorChecks, "Check", configMINIMAL_STACK_SIZE, NULL, mainCHECK_TASK_PRIORITY, ( TaskHandle_t * ) NULL );
214 /* Finally kick off the scheduler. This function should never return. */
215 vTaskStartScheduler();
217 /* Should never reach here as the tasks will now be executing under control
220 /*-----------------------------------------------------------*/
223 * Setup the hardware prior to using the scheduler. Most of the Cygnal
224 * specific initialisation is performed here leaving standard 8052 setup
225 * only in the driver code.
227 static void prvSetupHardware( void )
229 unsigned char ucOriginalSFRPage;
231 /* Remember the SFR page before it is changed so it can get set back
232 before the function exits. */
233 ucOriginalSFRPage = SFRPAGE;
235 /* Setup the SFR page to access the config SFR's. */
236 SFRPAGE = CONFIG_PAGE;
238 /* Don't allow the microcontroller to automatically switch SFR page, as the
239 SFR page is not stored as part of the task context. */
240 SFRPGCN = mainAUTO_SFR_OFF;
242 /* Disable the watchdog. */
243 WDTCN = mainDISABLE_BYTE_1;
244 WDTCN = mainDISABLE_BYTE_2;
246 /* Set the on board LED to push pull. */
247 P1MDOUT |= mainPORT_1_BIT_6;
249 /* Setup the cross bar to enable serial comms here as it is not part of the
250 standard 8051 setup and therefore is not in the driver code. */
251 XBR0 |= mainENABLE_COMS;
252 P0MDOUT |= mainCOMS_LINES_TO_PUSH_PULL;
254 /* Enable the cross bar so our hardware setup takes effect. */
255 XBR2 = mainENABLE_CROSS_BAR;
257 /* Setup a fast system clock. */
258 prvSetupSystemClock();
260 /* Return the SFR page. */
261 SFRPAGE = ucOriginalSFRPage;
263 /*-----------------------------------------------------------*/
265 static void prvSetupSystemClock( void )
267 volatile unsigned short usWait;
268 const unsigned short usWaitTime = ( unsigned short ) 0x2ff;
269 unsigned char ucOriginalSFRPage;
271 /* Remember the SFR page so we can set it back at the end. */
272 ucOriginalSFRPage = SFRPAGE;
273 SFRPAGE = CONFIG_PAGE;
275 /* Use the internal oscillator set to its fasted frequency. */
276 OSCICN = mainSELECT_INTERNAL_OSC | mainDIVIDE_CLOCK_BY_1;
278 /* Ensure the clock is stable. */
279 for( usWait = 0; usWait < usWaitTime; usWait++ );
281 /* Setup the clock source for the PLL. */
282 PLL0CN &= ~mainPLL_USES_INTERNAL_OSC;
284 /* Change the read timing for the flash ready for the fast clock. */
285 SFRPAGE = LEGACY_PAGE;
286 FLSCL |= mainFLASH_READ_TIMING;
288 /* Turn on the PLL power. */
289 SFRPAGE = CONFIG_PAGE;
290 PLL0CN |= mainPLL_POWER_ON;
292 /* Don't predivide the clock. */
293 PLL0DIV = mainPLL_NO_PREDIVIDE;
295 /* Set filter for fastest clock. */
296 PLL0FLT = mainPLL_FILTER;
297 PLL0MUL = mainPLL_MULTIPLICATION;
299 /* Ensure the clock is stable. */
300 for( usWait = 0; usWait < usWaitTime; usWait++ );
302 /* Enable the PLL and wait for it to lock. */
303 PLL0CN |= mainENABLE_PLL;
304 for( usWait = 0; usWait < usWaitTime; usWait++ )
306 if( PLL0CN & mainPLL_LOCKED )
312 /* Select the PLL as the clock source. */
313 CLKSEL |= mainSELECT_PLL_AS_SOURCE;
315 /* Return the SFR back to its original value. */
316 SFRPAGE = ucOriginalSFRPage;
318 /*-----------------------------------------------------------*/
320 static void prvToggleOnBoardLED( void )
322 /* If the on board LED is on, turn it off and vice versa. */
332 /*-----------------------------------------------------------*/
335 * See the documentation at the top of this file.
337 static void vErrorChecks( void *pvParameters )
339 portBASE_TYPE xErrorHasOccurred = pdFALSE;
341 /* Just to prevent compiler warnings. */
342 ( void ) pvParameters;
344 /* Cycle for ever, delaying then checking all the other tasks are still
345 operating without error. The delay period depends on whether an error
346 has ever been detected. */
349 if( xLatchedError == pdFALSE )
351 /* No errors have been detected so delay for a longer period. The
352 on board LED will get toggled every mainNO_ERROR_FLASH_PERIOD ms. */
353 vTaskDelay( mainNO_ERROR_FLASH_PERIOD );
357 /* We have at some time recognised an error in one of the demo
358 application tasks, delay for a shorter period. The on board LED
359 will get toggled every mainERROR_FLASH_PERIOD ms. */
360 vTaskDelay( mainERROR_FLASH_PERIOD );
365 /* Check the demo application tasks for errors. */
367 if( xAreIntegerMathsTaskStillRunning() != pdTRUE )
369 xErrorHasOccurred = pdTRUE;
372 if( xArePollingQueuesStillRunning() != pdTRUE )
374 xErrorHasOccurred = pdTRUE;
377 if( xAreComTestTasksStillRunning() != pdTRUE )
379 xErrorHasOccurred = pdTRUE;
382 if( xAreSemaphoreTasksStillRunning() != pdTRUE )
384 xErrorHasOccurred = pdTRUE;
387 /* If an error has occurred, latch it to cause the LED flash rate to
389 if( xErrorHasOccurred == pdTRUE )
391 xLatchedError = pdTRUE;
394 /* Toggle the LED to indicate the completion of a check cycle. The
395 frequency of check cycles is dependent on whether or not we have
397 prvToggleOnBoardLED();
400 /*-----------------------------------------------------------*/
403 * See the documentation at the top of this file. Also see the standard FLOP
404 * demo task documentation for the rationale of these tasks.
406 static void vFLOPCheck1( void *pvParameters )
408 volatile portFLOAT fVal1, fVal2, fResult;
410 ( void ) pvParameters;
414 fVal1 = ( portFLOAT ) -1234.5678;
415 fVal2 = ( portFLOAT ) 2345.6789;
417 fResult = fVal1 + fVal2;
418 if( ( fResult > ( portFLOAT ) 1111.15 ) || ( fResult < ( portFLOAT ) 1111.05 ) )
423 fResult = fVal1 / fVal2;
424 if( ( fResult > ( portFLOAT ) -0.51 ) || ( fResult < ( portFLOAT ) -0.53 ) )
430 /*-----------------------------------------------------------*/
433 * See the documentation at the top of this file.
435 static void vFLOPCheck2( void *pvParameters )
437 volatile portFLOAT fVal1, fVal2, fResult;
439 ( void ) pvParameters;
443 fVal1 = ( portFLOAT ) -12340.5678;
444 fVal2 = ( portFLOAT ) 23450.6789;
446 fResult = fVal1 + fVal2;
447 if( ( fResult > ( portFLOAT ) 11110.15 ) || ( fResult < ( portFLOAT ) 11110.05 ) )
452 fResult = fVal1 / -fVal2;
453 if( ( fResult > ( portFLOAT ) 0.53 ) || ( fResult < ( portFLOAT ) 0.51 ) )
459 /*-----------------------------------------------------------*/
462 * See the documentation at the top of this file.
464 static void vRegisterCheck( void *pvParameters )
466 ( void ) pvParameters;
470 if( SP != configSTACK_START )