FreeRTOS-Kernel/FreeRTOS/Demo/RX100-RSK_IAR/main_low_power.c
Rahul Kar 121fbe295b
Fix formatting in kernel demo application files (#1148)
* Fix formatting in kernel demo application files

* Fix header check fail in the demo files

* Add ignored patterns in core header check file

* Fix formatting

* Update vApplicationStackOverflowHook for AVR_ATMega4809_MPLAB.X/main.c

Co-authored-by: Soren Ptak <ptaksoren@gmail.com>

* Update vApplicationStackOverflowHook for AVR_ATMega4809_MPLAB.X/main.c

Co-authored-by: Soren Ptak <ptaksoren@gmail.com>

* Update vApplicationStackOverflowHook for AVR_Dx_IAR/main.c

Co-authored-by: Soren Ptak <ptaksoren@gmail.com>

* Update vApplicationStackOverflowHook for AVR_Dx_IAR/main.c

Co-authored-by: Soren Ptak <ptaksoren@gmail.com>

* Update vApplicationStackOverflowHook for AVR_Dx_MPLAB.X/main.c

Co-authored-by: Soren Ptak <ptaksoren@gmail.com>

* Update vApplicationMallocFailedHook for AVR_Dx_MPLAB.X/main.c

Co-authored-by: Soren Ptak <ptaksoren@gmail.com>

* Fix formatting AVR32_UC3

---------

Co-authored-by: Soren Ptak <ptaksoren@gmail.com>
2024-01-02 11:05:59 +05:30

391 lines
16 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
*
*/
/* ****************************************************************************
* When configCREATE_LOW_POWER_DEMO is set to 1 in FreeRTOSConfig.h main() will
* call main_low_power(), which is defined in this file. main_low_power()
* demonstrates FreeRTOS tick suppression being used to allow the MCU to be
* placed into both the low power deep sleep mode and the low power software
* standby mode. When configCREATE_LOW_POWER_DEMO is set to 0 main will
* instead call main_full(), which is a more comprehensive RTOS demonstration.
* ****************************************************************************
*
* This application demonstrates the FreeRTOS tickless idle mode (tick
* suppression). See http://www.freertos.org/low-power-tickless-rtos.html
* The demo is configured to execute on the Renesas RX100 RSK.
*
* Functionality:
*
* + Two tasks are created, an Rx task and a Tx task.
*
* + The Rx task repeatedly blocks on a queue to wait for data. The Rx task
* toggles LED 0 each time is receives a value from the queue.
*
* + The Tx task repeatedly enters the Blocked state for an amount of time
* that is set by the position of the potentiometer. On exiting the blocked
* state the Tx task sends a value through the queue to the Rx task (causing
* the Rx task to exit the blocked state and toggle LED 0).
*
* If the value read from the potentiometer is less than or equal to
* mainSOFTWARE_STANDBY_DELAY then the Tx task blocks for the equivalent
* number of milliseconds. For example, if the sampled analog value is
* 2000, then the Tx task blocks for 2000ms. Blocking for a finite period
* allows the kernel to stop the tick interrupt and place the RX100 into
* deep sleep mode.
*
* If the value read form the potentiometer is greater than
* mainSOFTWARE_STANDBY_DELAY then the Tx task blocks on a semaphore with
* an infinite timeout. Blocking with an infinite timeout allows the kernel
* to stop the tick interrupt and place the RX100 into software standby
* mode. Pressing a button will generate an interrupt that causes the RX100
* to exit software standby mode. The interrupt service routine 'gives' the
* semaphore to unblock the Tx task.
*
*
* Using the Demo and Observed Behaviour:
*
* 1) Turn the potentiometer completely counter clockwise.
*
* 2) Program the RX100 with the application, then disconnect the programming/
* debugging hardware to ensure power readings are not effected by any
* connected interfaces.
*
* 3) Start the application running. LED 0 will toggle quickly because the
* potentiometer is turned to its lowest value. LED 1 will be illuminated
* when the RX100 is not in a power saving mode, but will appear to be off
* because most execution time is spent in a sleep mode. Led 2 will be
* illuminated when the RX100 is in deep sleep mode, and will appear to be
* always on, again because most execution time is spent in deep sleep mode.
* The LEDs are turned on and off by the application defined pre and post
* sleep macros (see the definitions of configPRE_SLEEP_PROCESSING() and
* configPOST_SLEEP_PROCESSING() in FreeRTOSConfig.h).
*
* 4) Slowly turn the potentiometer in the clockwise direction. This will
* increase the value read from the potentiometer, which will increase the
* time the Tx task spends in the Blocked state, which will therefore
* decrease the frequency at which the Tx task sends data to the queue (and
* the rate at which LED 0 is toggled).
*
* 5) Keep turning the potentiometer in the clockwise direction. Eventually
* the value read from the potentiometer will go above
* mainSOFTWARE_STANDBY_DELAY, causing the Tx task to block on the semaphore
* with an infinite timeout. LED 0 will stop toggling because the Tx task is
* no longer sending to the queue. LED 1 and LED 2 will both be off because
* the RX100 is neither running or in deep sleep mode (it is in software
* standby mode).
*
* 6) Turn the potentiometer counter clockwise again to ensure its value goes
* back below mainSOFTWARE_STANDBY_DELAY.
*
* 7) Press any of the three buttons to generate an interrupt. The interrupt
* will take the RX100 out of software standby mode, and the interrupt
* service routine will unblock the Tx task by 'giving' the semaphore. LED 0
* will then start to toggle again.
*
*/
/* Hardware specific includes. */
#include "platform.h"
#include "r_switches_if.h"
/* Kernel includes. */
#include "FreeRTOS.h"
#include "task.h"
#include "queue.h"
#include "semphr.h"
/* Common demo includes. */
#include "partest.h"
/* Priorities at which the Rx and Tx tasks are created. */
#define configQUEUE_RECEIVE_TASK_PRIORITY ( tskIDLE_PRIORITY + 1 )
#define configQUEUE_SEND_TASK_PRIORITY ( tskIDLE_PRIORITY + 2 )
/* The number of items the queue can hold. This is 1 as the Rx task will
* remove items as they are added so the Tx task should always find the queue
* empty. */
#define mainQUEUE_LENGTH ( 1 )
/* The LED used to indicate that a value has been received on the queue. */
#define mainQUEUE_LED ( 0 )
/* The LED used to indicate that full power is being used (the MCU is not in
* deep sleep or software standby mode). */
#define mainFULL_POWER_LED ( 1 )
/* The LED used to indicate that deep sleep mode is being used. */
#define mainDEEP_SLEEP_LED ( 2 )
/* The Tx task sends to the queue with a frequency that is set by the value
* read from the potentiometer until the value goes above that set by the
* mainSOFTWARE_STANDBY_DELAY constant - at which time the Tx task instead blocks
* indefinitely on a semaphore. */
#define mainSOFTWARE_STANDBY_DELAY ( 3000UL )
/* A block time of zero simply means "don't block". */
#define mainDONT_BLOCK ( 0 )
/* The value that is sent from the Tx task to the Rx task on the queue. */
#define mainQUEUED_VALUE ( 100UL )
/*-----------------------------------------------------------*/
/*
* The Rx and Tx tasks as described at the top of this file.
*/
static void prvQueueReceiveTask( void * pvParameters );
static void prvQueueSendTask( void * pvParameters );
/*
* Reads and returns the value of the ADC connected to the potentiometer built
* onto the RSK.
*/
static unsigned short prvReadPOT( void );
/*
* The handler for the interrupt generated when any of the buttons are pressed.
*/
__interrupt void vButtonInterrupt( void );
/*-----------------------------------------------------------*/
/* The queue to pass data from the Tx task to the Rx task. */
static QueueHandle_t xQueue = NULL;
/* The semaphore that is 'given' by interrupts generated from button pushes. */
static SemaphoreHandle_t xSemaphore = NULL;
/*-----------------------------------------------------------*/
void main_low_power( void )
{
/* Create the queue. */
xQueue = xQueueCreate( mainQUEUE_LENGTH, sizeof( unsigned long ) );
configASSERT( xQueue );
/* Create the semaphore that is 'given' by an interrupt generated from a
* button push. */
vSemaphoreCreateBinary( xSemaphore );
configASSERT( xSemaphore );
/* Make sure the semaphore starts in the expected state - no button pushes
* have yet occurred. A block time of zero can be used as it is guaranteed
* that the semaphore will be available because it has just been created. */
xSemaphoreTake( xSemaphore, mainDONT_BLOCK );
/* Start the two tasks as described at the top of this file. */
xTaskCreate( prvQueueReceiveTask, "Rx", configMINIMAL_STACK_SIZE, NULL, configQUEUE_RECEIVE_TASK_PRIORITY, NULL );
xTaskCreate( prvQueueSendTask, "TX", configMINIMAL_STACK_SIZE, NULL, configQUEUE_SEND_TASK_PRIORITY, NULL );
/* The CPU is currently running, not sleeping, so turn on the LED that
* shows the CPU is not in a sleep mode. */
vParTestSetLED( mainFULL_POWER_LED, pdTRUE );
/* Start the scheduler running running. */
vTaskStartScheduler();
/* If all is well the next line of code will not be reached as the
* scheduler will be running. If the next line is reached then it is likely
* there was insufficient FreeRTOS heap available for the idle task and/or
* timer task to be created. See http://www.freertos.org/a00111.html. */
for( ; ; )
{
}
}
/*-----------------------------------------------------------*/
static void prvQueueSendTask( void * pvParameters )
{
TickType_t xDelay;
const unsigned long ulValueToSend = mainQUEUED_VALUE;
/* Remove compiler warning about unused parameter. */
( void ) pvParameters;
for( ; ; )
{
/* The delay period between successive sends to the queue is set by
* the potentiometer reading. */
xDelay = ( TickType_t ) prvReadPOT();
/* If the block time is greater than 3000 milliseconds then block
* indefinitely waiting for a button push. */
if( xDelay > mainSOFTWARE_STANDBY_DELAY )
{
/* As this is an indefinite delay the kernel will place the CPU
* into software standby mode the next time the idle task runs. */
xSemaphoreTake( xSemaphore, portMAX_DELAY );
}
else
{
/* Convert a time in milliseconds to a time in ticks. */
xDelay /= portTICK_PERIOD_MS;
/* Place this task in the blocked state until it is time to run
* again. As this is not an indefinite sleep the kernel will place
* the CPU into the deep sleep state when the idle task next runs. */
vTaskDelay( xDelay );
}
/* Send to the queue - causing the queue receive task to flash its LED.
* It should not be necessary to block on the queue send because the Rx
* task will have removed the last queued item. */
xQueueSend( xQueue, &ulValueToSend, mainDONT_BLOCK );
}
}
/*-----------------------------------------------------------*/
static void prvQueueReceiveTask( void * pvParameters )
{
unsigned long ulReceivedValue;
/* Remove compiler warning about unused parameter. */
( void ) pvParameters;
for( ; ; )
{
/* Wait until something arrives in the queue - this will block
* indefinitely provided INCLUDE_vTaskSuspend is set to 1 in
* FreeRTOSConfig.h. */
xQueueReceive( xQueue, &ulReceivedValue, portMAX_DELAY );
/* To get here something must have arrived, but is it the expected
* value? If it is, toggle the LED. */
if( ulReceivedValue == mainQUEUED_VALUE )
{
vParTestToggleLED( mainQUEUE_LED );
}
}
}
/*-----------------------------------------------------------*/
void vPreSleepProcessing( unsigned long ulExpectedIdleTime )
{
/* Called by the kernel before it places the MCU into a sleep mode because
* configPRE_SLEEP_PROCESSING() is #defined to vPreSleepProcessing().
*
* NOTE: Additional actions can be taken here to get the power consumption
* even lower. For example, the ADC input used by this demo could be turned
* off here, and then back on again in the post sleep processing function.
* For maximum power saving ensure all unused pins are in their lowest power
* state. */
/* Avoid compiler warnings about the unused parameter. */
( void ) ulExpectedIdleTime;
/* Is the MCU about to enter deep sleep mode or software standby mode? */
if( SYSTEM.SBYCR.BIT.SSBY == 0 )
{
/* Turn on the LED that indicates deep sleep mode is being entered. */
vParTestSetLED( mainDEEP_SLEEP_LED, pdTRUE );
}
else
{
/* Software standby mode is being used, so no LEDs are illuminated to
* ensure minimum power readings are obtained. Ensure the Queue LED is
* also off. */
vParTestSetLED( mainQUEUE_LED, pdFALSE );
}
/* Turn off the LED that indicates full power is being used. */
vParTestSetLED( mainFULL_POWER_LED, pdFALSE );
}
/*-----------------------------------------------------------*/
void vPostSleepProcessing( unsigned long ulExpectedIdleTime )
{
/* Called by the kernel when the MCU exits a sleep mode because
* configPOST_SLEEP_PROCESSING is #defined to vPostSleepProcessing(). */
/* Avoid compiler warnings about the unused parameter. */
( void ) ulExpectedIdleTime;
/* Turn off the LED that indicates deep sleep mode, and turn on the LED
* that indicates full power is being used. */
vParTestSetLED( mainDEEP_SLEEP_LED, pdFALSE );
vParTestSetLED( mainFULL_POWER_LED, pdTRUE );
}
/*-----------------------------------------------------------*/
static unsigned short prvReadPOT( void )
{
unsigned short usADCValue;
const unsigned short usMinADCValue = 128;
/* Start an ADC scan. */
S12AD.ADCSR.BIT.ADST = 1;
while( S12AD.ADCSR.BIT.ADST == 1 )
{
/* Just waiting for the ADC scan to complete. Inefficient
* polling! */
}
usADCValue = S12AD.ADDR4;
/* Don't let the ADC value get too small as the LED behaviour will look
* erratic. */
if( usADCValue < usMinADCValue )
{
usADCValue = usMinADCValue;
}
return usADCValue;
}
/*-----------------------------------------------------------*/
#pragma vector = VECT_ICU_IRQ0, VECT_ICU_IRQ1, VECT_ICU_IRQ4
__interrupt void vButtonInterrupt1( void )
{
long lHigherPriorityTaskWoken = pdFALSE;
/* The semaphore is only created when the build is configured to create the
* low power demo. */
if( xSemaphore != NULL )
{
/* This interrupt will bring the CPU out of deep sleep and software
* standby modes. Give the semaphore that was used to place the Tx task
* into an indefinite sleep. */
if( uxQueueMessagesWaitingFromISR( xSemaphore ) == 0 )
{
xSemaphoreGiveFromISR( xSemaphore, &lHigherPriorityTaskWoken );
}
else
{
/* The semaphore was already available, so the task is not blocked
* on it and there is no point giving it. */
}
/* If giving the semaphore caused a task to leave the Blocked state,
* and the task that left the Blocked state has a priority equal to or
* above the priority of the task that this interrupt interrupted, then
* lHigherPriorityTaskWoken will have been set to pdTRUE inside the call
* to xSemaphoreGiveFromISR(), and calling portYIELD_FROM_ISR() will cause
* a context switch to the unblocked task. */
portYIELD_FROM_ISR( lHigherPriorityTaskWoken );
}
}