blob: d6bef9ea28c11b82bb2fc94347f1be8cd54f7179 [file] [log] [blame]
/*
* Copyright (c) 2018, Piotr Mienkowski
*
* SPDX-License-Identifier: Apache-2.0
*/
#include <zephyr.h>
#include <power.h>
#include <em_emu.h>
#include <logging/log.h>
LOG_MODULE_DECLARE(soc, CONFIG_SOC_LOG_LEVEL);
/*
* Power state map:
* SYS_POWER_STATE_LOW_POWER_1: EM1 Sleep
* SYS_POWER_STATE_LOW_POWER_2: EM2 Deep Sleep
* SYS_POWER_STATE_LOW_POWER_3: EM3 Stop
*/
/* Invoke Low Power/System Off specific Tasks */
void sys_set_power_state(enum power_states state)
{
LOG_DBG("SoC entering power state %d", state);
/* FIXME: When this function is entered the Kernel has disabled
* interrupts using BASEPRI register. This is incorrect as it prevents
* waking up from any interrupt which priority is not 0. Work around the
* issue and disable interrupts using PRIMASK register as recommended
* by ARM.
*/
/* Set PRIMASK */
__disable_irq();
/* Set BASEPRI to 0 */
irq_unlock(0);
switch (state) {
#ifdef CONFIG_SYS_POWER_LOW_POWER_STATES
#ifdef CONFIG_HAS_STATE_LOW_POWER_1
case SYS_POWER_STATE_LOW_POWER_1:
EMU_EnterEM1();
break;
#endif /* CONFIG_HAS_STATE_LOW_POWER_1 */
#ifdef CONFIG_HAS_STATE_LOW_POWER_2
case SYS_POWER_STATE_LOW_POWER_2:
EMU_EnterEM2(true);
break;
#endif /* CONFIG_HAS_STATE_LOW_POWER_2 */
#ifdef CONFIG_HAS_STATE_LOW_POWER_3
case SYS_POWER_STATE_LOW_POWER_3:
EMU_EnterEM3(true);
break;
#endif /* CONFIG_HAS_STATE_LOW_POWER_3 */
#endif /* CONFIG_SYS_POWER_LOW_POWER_STATES */
default:
LOG_ERR("Unsupported power state %u", state);
break;
}
LOG_DBG("SoC leaving power state %d", state);
/* Clear PRIMASK */
__enable_irq();
}
/* Handle SOC specific activity after Low Power Mode Exit */
void sys_power_state_post_ops(enum power_states state)
{
ARG_UNUSED(state);
}