mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-27 00:37:22 +08:00
261 lines
5.5 KiB
C
261 lines
5.5 KiB
C
/*
|
|
* Copyright (C) 2018 Shanghai Eastsoft Microelectronics Co., Ltd.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2019-04-08 wangyq the first version
|
|
* 2019-11-01 wangyq adapt to the new power management interface
|
|
*/
|
|
#include <rthw.h>
|
|
#include <board.h>
|
|
#include <rtdevice.h>
|
|
#include <drv_lptim.h>
|
|
#include <ald_cmu.h>
|
|
#include <ald_pmu.h>
|
|
|
|
#ifdef RT_USING_PM
|
|
|
|
static void uart_console_reconfig(void)
|
|
{
|
|
struct serial_configure config = RT_SERIAL_CONFIG_DEFAULT;
|
|
|
|
rt_device_control(rt_console_get_device(), RT_DEVICE_CTRL_CONFIG, &config);
|
|
}
|
|
|
|
static void delay(void)
|
|
{
|
|
long i;
|
|
rt_base_t level;
|
|
|
|
level = rt_hw_interrupt_disable();
|
|
i = 0;
|
|
do{
|
|
i++;
|
|
}
|
|
while (i < 10000);
|
|
|
|
rt_hw_interrupt_enable(level);
|
|
}
|
|
|
|
/**
|
|
* This function will put ES32F033x into sleep mode.
|
|
*
|
|
* @param pm pointer to power manage structure
|
|
*/
|
|
static void sleep(struct rt_pm *pm, uint8_t mode)
|
|
{
|
|
switch (mode)
|
|
{
|
|
case PM_SLEEP_MODE_NONE:
|
|
break;
|
|
|
|
case PM_SLEEP_MODE_IDLE:
|
|
//__WFI();
|
|
break;
|
|
|
|
case PM_SLEEP_MODE_LIGHT:
|
|
if (pm->run_mode == PM_RUN_MODE_LOW_SPEED)
|
|
{
|
|
/* Enter LP SLEEP Mode, Enable low-power regulator */
|
|
ald_pmu_lprun_config(PMU_LDO_LPMODE_OUTPUT_1_5, ENABLE);
|
|
}
|
|
else
|
|
{
|
|
/* Enter SLEEP Mode, Main regulator is ON */
|
|
ald_pmu_stop1_enter();
|
|
delay();
|
|
}
|
|
break;
|
|
|
|
case PM_SLEEP_MODE_DEEP:
|
|
/* Enter STOP 2 mode */
|
|
ald_pmu_stop2_enter();
|
|
delay();
|
|
break;
|
|
|
|
case PM_SLEEP_MODE_STANDBY:
|
|
/* Enter STANDBY mode */
|
|
ald_pmu_stop2_enter();
|
|
delay();
|
|
break;
|
|
|
|
case PM_SLEEP_MODE_SHUTDOWN:
|
|
/* Enter SHUTDOWNN mode */
|
|
ald_pmu_stop2_enter();
|
|
delay();
|
|
break;
|
|
|
|
default:
|
|
RT_ASSERT(0);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static uint8_t run_speed[PM_RUN_MODE_MAX][2] =
|
|
{
|
|
{48, 0},
|
|
{48, 1},
|
|
{24, 2},
|
|
{2, 3},
|
|
};
|
|
|
|
static void run(struct rt_pm *pm, uint8_t mode)
|
|
{
|
|
static uint8_t last_mode;
|
|
static char *run_str[] = PM_RUN_MODE_NAMES;
|
|
extern uint32_t __system_clock;
|
|
|
|
if (mode == last_mode)
|
|
return;
|
|
last_mode = mode;
|
|
|
|
ald_cmu_clock_config_default();
|
|
__system_clock = 24000000;
|
|
switch (mode)
|
|
{
|
|
case PM_RUN_MODE_HIGH_SPEED:
|
|
case PM_RUN_MODE_NORMAL_SPEED:
|
|
/* hosc 12MHz, from hosc/3 pll to 48MHz */
|
|
ald_cmu_pll1_config(CMU_PLL1_INPUT_HRC_6, CMU_PLL1_OUTPUT_48M);
|
|
/* MCLK 48MHz */
|
|
ald_cmu_clock_config(CMU_CLOCK_PLL1, 48000000);
|
|
break;
|
|
case PM_RUN_MODE_MEDIUM_SPEED:
|
|
break;
|
|
case PM_RUN_MODE_LOW_SPEED:
|
|
ald_cmu_clock_config(CMU_CLOCK_HRC, 2000000);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* 4. 更新外设时钟 */
|
|
uart_console_reconfig();
|
|
/* Re-Configure the Systick time */
|
|
SysTick_Config(ald_cmu_get_sys_clock() / RT_TICK_PER_SECOND);
|
|
|
|
rt_kprintf("switch to %s mode, frequency = %d MHz\n", run_str[mode], run_speed[mode][0]);
|
|
}
|
|
|
|
/**
|
|
* This function caculate the PM tick from OS tick
|
|
*
|
|
* @param tick OS tick
|
|
*
|
|
* @return the PM tick
|
|
*/
|
|
static rt_tick_t es32f0_pm_tick_from_os_tick(rt_tick_t tick)
|
|
{
|
|
rt_uint32_t freq = es32f0_lptim_get_countfreq();
|
|
|
|
return (freq * tick / RT_TICK_PER_SECOND);
|
|
}
|
|
|
|
/**
|
|
* This function caculate the OS tick from PM tick
|
|
*
|
|
* @param tick PM tick
|
|
*
|
|
* @return the OS tick
|
|
*/
|
|
static rt_tick_t es32f0_os_tick_from_pm_tick(rt_uint32_t tick)
|
|
{
|
|
static rt_uint32_t os_tick_remain = 0;
|
|
rt_uint32_t ret, freq;
|
|
|
|
freq = es32f0_lptim_get_countfreq();
|
|
ret = (tick * RT_TICK_PER_SECOND + os_tick_remain) / freq;
|
|
|
|
os_tick_remain += (tick * RT_TICK_PER_SECOND);
|
|
os_tick_remain %= freq;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* This function start the timer of pm
|
|
*
|
|
* @param pm Pointer to power manage structure
|
|
* @param timeout How many OS Ticks that MCU can sleep
|
|
*/
|
|
static void pm_timer_start(struct rt_pm *pm, rt_uint32_t timeout)
|
|
{
|
|
RT_ASSERT(pm != RT_NULL);
|
|
RT_ASSERT(timeout > 0);
|
|
|
|
if (timeout != RT_TICK_MAX)
|
|
{
|
|
/* Convert OS Tick to pmtimer timeout value */
|
|
timeout = es32f0_pm_tick_from_os_tick(timeout);
|
|
/* MAX 0xFFFF */
|
|
if (timeout > es32f0_lptim_get_tick_max())
|
|
{
|
|
timeout = es32f0_lptim_get_tick_max();
|
|
}
|
|
|
|
/* Enter PM_TIMER_MODE */
|
|
es32f0_lptim_start(timeout);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This function stop the timer of pm
|
|
*
|
|
* @param pm Pointer to power manage structure
|
|
*/
|
|
static void pm_timer_stop(struct rt_pm *pm)
|
|
{
|
|
RT_ASSERT(pm != RT_NULL);
|
|
|
|
/* Reset pmtimer status */
|
|
es32f0_lptim_stop();
|
|
}
|
|
|
|
/**
|
|
* This function calculate how many OS Ticks that MCU have suspended
|
|
*
|
|
* @param pm Pointer to power manage structure
|
|
*
|
|
* @return OS Ticks
|
|
*/
|
|
static rt_tick_t pm_timer_get_tick(struct rt_pm *pm)
|
|
{
|
|
rt_uint32_t timer_tick;
|
|
|
|
RT_ASSERT(pm != RT_NULL);
|
|
|
|
timer_tick = es32f0_lptim_get_current_tick();
|
|
|
|
return es32f0_os_tick_from_pm_tick(timer_tick);
|
|
}
|
|
|
|
/**
|
|
* This function initialize the power manager
|
|
*/
|
|
int drv_pm_hw_init(void)
|
|
{
|
|
static const struct rt_pm_ops _ops =
|
|
{
|
|
sleep,
|
|
run,
|
|
pm_timer_start,
|
|
pm_timer_stop,
|
|
pm_timer_get_tick
|
|
};
|
|
|
|
rt_uint8_t timer_mask = 0;
|
|
|
|
/* initialize timer mask */
|
|
timer_mask = 1UL << PM_SLEEP_MODE_DEEP;
|
|
|
|
/* initialize system pm module */
|
|
rt_system_pm_init(&_ops, timer_mask, RT_NULL);
|
|
|
|
return 0;
|
|
}
|
|
INIT_BOARD_EXPORT(drv_pm_hw_init);
|
|
|
|
#endif
|