2017-01-09 22:55:53 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* 2014 Eistec AB
|
2017-01-13 12:28:17 +01:00
|
|
|
* 2017 Freie Universität Berlin
|
2017-01-09 22:55:53 +01:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-10-27 01:26:26 +02:00
|
|
|
* @ingroup cpu_kinetis
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup drivers_periph_pm
|
2017-01-09 22:55:53 +01:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of the kernels power management interface
|
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "periph/pm.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2020-03-27 23:41:49 +01:00
|
|
|
/* Set to 1 to use the LEDx macros to show which sleep mode is entered */
|
|
|
|
#define ENABLE_LEDS (0)
|
|
|
|
|
|
|
|
#if ENABLE_LEDS
|
|
|
|
#include "board.h"
|
|
|
|
#define PM_LED(x, state) LED ## x ## _ ## state
|
|
|
|
#else
|
|
|
|
#define PM_LED(x, state)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* SMC_PMCTRL_STOPM masks */
|
|
|
|
enum {
|
|
|
|
SMC_PMCTRL_STOPM_STOP = 0,
|
|
|
|
/* 1 is reserved */
|
|
|
|
SMC_PMCTRL_STOPM_VLPS = 2,
|
|
|
|
SMC_PMCTRL_STOPM_LLS = 3,
|
|
|
|
/* VLLS is not supported */
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Configure which stop mode will be entered when cortexm_sleep(1) is called */
|
|
|
|
static inline void pm_stopm(uint8_t stopm)
|
|
|
|
{
|
|
|
|
SMC->PMCTRL = (SMC->PMCTRL & ~(SMC_PMCTRL_STOPM_MASK)) | SMC_PMCTRL_STOPM(stopm);
|
|
|
|
}
|
|
|
|
|
2017-01-09 22:55:53 +01:00
|
|
|
void pm_set(unsigned mode)
|
|
|
|
{
|
2020-03-27 23:41:49 +01:00
|
|
|
unsigned deep = 1;
|
2017-01-09 22:55:53 +01:00
|
|
|
switch (mode) {
|
2020-03-27 23:41:49 +01:00
|
|
|
/* Only the CPU stops in this mode. No clocks or peripherals are powered
|
|
|
|
* off. This state should always be safe to use, as any interrupt will
|
|
|
|
* wake it. Power consumption in this state is reduced by about 50%. */
|
|
|
|
case KINETIS_PM_WAIT: /* 3 - ARM Sleep */
|
|
|
|
/* WAIT */
|
|
|
|
deep = 0;
|
|
|
|
PM_LED(0, ON);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Some system clocks are stopped in this mode. Usage, means of waking,
|
|
|
|
* and power consumption depend heavily on board-level clock and
|
|
|
|
* peripheral configuration.*/
|
|
|
|
/* If the debug hardware has been used since the last POR, the debug
|
|
|
|
* hardware as well as the clock needed to support it will stay on in
|
|
|
|
* this mode. Thus, after flashing, issue a POR before testing it. */
|
|
|
|
case KINETIS_PM_STOP: /* 2 - ARM Deep Sleep */
|
|
|
|
/* STOP */
|
|
|
|
pm_stopm(SMC_PMCTRL_STOPM_STOP);
|
|
|
|
PM_LED(1, ON);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Some system clocks are stopped in this mode. Usage, means of waking,
|
|
|
|
* and power consumption depend heavily on board-level clock and
|
|
|
|
* peripheral configuration.*/
|
|
|
|
/* If the debug hardware has been used since the last POR, the debug
|
|
|
|
* hardware as well as the clock needed to support it will stay on in
|
|
|
|
* this mode. Thus, after flashing, issue a POR before testing it. */
|
|
|
|
case KINETIS_PM_VLPS: /* 1 - ARM Deep Sleep */
|
|
|
|
/* VLPS */
|
|
|
|
pm_stopm(SMC_PMCTRL_STOPM_VLPS);
|
|
|
|
PM_LED(2, ON);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* All clocks except 32kHz are stopped. Wakeup can occur from
|
|
|
|
* interrupts, several (LLWU-enabled) GPIOs, RTC, or (if the board
|
|
|
|
* configuration uses the 32kHz clock for xtimer) xtimer. Debug hardware
|
|
|
|
* is powered off. Power consumption in this state should be on the
|
|
|
|
* order of 5 uA for the MCU alone, with most boards having additional
|
|
|
|
* parts that will raise this amount. */
|
|
|
|
case KINETIS_PM_LLS: /* 0 - ARM Deep Sleep */
|
|
|
|
/* LLSx */
|
|
|
|
pm_stopm(SMC_PMCTRL_STOPM_LLS);
|
|
|
|
PM_LED(0, ON);
|
|
|
|
PM_LED(2, ON);
|
|
|
|
/* Enable LLWU interrupt, or else we can never resume from LLS */
|
|
|
|
/* Clear pending flag first, the LLWU has no purpose in RUN mode, so
|
|
|
|
* if the flag is set then it must be a remainder from handling the
|
|
|
|
* previous wakeup. */
|
|
|
|
LLWU->F1 = 0xffu;
|
|
|
|
LLWU->F2 = 0xffu;
|
|
|
|
NVIC_ClearPendingIRQ(LLWU_IRQn);
|
|
|
|
NVIC_EnableIRQ(LLWU_IRQn);
|
2017-01-09 22:55:53 +01:00
|
|
|
break;
|
|
|
|
}
|
2020-03-27 23:41:49 +01:00
|
|
|
DEBUG("pm_set(%u)\n", mode);
|
|
|
|
cortexm_sleep(deep);
|
|
|
|
PM_LED(0, OFF);
|
|
|
|
PM_LED(1, OFF);
|
|
|
|
PM_LED(2, OFF);
|
2017-01-09 22:55:53 +01:00
|
|
|
}
|