407 lines
11 KiB
C
407 lines
11 KiB
C
|
/********************************** (C) COPYRIGHT *******************************
|
|||
|
* File Name : CH59x_pwr.c
|
|||
|
* Author : WCH
|
|||
|
* Version : V1.2
|
|||
|
* Date : 2021/11/17
|
|||
|
* Description
|
|||
|
*********************************************************************************
|
|||
|
* Copyright (c) 2021 Nanjing Qinheng Microelectronics Co., Ltd.
|
|||
|
* Attention: This software (modified or not) and binary are used for
|
|||
|
* microcontroller manufactured by Nanjing Qinheng Microelectronics.
|
|||
|
*******************************************************************************/
|
|||
|
|
|||
|
#include "CH59x_common.h"
|
|||
|
|
|||
|
/*********************************************************************
|
|||
|
* @fn PWR_DCDCCfg
|
|||
|
*
|
|||
|
* @brief 启用内部DC/DC电源,用于节约系统功耗
|
|||
|
*
|
|||
|
* @param s - 是否打开DCDC电源
|
|||
|
*
|
|||
|
* @return none
|
|||
|
*/
|
|||
|
void PWR_DCDCCfg(FunctionalState s)
|
|||
|
{
|
|||
|
uint16_t adj = R16_AUX_POWER_ADJ;
|
|||
|
uint16_t plan = R16_POWER_PLAN;
|
|||
|
|
|||
|
if(s == DISABLE)
|
|||
|
{
|
|||
|
|
|||
|
adj &= ~RB_DCDC_CHARGE;
|
|||
|
plan &= ~(RB_PWR_DCDC_EN | RB_PWR_DCDC_PRE); // 旁路 DC/DC
|
|||
|
sys_safe_access_enable();
|
|||
|
R16_AUX_POWER_ADJ = adj;
|
|||
|
R16_POWER_PLAN = plan;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
uint32_t HW_Data[2];
|
|||
|
FLASH_EEPROM_CMD(CMD_GET_ROM_INFO, ROM_CFG_ADR_HW, HW_Data, 0);
|
|||
|
if((HW_Data[0]) & (1 << 13))
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
adj |= RB_DCDC_CHARGE;
|
|||
|
plan |= RB_PWR_DCDC_PRE;
|
|||
|
sys_safe_access_enable();
|
|||
|
R16_AUX_POWER_ADJ = adj;
|
|||
|
R16_POWER_PLAN = plan;
|
|||
|
sys_safe_access_disable();
|
|||
|
DelayUs(10);
|
|||
|
sys_safe_access_enable();
|
|||
|
R16_POWER_PLAN |= RB_PWR_DCDC_EN;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/*********************************************************************
|
|||
|
* @fn PWR_UnitModCfg
|
|||
|
*
|
|||
|
* @brief 可控单元模块的电源控制
|
|||
|
*
|
|||
|
* @param s - 是否打开电源
|
|||
|
* @param unit - please refer to unit of controllable power supply
|
|||
|
*
|
|||
|
* @return none
|
|||
|
*/
|
|||
|
void PWR_UnitModCfg(FunctionalState s, uint8_t unit)
|
|||
|
{
|
|||
|
uint8_t ck32k_cfg = R8_CK32K_CONFIG;
|
|||
|
|
|||
|
if(s == DISABLE) //关闭
|
|||
|
{
|
|||
|
ck32k_cfg &= ~(unit & 0x03);
|
|||
|
}
|
|||
|
else //打开
|
|||
|
{
|
|||
|
ck32k_cfg |= (unit & 0x03);
|
|||
|
}
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_CK32K_CONFIG = ck32k_cfg;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
|
|||
|
/*********************************************************************
|
|||
|
* @fn PWR_PeriphClkCfg
|
|||
|
*
|
|||
|
* @brief 外设时钟控制位
|
|||
|
*
|
|||
|
* @param s - 是否打开对应外设时钟
|
|||
|
* @param perph - please refer to Peripher CLK control bit define
|
|||
|
*
|
|||
|
* @return none
|
|||
|
*/
|
|||
|
void PWR_PeriphClkCfg(FunctionalState s, uint16_t perph)
|
|||
|
{
|
|||
|
uint32_t sleep_ctrl = R32_SLEEP_CONTROL;
|
|||
|
|
|||
|
if(s == DISABLE)
|
|||
|
{
|
|||
|
sleep_ctrl |= perph;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
sleep_ctrl &= ~perph;
|
|||
|
}
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R32_SLEEP_CONTROL = sleep_ctrl;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
|
|||
|
/*********************************************************************
|
|||
|
* @fn PWR_PeriphWakeUpCfg
|
|||
|
*
|
|||
|
* @brief 睡眠唤醒源配置
|
|||
|
*
|
|||
|
* @param s - 是否打开此外设睡眠唤醒功能
|
|||
|
* @param perph - 需要设置的唤醒源
|
|||
|
* RB_SLP_USB_WAKE - USB 为唤醒源
|
|||
|
* RB_SLP_RTC_WAKE - RTC 为唤醒源
|
|||
|
* RB_SLP_GPIO_WAKE - GPIO 为唤醒源
|
|||
|
* RB_SLP_BAT_WAKE - BAT 为唤醒源
|
|||
|
* @param mode - refer to WakeUP_ModeypeDef
|
|||
|
*
|
|||
|
* @return none
|
|||
|
*/
|
|||
|
void PWR_PeriphWakeUpCfg(FunctionalState s, uint8_t perph, WakeUP_ModeypeDef mode)
|
|||
|
{
|
|||
|
uint8_t m;
|
|||
|
|
|||
|
if(s == DISABLE)
|
|||
|
{
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_SLP_WAKE_CTRL &= ~perph;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
switch(mode)
|
|||
|
{
|
|||
|
case Short_Delay:
|
|||
|
m = 0x01;
|
|||
|
break;
|
|||
|
|
|||
|
case Long_Delay:
|
|||
|
m = 0x00;
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
m = 0x01;
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_SLP_WAKE_CTRL |= RB_WAKE_EV_MODE | perph;
|
|||
|
sys_safe_access_disable();
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_SLP_POWER_CTRL &= ~(RB_WAKE_DLY_MOD);
|
|||
|
sys_safe_access_disable();
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_SLP_POWER_CTRL |= m;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/*********************************************************************
|
|||
|
* @fn PowerMonitor
|
|||
|
*
|
|||
|
* @brief 电源监控
|
|||
|
*
|
|||
|
* @param s - 是否打开此功能
|
|||
|
* @param vl - refer to VolM_LevelypeDef
|
|||
|
*
|
|||
|
* @return none
|
|||
|
*/
|
|||
|
void PowerMonitor(FunctionalState s, VolM_LevelypeDef vl)
|
|||
|
{
|
|||
|
uint8_t ctrl = R8_BAT_DET_CTRL;
|
|||
|
uint8_t cfg = R8_BAT_DET_CFG;
|
|||
|
|
|||
|
if(s == DISABLE)
|
|||
|
{
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_BAT_DET_CTRL = 0;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if(vl & 0x80)
|
|||
|
{
|
|||
|
cfg = vl & 0x03;
|
|||
|
ctrl = RB_BAT_MON_EN | ((vl >> 2) & 1);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
|
|||
|
cfg = vl & 0x03;
|
|||
|
ctrl = RB_BAT_DET_EN;
|
|||
|
}
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_BAT_DET_CTRL = ctrl;
|
|||
|
R8_BAT_DET_CFG = cfg;
|
|||
|
sys_safe_access_disable();
|
|||
|
|
|||
|
mDelayuS(1);
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_BAT_DET_CTRL |= RB_BAT_LOW_IE | RB_BAT_LOWER_IE;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/*********************************************************************
|
|||
|
* @fn LowPower_Idle
|
|||
|
*
|
|||
|
* @brief 低功耗-Idle模式
|
|||
|
*
|
|||
|
* @param none
|
|||
|
*
|
|||
|
* @return none
|
|||
|
*/
|
|||
|
__HIGH_CODE
|
|||
|
void LowPower_Idle(void)
|
|||
|
{
|
|||
|
FLASH_ROM_SW_RESET();
|
|||
|
R8_FLASH_CTRL = 0x04; //flash关闭
|
|||
|
|
|||
|
PFIC->SCTLR &= ~(1 << 2); // sleep
|
|||
|
__WFI();
|
|||
|
__nop();
|
|||
|
__nop();
|
|||
|
}
|
|||
|
|
|||
|
/*********************************************************************
|
|||
|
* @fn LowPower_Halt
|
|||
|
*
|
|||
|
* @brief 低功耗-Halt模式,此低功耗切到HSI/5时钟运行,唤醒后需要用户自己重新选择系统时钟源
|
|||
|
*
|
|||
|
* @param none
|
|||
|
*
|
|||
|
* @return none
|
|||
|
*/
|
|||
|
__HIGH_CODE
|
|||
|
void LowPower_Halt(void)
|
|||
|
{
|
|||
|
uint8_t x32Kpw, x32Mpw;
|
|||
|
|
|||
|
FLASH_ROM_SW_RESET();
|
|||
|
R8_FLASH_CTRL = 0x04; //flash关闭
|
|||
|
x32Kpw = R8_XT32K_TUNE;
|
|||
|
x32Mpw = R8_XT32M_TUNE;
|
|||
|
x32Mpw = (x32Mpw & 0xfc) | 0x03; // 150%额定电流
|
|||
|
if(R16_RTC_CNT_32K > 0x3fff)
|
|||
|
{ // 超过500ms
|
|||
|
x32Kpw = (x32Kpw & 0xfc) | 0x01; // LSE驱动电流降低到额定电流
|
|||
|
}
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_BAT_DET_CTRL = 0; // 关闭电压监控
|
|||
|
sys_safe_access_disable();
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_XT32K_TUNE = x32Kpw;
|
|||
|
R8_XT32M_TUNE = x32Mpw;
|
|||
|
sys_safe_access_disable();
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_PLL_CONFIG |= (1 << 5);
|
|||
|
sys_safe_access_disable();
|
|||
|
|
|||
|
PFIC->SCTLR |= (1 << 2); //deep sleep
|
|||
|
__WFI();
|
|||
|
__nop();
|
|||
|
__nop();
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_PLL_CONFIG &= ~(1 << 5);
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
|
|||
|
/*******************************************************************************
|
|||
|
* Function Name : LowPower_Sleep
|
|||
|
* Description : 低功耗-Sleep模式。
|
|||
|
注意当主频为80M时,睡眠唤醒中断不可调用flash内代码,且退出此函数前需要加上30us延迟。
|
|||
|
* Input : rm:
|
|||
|
RB_PWR_RAM2K - 2K retention SRAM 供电
|
|||
|
RB_PWR_RAM24K - 24K main SRAM 供电
|
|||
|
RB_PWR_EXTEND - USB 和 BLE 单元保留区域供电
|
|||
|
RB_PWR_XROM - FlashROM 供电
|
|||
|
NULL - 以上单元都断电
|
|||
|
* Return : None
|
|||
|
*******************************************************************************/
|
|||
|
__HIGH_CODE
|
|||
|
void LowPower_Sleep(uint16_t rm)
|
|||
|
{
|
|||
|
__attribute__((aligned(4))) uint8_t MacAddr[6] = {0};
|
|||
|
uint8_t x32Kpw, x32Mpw;
|
|||
|
uint16_t power_plan;
|
|||
|
|
|||
|
GetMACAddress(MacAddr);
|
|||
|
|
|||
|
x32Kpw = R8_XT32K_TUNE;
|
|||
|
x32Mpw = R8_XT32M_TUNE;
|
|||
|
x32Mpw = (x32Mpw & 0xfc) | 0x03; // 150%额定电流
|
|||
|
if(R16_RTC_CNT_32K > 0x3fff)
|
|||
|
{ // 超过500ms
|
|||
|
x32Kpw = (x32Kpw & 0xfc) | 0x01; // LSE驱动电流降低到额定电流
|
|||
|
}
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_BAT_DET_CTRL = 0; // 关闭电压监控
|
|||
|
sys_safe_access_disable();
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_XT32K_TUNE = x32Kpw;
|
|||
|
R8_XT32M_TUNE = x32Mpw;
|
|||
|
sys_safe_access_disable();
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R16_POWER_PLAN &= ~RB_XT_PRE_EN;
|
|||
|
sys_safe_access_disable();
|
|||
|
|
|||
|
PFIC->SCTLR |= (1 << 2); //deep sleep
|
|||
|
|
|||
|
power_plan = R16_POWER_PLAN & (RB_PWR_DCDC_EN | RB_PWR_DCDC_PRE);
|
|||
|
power_plan |= RB_PWR_PLAN_EN | RB_PWR_CORE | rm | (2<<11);
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_SLP_POWER_CTRL |= RB_RAM_RET_LV;
|
|||
|
R16_POWER_PLAN = power_plan;
|
|||
|
sys_safe_access_disable();
|
|||
|
if((rm & RB_XT_PRE_EN) == 0)
|
|||
|
{
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_PLL_CONFIG |= (1 << 5);
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|
|||
|
|
|||
|
__WFI();
|
|||
|
__nop();
|
|||
|
__nop();
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R16_POWER_PLAN &= ~RB_XT_PRE_EN;
|
|||
|
sys_safe_access_disable();
|
|||
|
|
|||
|
if((rm & RB_XT_PRE_EN) == 0)
|
|||
|
{
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_PLL_CONFIG &= ~(1 << 5);
|
|||
|
sys_safe_access_disable();
|
|||
|
DelayUs(20);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/*********************************************************************
|
|||
|
* @fn LowPower_Shutdown
|
|||
|
*
|
|||
|
* @brief 低功耗-Shutdown模式,此低功耗切到HSI/5时钟运行,唤醒后需要用户自己重新选择系统时钟源
|
|||
|
* @note 注意调用此函数,DCDC功能强制关闭,唤醒后可以手动再次打开
|
|||
|
*
|
|||
|
* @param rm - 供电模块选择
|
|||
|
* RB_PWR_RAM2K - 2K retention SRAM 供电
|
|||
|
* RB_PWR_RAM16K - 16K main SRAM 供电
|
|||
|
* NULL - 以上单元都断电
|
|||
|
*
|
|||
|
* @return none
|
|||
|
*/
|
|||
|
__HIGH_CODE
|
|||
|
void LowPower_Shutdown(uint16_t rm)
|
|||
|
{
|
|||
|
uint8_t x32Kpw, x32Mpw;
|
|||
|
|
|||
|
FLASH_ROM_SW_RESET();
|
|||
|
x32Kpw = R8_XT32K_TUNE;
|
|||
|
x32Mpw = R8_XT32M_TUNE;
|
|||
|
x32Mpw = (x32Mpw & 0xfc) | 0x03; // 150%额定电流
|
|||
|
if(R16_RTC_CNT_32K > 0x3fff)
|
|||
|
{ // 超过500ms
|
|||
|
x32Kpw = (x32Kpw & 0xfc) | 0x01; // LSE驱动电流降低到额定电流
|
|||
|
}
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_BAT_DET_CTRL = 0; // 关闭电压监控
|
|||
|
sys_safe_access_disable();
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_XT32K_TUNE = x32Kpw;
|
|||
|
R8_XT32M_TUNE = x32Mpw;
|
|||
|
sys_safe_access_disable();
|
|||
|
SetSysClock(CLK_SOURCE_HSE_6_4MHz);
|
|||
|
|
|||
|
PFIC->SCTLR |= (1 << 2); //deep sleep
|
|||
|
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_SLP_POWER_CTRL |= RB_RAM_RET_LV;
|
|||
|
sys_safe_access_disable();
|
|||
|
sys_safe_access_enable();
|
|||
|
R16_POWER_PLAN = RB_PWR_PLAN_EN | rm;
|
|||
|
sys_safe_access_disable();
|
|||
|
__WFI();
|
|||
|
__nop();
|
|||
|
__nop();
|
|||
|
FLASH_ROM_SW_RESET();
|
|||
|
sys_safe_access_enable();
|
|||
|
R8_RST_WDOG_CTRL |= RB_SOFTWARE_RESET;
|
|||
|
sys_safe_access_disable();
|
|||
|
}
|