一、在一般没有功耗管理单元的MCU中,我们时钟树的配置就是
CMU_OscillatorEnable(cmuOsc_HFXO, true, true); //选择时钟源 并使能 高速外部时钟
CMU_OscillatorEnable(cmuOsc_LFXO, true, true);
CMU_ClockSelectSet( cmuClock_HF, cmuSelect_HFXO );
CMU_ClockSelectSet(cmuClock_LFA, cmuSelect_LFXO);//Select LFXO as clock source for LFACLK
CMU_ClockSelectSet(cmuClock_LFB, cmuSelect_LFXO);//Select LFXO as clock source for LFBCLK
CMU_ClockEnable(cmuClock_HF, true);
CMU_ClockEnable(cmuClock_LFA, true);
CMU_ClockEnable(cmuClock_LFB, true);
@1:
选择时钟源,使能外部时钟,外部时钟包括 高速时钟和低速时钟,以上面的为例子,高速时钟选择 cmuOsc_HFXO ,低速时钟选择 cmuOsc_LFXO
@2:
设置总线时钟,MCU的高速时钟总线cmuClock_HF以 cmuSelect_HFXO 作为源时钟。
MCU的低速总线2个,低速时钟总线cmuClock_LFA 以cmuSelect_LFXO作为时钟源。
低速时钟总线cmuClock_LFB以cmuSelect_LFXO作为时钟源。
@3:
使能各个总线时钟开关。
二、在低功耗的MCU中,有功耗管理单元,时钟开关要多开一个
CMU_OscillatorEnable(cmuOsc_HFXO, true, true);
CMU_OscillatorEnable(cmuOsc_LFXO, true, true);
CMU_ClockSelectSet( cmuClock_HF, cmuSelect_HFXO );
CMU_ClockSelectSet(cmuClock_LFA, cmuSelect_LFXO);//Select LFXO as clock source for LFACLK
CMU_ClockSelectSet(cmuClock_LFB, cmuSelect_LFXO);//Select LFXO as clock source for LFBCLK
CMU_ClockEnable(cmuClock_HF, true);
CMU_ClockEnable(cmuClock_LFA, true);
CMU_ClockEnable(cmuClock_LFB, true);
CMU_ClockEnable(cmuClock_CORELE, true);//Enable LE clock for CPU access to BURTC registers
cmuClock_CORELE这个时钟单元模块是内核低功耗管理时钟CORELE
如果是这个时钟不使能的话,在你所用的所有低功耗的接口模块里面是是无法执行的,例如在你的letimer的代码中
即使你定义了letimer的定时中断。但是这个中断是一直不会执行的,除非CMU_ClockEnable(cmuClock_CORELE, true);
之后才可以。
除了letimer中还有RTC模块,LETART模块,Lenses模块,只要与低功耗挂钩的模块功能都不起作用。
三、以上时钟开关执行以后,程序正常运行,低功耗模块正常运行。
否则 当程序主代码写上
EMU_EnterEM1();
EMU_EnterEM2(true);
这种进入低功耗模式的代码,MCU会一直休眠,无法被唤醒。程序一直停止在__WFI()上,等到中断唤醒才可以。
__STATIC_INLINE void EMU_EnterEM1(void)
{
/* Just enter Cortex-M3 sleep mode */
SCB->SCR &= ~SCB_SCR_SLEEPDEEP_Msk;
SCB->SCR |= SCB_SCR_SLEEPONEXIT_Msk;
__WFI();
}
四、从低功耗定时器中断处理函数中要加上一句代码才可以唤醒MCU去运行主函数否则 就是中断过来了,不加这句代码页无法唤醒MCU去运行主函数
(SCB->SCR &= ~SCB_SCR_SLEEPONEXIT_Msk) //退出休眠模式。在中断处理函数中添加这个函数之后 才可以进行唤醒MCU