EFM32G232F64时钟树
1.為了熟悉MCU的時鐘樹,先看看EFM32G232F64的CMU(ClockManagementUnit)
時鐘管理單元(CMU)用于管控晶振(時鐘源)和各個時鐘節(jié)點。出于降低功耗和啟動時間的目的,EFM32G的時鐘源支持?jǐn)?shù)個不同規(guī)格的晶振。另外還有一個獨立的RC振蕩器用于燒寫Flash和調(diào)試。當(dāng)然,時鐘管理單元也從硬件級別上支持RC振蕩器的校準(zhǔn)。
晶振和系統(tǒng)時鐘樹,對于降低MCU的系統(tǒng)功耗有巨大意義。有了低功耗的晶振再結(jié)合彈性的時鐘控制策略,就可以在各種應(yīng)用場合下降低功耗。不使用的外設(shè)就關(guān)掉它的時鐘源以節(jié)能。
時鐘管理單元,能夠配置選擇不同的時鐘源,使能和失能單個時鐘分支和所有外設(shè)的時鐘源,還能為不同的時鐘分支設(shè)置預(yù)分頻系數(shù)。短的晶振啟動時間使得從非低功耗模式(active)到低功耗模式(EM2~EM4)的占空比變得非常高效。RC震蕩校準(zhǔn)特性保證了其高精度的RC震蕩頻率。
CMU_ClockSelectSet(cmuClock_LFB, cmuSelect_LFXO); //使能時鐘分支
CMU_ClockEnable(cmuClock_LEUART0, true); //使能某個外設(shè)的時鐘
1.1邏輯圖
1.2 實際圖
2.
從公司的硬件工程師那里得知,項目的PCB原理圖上只外接了32.768KHz的低速晶振到15和16腳,那個13.56MHz的晶振不是給EFM32G232F64用的,而是給射頻芯片THM3070射頻芯片用的。所以,可以推測,EFM32G還是用了內(nèi)部自帶的高速時鐘源。
2.1 從代碼里找時鐘樹的配置細(xì)節(jié)
找時鐘樹配置是從公司的某個項目代碼中找的。參考了以下博文,沒有明顯發(fā)現(xiàn),但是博文還是值得一看。
2.2 CMU外設(shè)庫函數(shù) -?CMU_ClockSelectSet
void CMU_ClockSelectSet(CMU_Clock_TypeDef clock, CMU_Select_TypeDef ref)中,clock是時鐘分支,ref是外界的時鐘輸入源。《EMF32G-RM》page97上說,默認(rèn)HFCLK分支的時鐘來源選擇HFRCO,但是在多數(shù)應(yīng)用時,推薦使用高頻晶振,可以修改 HFCLKSEL in CMU_CMD(寄存器)去配置HFCLK的時鐘源。
也就是說,在為各個時鐘分支選好時鐘源之后,剩下的就是配置各個時鐘分支的分頻系數(shù)了。這里沒有發(fā)現(xiàn)PLL。選擇時鐘源使用函數(shù)CMU_ClockSelectSet。 補(bǔ)充:
周立功的資料上說,默認(rèn)所有的外設(shè)時鐘都是關(guān)閉的。
2. 常見問題解答
2.1 EFM32 系列MCU
1. 當(dāng)我向某個控制寄存器寫入數(shù)據(jù)然后讀此寄存器,發(fā)現(xiàn)全部是 0,不知是什么原
因?
A:默認(rèn)情況下, EFM32 系列 MCU 內(nèi)部外設(shè)的時鐘都是關(guān)閉的。如果要設(shè)置某個外設(shè)
的 寄 存 器 , 必 須 先 通 過 CMU_HFPERCLKEN0 、 HFCORECLKEN0 、 LFACLKEN 或
LFBCLKEN 來使能相關(guān)外設(shè)的時鐘。若使用低頻外設(shè),還需在 CMU_HFCORECLKEN0 寄
存器中使能 LE 時鐘。
附錄
[轉(zhuǎn)發(fā)]在main()之前,IAR都做了啥?
【原文鏈接】http://www.cnblogs.com/mssql/archive/2011/01/29/tt146.html
最近要在Cortex-M3上寫一個簡單的操作系統(tǒng),打算使用IAR,為了寫好啟動代碼,花了一些時間了解了IAR在main()以前做了些什么事。
首先系統(tǒng)復(fù)位時,Cortex-M3從代碼區(qū)偏移0x0000'0000處獲取棧頂?shù)刂?#xff0c;用來初始化MSP寄存器的值。
接下來從代碼區(qū)偏移0x0000'0004獲取第一個指令的跳轉(zhuǎn)地址。這些地址,是CM3要求放置中斷向量表的地方。
這里是一個程序的啟動區(qū)的反匯編:
__vector_table:
? 08004000??2600??????
? 08004002??2000??????
? 08004004??7E1D??????
? 08004006??0800?????
這個程序是由IAP程序來啟動的,IAP程序獲取0x0800'4000處的MSP值(0x20002600),并設(shè)置為MSP的值,即主堆棧最大范圍是0x2000'0000~0x2000'25FF。接下來IAP程序獲取0x0800'4004處的Reset_Handler的地址(0x0800'7E1D),并跳轉(zhuǎn)到Reset_Handler()執(zhí)行。
IAP在這里完全是模仿了Cortex-M3的復(fù)位序列,也就是說,在沒有IAP的系統(tǒng)上,CM3只能從0x0800'0000獲取MSP,從0x0800'0004獲取第一條指令所處地址。而IAP就存在在0x0800'0000這個地址上,IAP的啟動,已經(jīng)消耗掉了這個復(fù)位序列,所以IAP要啟動UserApp程序的時候,也是完全模仿Cortex-M3的復(fù)位序列的。
接下來我們看看復(fù)位后第一句指令——Reset_Handler()函數(shù)里有什么。
若我們使用的是ST公司標(biāo)準(zhǔn)外設(shè)庫,那么已經(jīng)有了現(xiàn)成的Reset_Handler,不過他是弱定義——PUBWEAK,可以被我們重寫的同名函數(shù)覆蓋。一般來說,我們使用的都是ST提供的Reset_Handler,在V3.4版本的庫中,可以在startup_stm32f10x_xx.s中找到這個函數(shù):
??????? PUBWEAK Reset_Handler
??????? SECTION .text:CODE:REORDER(2)
Reset_Handler
??????? LDR???? R0, =SystemInit
??????? BLX???? R0
??????? LDR???? R0, =__iar_program_start
??????? BX????? R0
看來ST沒有做太多的事,他只調(diào)用了自家?guī)焯峁┑腟ystemInit函數(shù)進(jìn)行系統(tǒng)時鐘、Flash讀取的初始化,并把大權(quán)交給了__iar_program_start這個IAR提供的“內(nèi)部函數(shù)”了,我們就跟緊這個__iar_program_start跳轉(zhuǎn),看看IAR做了什么,上面一段代碼的反匯編如下:
???????Reset_Handler:
__iar_section$$root:
? 08007E1C? 4801????? LDR????????? R0, [PC, #0x4]; LDR???? R0, =SystemInit
??08007E1E? 4780????? BLX????????? R0;BLX???? R0
??08007E20? 4801??????LDR????????? R0, [PC, #0x4];LDR???? R0, =__iar_program_start
??08007E22? 4700????? BX?????????? R0;BX????? R0
? 08007E24? 6C69??????
? 08007E26? 0800??????
? 08007E28? 7D8D??????
? 08007E2A? 0800?????
細(xì)心的觀眾會發(fā)現(xiàn)地址是0x0800'7E1C,比我們查到的0x0800'7E1D差了1,這是ARM家族的遺留問題,因為ARM處理器的指令至少是半字對齊的(16位THUMB指令集 or 32位ARM指令集),所以PC指針的LSB是常為0的,為了充分利用寄存器,ARM公司給PC的LSB了一個重要的使命,那就是在執(zhí)行分支跳轉(zhuǎn)時,PC的LSB=1,表示使用THUMB模式,LSB=0,表示使用ARM模式,但在最新的Cortex-M3內(nèi)核上,只使用了THUMB-2指令集挑大梁,所以這一位要常保持1,所以我們查到的地址是0x0800'7E1D(C=1100,D=1101),放心,我們的CM3內(nèi)核會忽略掉LSB(除非為0,那么會引起一個fault),從而正確跳轉(zhuǎn)到0x0800'7E1C。
從0x0800'7E20處的加載指令,我們可以算出__iar_program_start所處的位置,就是當(dāng)前PC指針(0x0800'7E24),再加上4,即0x0800'7E28處的所指向的地址——0x0800'7D8D(0x0800'7D8C),我們跟緊著跳轉(zhuǎn),__iar_program_start果然在這里:
__iar_program_start:
? 08007D8C? F000F88C? BL???????????__low_level_init
? 08007D90? 2800????? CMP????????? R0, #0x0
? 08007D92? D001????? BEQ??????????__iar_init$$done
? 08007D94? F7FFFFDE? BL???????????__iar_data_init2
? 08007D98? 2000????? MOVS???????? R0, #0x0
? 08007D9A? F7FDFC49? BL???????????main
我們看到IAR提供了__low_level_init這個函數(shù)進(jìn)行了“底層”的初始化,進(jìn)一步跟蹤,我們可以查到__low_level_init這個函數(shù)做了些什么,不是不是我們想象中的不可告人。
__low_level_init:
? 08007EA8? 2001????? MOVS???????? R0, #0x1
? 08007EAA? 4770????? BX?????????? LR
__low_level_init出乎想象的簡單,只是往R0寄存器寫入了1,就立即執(zhí)行"BX LR"回到調(diào)用處了,接下來,__iar_program_start檢查了R0是否為0,為0,則執(zhí)行__iar_init$$done,若不是0,就執(zhí)行__iar_data_init2。__iar_init$$done這個函數(shù)很簡單,只有2句話,第一句是把R0清零,第二句就直接"BL main",跳轉(zhuǎn)到main()函數(shù)了。不過既然__low_level_init已經(jīng)往R0寫入了1,那么我們還是得走下遠(yuǎn)路——看看__iar_data_init2做了些什么,雖然距離main只有一步之遙,不過這中間隱藏了編譯器的思想,我們得耐心看下去。
__iar_data_init2:
? 08007D54? B510????? PUSH???????? {R4,LR}
? 08007D56? 4804????? LDR????????? R0, [PC, #0x10]
? 08007D58? 4C04????? LDR????????? R4, [PC, #0x10]
? 08007D5A? E002????? B??????????? 0x8007D62
? 08007D5C? F8501B04? LDR????????? R1, [R0], #0x4
? 08007D60? 4788????? BLX????????? R1
? 08007D62? 42A0????? CMP????????? R0, R4
? 08007D64? D1FA????? BNE????????? 0x8007D5C
? 08007D66? BD10????? POP????????? {R4,PC}
? 08007D68? 7C78??????
? 08007D6A? 0800??????
? 08007D6C? 7C9C?????
? 08007D6E? 0800?????
看來IAR遲遲不執(zhí)行main()函數(shù),就是為了執(zhí)行__iar_data_init2,我們來分析分析IAR都干了些什么壞事~
首先壓R4,LR入棧,然后加載0x0800'7C78至R0,0x0800'7C9C至R4,馬上跳轉(zhuǎn)到0x0800'7D62執(zhí)行R0,R4的比較,結(jié)果若是相等,則彈出R4,PC,然后立即進(jìn)入main()。不過IAR請君入甕是自不會那么快放我們出來的——結(jié)果不相等,跳轉(zhuǎn)到0x0800'7D5C執(zhí)行,在這里,把R0指向的地址——0x0800'7C78中的值——0x0800'7D71加載到R1,并且R0中的值自加4,更新為0x0800'7C7C,并跳轉(zhuǎn)到R1指向的地址處執(zhí)行,這里是另一個IAR函數(shù):__iar_zero_init2:
__iar_zero_init2:
? 08007D70? 2300????? MOVS???????? R3, #0x0
? 08007D72? E005????? B??????????? 0x8007D80
? 08007D74? F8501B04? LDR????????? R1, [R0], #0x4
? 08007D78? F8413B04? STR????????? R3, [R1], #0x4
? 08007D7C? 1F12????? SUBS???????? R2, R2, #0x4
? 08007D7E? D1FB????? BNE????????? 0x8007D78
? 08007D80? F8502B04? LDR????????? R2, [R0], #0x4
? 08007D84? 2A00????? CMP????????? R2, #0x0
? 08007D86? D1F5????? BNE????????? 0x8007D74
? 08007D88? 4770????? BX?????????? LR
? 08007D8A? 0000????? MOVS???????? R0, R0
__iar_data_init2還沒執(zhí)行完畢,就跳轉(zhuǎn)到了這個__iar_zero_inti2,且看我們慢慢分析這個幫兇——__iar_zero_inti2做了什么。
__iar_zero_inti2將R3寄存器清零,立即跳轉(zhuǎn)到0x0800'7D80執(zhí)行'LDR????????? R2, [R0], #0x4',這句指令與剛才在__iar_data_init2見到的'LDR????????? R1, [R0], #0x4'很類似,都為“后索引”。這回,將R0指向的地址——0x0800'7C7C中的值——0x0000'02F4加載到R2寄存器,然后R0中的值自加4,更新為0x0800'7C80。接下來的指令檢查了R2是否為0,顯然這個函數(shù)沒那么簡單想放我我們,R2的值為2F4,我們又被帶到了0x0800'7D74處,隨后4條指令做了如下的事情:
1、將R0指向的地址——0x0800'7C80中的值——0x2000'27D4加載到R1寄存器,然后R0中的值自加4,更新為0x0800'7C84。
2、將R1指向的地址——0x2000'27D4中的值——改寫為R3寄存器的值——0,然后R1中的值自加4,更新為0x2000'27D8。
3、R2自減4
4、檢查R2是否為0,不為0,跳轉(zhuǎn)到第二條執(zhí)行。不為,則執(zhí)行下一條。
這簡直就是一個循環(huán)!——C語言的循環(huán)for(r2=0x2F4;r2-=4;r!=0){...},我們看看循環(huán)中做了什么。
第一條指令把一個地址加載到了R1——0x2000'27D4?是一個RAM地址,以這個為起點,在循環(huán)中,對長度為2F4的RAM空間進(jìn)行了清零的操作。那為什么IAR要做這個事情呢?消除什么記錄么?用Jlink查看這片內(nèi)存區(qū)域,可以發(fā)現(xiàn)這片區(qū)域是我們定義的全局變量的所在地。也就是說,IAR在每次系統(tǒng)復(fù)位后,都會自動將我們定義的全局變量清零0。
清零完畢后,接下來的指令"LDR????????? R2, [R0], #0x4"將R0指向的地址——0x0800'7C84中的值——0加載到R2寄存器,然后R0中的值自加4,更新為0x0800'7C88。隨后檢查R2是否為0,這里R2為0,執(zhí)行'BX LR'返回到__iar_data_init2函數(shù),若是不為0,我們可以發(fā)現(xiàn)又會跳轉(zhuǎn)至“4指令”處進(jìn)行一個循環(huán)清零的操作。
讀到這里,我們應(yīng)該可以猜到IAR的意圖了:__iar_data_init2一開始加載了0x0800'7C78至R0,0x0800'7C9C至R4,[R0,R4]就是一段啟動代碼區(qū),在這個區(qū)域內(nèi)保存了要“處理”的所有地址與信息——執(zhí)行的函數(shù)地址或者參數(shù),實際上,這片區(qū)域也有一個名字,叫做:Region$$Table$$Base。在這個區(qū)域內(nèi),程序以R0為索引,R4為上限,當(dāng)R0=R4,__iar_data_init2執(zhí)行完畢,跳轉(zhuǎn)至main()函數(shù)。
好了,保持我們這個猜想,繼續(xù)跟蹤我們的PC指針——我們回到了__iar_data_init2函數(shù)中,第一件事就是比較R0,R4的值,可惜的是,仍然不相等,我們又被帶到了0x0800'7D5C,至此,我們應(yīng)該能看出這是一個__iar_data_init2的“主循環(huán)”,這也驗證了我們對IAR意圖的猜想~
??__iar_data_init2中的“主循環(huán)”:
? 08007D5C? F8501B04? LDR????????? R1, [R0], #0x4
? 08007D60? 4788????? BLX????????? R1
? 08007D62? 42A0????? CMP????????? R0, R4
我們可以等價寫為:for(r0=0x0800'7C78,r4=0x0800'7C9C;r0!=r4;r0+=4){...}
此時,我們的R0為0x0800'7C88,經(jīng)過“指令1”,R0變?yōu)?x0800'7C8C,R1為0x0800'7C55。我們來看看,7C55處,IAR又要執(zhí)行何種操作。
__iar_copy_init2:
? 08007C54? B418????? PUSH???????? {R3,R4}
? 08007C56? E009????? B??????????? 0x8007C6C
? 08007C58? F8501B04? LDR????????? R1, [R0], #0x4
? 08007C5C? F8502B04? LDR????????? R2, [R0], #0x4
? 08007C60? F8514B04? LDR????????? R4, [R1], #0x4
? 08007C64? F8424B04? STR????????? R4, [R2], #0x4
? 08007C68? 1F1B????? SUBS???????? R3, R3, #0x4
? 08007C6A? D1F9????? BNE????????? 0x8007C60
? 08007C6C? F8503B04? LDR????????? R3, [R0], #0x4
? 08007C70? 2B00????? CMP????????? R3, #0x0
? 08007C72? D1F1????? BNE????????? 0x8007C58
? 08007C74? BC12????? POP????????? {R1,R4}
? 08007C76? 4770????? BX?????????? LR
這是一個名為__iar_copy_init2的函數(shù),他執(zhí)行了什么"copy"操作呢?
首先壓R3,R4入棧,然后跳轉(zhuǎn)到0x0800'7C6C,從R0——Region$$Table$$Base中取出參數(shù)0x238放入R3,接下來的指令大家應(yīng)該都熟悉了,0x238不為0,所以我們被帶至7C58處,再次從Region$$Table$$Base中取出參數(shù)0x0800'7F14放入R1,從Region$$Table$$Base取出參數(shù)0x2000'2AC8放入R2處。細(xì)心的觀眾應(yīng)該能察覺這和__iar_zero_init2中取參數(shù)的幾乎一樣:先取出大小,隨后取出了地址——只不過這里多出了1個地址,沒錯這就是"copy",隨后的指令
? 08007C60? F8514B04? LDR????????? R4, [R1], #0x4
? 08007C64? F8424B04? STR????????? R4, [R2], #0x4
? 08007C68? 1F1B????? SUBS???????? R3, R3, #0x4
? 08007C6A? D1F9????? BNE????????? 0x8007C60
則是另一個“4指令”,指令1將R1指向地址的數(shù)據(jù)讀到R4,指令2將R2指向地址的數(shù)據(jù)改寫為R4的數(shù)據(jù),指令3、4是完成一個循環(huán)。
說到這里大家都應(yīng)該明白了——這就是一個"copy"的操作,從Flash地址0x0800'7F14起,將長度0x238的數(shù)據(jù)拷貝到RAM地址0x2000'2AC8中。
通過Jlink,我們可以看到這片區(qū)域是我們定義的并且已初始化的全局變量。也就是說,每次復(fù)位后,IAR在此處進(jìn)行全局變量的初始化。
在這“4指令”執(zhí)行完畢后,再次從Region$$Table$$Base中取出參數(shù),為0,比較之后條件符合,函數(shù)返回__iar_data_init2。
此時的R0已經(jīng)為0x0800'7C9C與R4相等,__iar_data_init2終于完成它的使命。
? 08007D98? 2000????? MOVS???????? R0, #0x0
? 08007D9A? F7FDFC49? BL???????????main
將R0清零以后,IAR放棄主動權(quán),把PC指針交給了用戶程序的入口——main()。
但請注意,這里使用的是BL指令進(jìn)行main跳轉(zhuǎn),也就是說,main函數(shù)只是IAR手中的一個子程序,若是main函數(shù)執(zhí)行到了結(jié)尾,接下來則會執(zhí)行exit等IAR提供的“退出”函數(shù)。這些函數(shù),等待下回分解~
總之,IAR在啟動main()函數(shù)以前,執(zhí)行了Reset_Handler,調(diào)用SystemInit()(ST庫提供)進(jìn)行時鐘,Flash讀取初始化,并轉(zhuǎn)入__iar_program_start中執(zhí)行__low_level_init與__iar_data_init2,并在__iar_data_init2中,先后調(diào)用__iar_zero_init2與__iar_copy_init2對全局變量、全局已初始化變量進(jìn)行相應(yīng)的初始化操作。最后,調(diào)用main()函數(shù)執(zhí)行。
這就是IAR在啟動main()函數(shù)之前做的事情,它并沒有那么神秘,只要花些時間,就可以跟跟蹤分析出這個過程。
? /***************************************************************************//*** @brief* Select reference clock/oscillator used for a clock branch.** @details* Notice that if a selected reference is not enabled prior to selecting its* use, it will be enabled, and this function will wait for the selected* oscillator to be stable. It will however NOT be disabled if another* reference clock is selected later.** This feature is particularly important if selecting a new reference* clock for the clock branch clocking the core, otherwise the system* may halt.** @param[in] clock* Clock branch to select reference clock for. One of:* @li #cmuClock_HF* @li #cmuClock_LFA* @li #cmuClock_LFB @if _CMU_LFCLKSEL_LFAE_ULFRCO* @li #cmuClock_LFC* @endif @if _SILICON_LABS_32B_PLATFORM_2* @li #cmuClock_LFE* @endif* @li #cmuClock_DBG @if DOXYDOC_USB_PRESENT* @li #cmuClock_USBC* @endif** @param[in] ref* Reference selected for clocking, please refer to reference manual for* for details on which reference is available for a specific clock branch.* @li #cmuSelect_HFRCO* @li #cmuSelect_LFRCO* @li #cmuSelect_HFXO* @li #cmuSelect_LFXO* @li #cmuSelect_HFCLKLE* @li #cmuSelect_AUXHFRCO* @li #cmuSelect_HFCLK @ifnot DOXYDOC_EFM32_GECKO_FAMILY* @li #cmuSelect_ULFRCO* @endif******************************************************************************/ void CMU_ClockSelectSet(CMU_Clock_TypeDef clock, CMU_Select_TypeDef ref) {uint32_t select = cmuOsc_HFRCO;CMU_Osc_TypeDef osc = cmuOsc_HFRCO;uint32_t freq;uint32_t tmp;uint32_t selRegId; #if defined( _SILICON_LABS_32B_PLATFORM_2 )volatile uint32_t *selReg = NULL; #endif #if defined( CMU_LFCLKSEL_LFAE_ULFRCO )uint32_t lfExtended = 0; #endifselRegId = (clock >> CMU_SEL_REG_POS) & CMU_SEL_REG_MASK;switch (selRegId){case CMU_HFCLKSEL_REG:switch (ref){case cmuSelect_LFXO: #if defined( _SILICON_LABS_32B_PLATFORM_2 )select = CMU_HFCLKSEL_HF_LFXO; #elif defined( _SILICON_LABS_32B_PLATFORM_1 )select = CMU_CMD_HFCLKSEL_LFXO; #endifosc = cmuOsc_LFXO;break;case cmuSelect_LFRCO: #if defined( _SILICON_LABS_32B_PLATFORM_2 )select = CMU_HFCLKSEL_HF_LFRCO; #elif defined( _SILICON_LABS_32B_PLATFORM_1 )select = CMU_CMD_HFCLKSEL_LFRCO; #endifosc = cmuOsc_LFRCO;break;case cmuSelect_HFXO: #if defined( CMU_HFCLKSEL_HF_HFXO )select = CMU_HFCLKSEL_HF_HFXO; #elif defined( CMU_CMD_HFCLKSEL_HFXO )select = CMU_CMD_HFCLKSEL_HFXO; #endifosc = cmuOsc_HFXO; #if defined( CMU_MAX_FREQ_HFLE )/* Set 1 HFLE wait-state until the new HFCLKLE frequency is known.This is known after 'select' is written below. */setHfLeConfig(CMU_MAX_FREQ_HFLE + 1, CMU_MAX_FREQ_HFLE); #endif #if defined( CMU_CTRL_HFXOBUFCUR_BOOSTABOVE32MHZ )/* Adjust HFXO buffer current for frequencies above 32MHz */if (SystemHFXOClockGet() > 32000000){CMU->CTRL = (CMU->CTRL & ~_CMU_CTRL_HFXOBUFCUR_MASK)| CMU_CTRL_HFXOBUFCUR_BOOSTABOVE32MHZ;}else{CMU->CTRL = (CMU->CTRL & ~_CMU_CTRL_HFXOBUFCUR_MASK)| CMU_CTRL_HFXOBUFCUR_BOOSTUPTO32MHZ;} #endifbreak;case cmuSelect_HFRCO: #if defined( _SILICON_LABS_32B_PLATFORM_2 )select = CMU_HFCLKSEL_HF_HFRCO; #elif defined( _SILICON_LABS_32B_PLATFORM_1 )select = CMU_CMD_HFCLKSEL_HFRCO; #endifosc = cmuOsc_HFRCO; #if defined( CMU_MAX_FREQ_HFLE )/* Set 1 HFLE wait-state until the new HFCLKLE frequency is known.This is known after 'select' is written below. */setHfLeConfig(CMU_MAX_FREQ_HFLE + 1, CMU_MAX_FREQ_HFLE); #endifbreak;#if defined( CMU_CMD_HFCLKSEL_USHFRCODIV2 )case cmuSelect_USHFRCODIV2:select = CMU_CMD_HFCLKSEL_USHFRCODIV2;osc = cmuOsc_USHFRCO;break; #endif#if defined( CMU_LFCLKSEL_LFAE_ULFRCO ) || defined( CMU_LFACLKSEL_LFA_ULFRCO )case cmuSelect_ULFRCO:/* ULFRCO cannot be used as HFCLK */EFM_ASSERT(0);return; #endifdefault:EFM_ASSERT(0);return;}/* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(osc, true, true);/* Configure worst case wait states for flash access before selecting */flashWaitStateMax();/* Switch to selected oscillator */ #if defined( _CMU_HFCLKSEL_MASK )CMU->HFCLKSEL = select; #elseCMU->CMD = select; #endif #if defined( CMU_MAX_FREQ_HFLE )/* Update HFLE configuration after 'select' is set.Note that the HFCLKLE clock is connected differently on planform 1 and 2 */setHfLeConfig(CMU_ClockFreqGet(cmuClock_HFLE), CMU_MAX_FREQ_HFLE); #endif/* Keep EMU module informed */EMU_UpdateOscConfig();/* Update CMSIS core clock variable *//* (The function will update the global variable) */freq = SystemCoreClockGet();/* Optimize flash access wait state setting for currently selected core clk */flashWaitStateControl(freq);break;#if defined( _SILICON_LABS_32B_PLATFORM_2 )case CMU_LFACLKSEL_REG:selReg = (selReg == NULL) ? &CMU->LFACLKSEL : selReg; #if !defined( _CMU_LFACLKSEL_LFA_HFCLKLE )/* HFCLKCLE can not be used as LFACLK */EFM_ASSERT(ref != cmuSelect_HFCLKLE); #endif/* Fall through and select clock source */case CMU_LFECLKSEL_REG:selReg = (selReg == NULL) ? &CMU->LFECLKSEL : selReg; #if !defined( _CMU_LFECLKSEL_LFE_HFCLKLE )/* HFCLKCLE can not be used as LFECLK */EFM_ASSERT(ref != cmuSelect_HFCLKLE); #endif/* Fall through and select clock source */case CMU_LFBCLKSEL_REG:selReg = (selReg == NULL) ? &CMU->LFBCLKSEL : selReg;switch (ref){case cmuSelect_Disabled:tmp = _CMU_LFACLKSEL_LFA_DISABLED;break;case cmuSelect_LFXO:/* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_LFXO, true, true);tmp = _CMU_LFACLKSEL_LFA_LFXO;break;case cmuSelect_LFRCO:/* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_LFRCO, true, true);tmp = _CMU_LFACLKSEL_LFA_LFRCO;break;case cmuSelect_HFCLKLE:/* Ensure correct HFLE wait-states and enable HFCLK to LE */setHfLeConfig(SystemCoreClockGet(), CMU_MAX_FREQ_HFLE);BUS_RegBitWrite(&CMU->HFBUSCLKEN0, _CMU_HFBUSCLKEN0_LE_SHIFT, 1);tmp = _CMU_LFBCLKSEL_LFB_HFCLKLE;break;case cmuSelect_ULFRCO:/* ULFRCO is always on, there is no need to enable it. */tmp = _CMU_LFACLKSEL_LFA_ULFRCO;break;default:EFM_ASSERT(0);return;}*selReg = tmp;break;#elif defined( _SILICON_LABS_32B_PLATFORM_1 )case CMU_LFACLKSEL_REG:case CMU_LFBCLKSEL_REG:switch (ref){case cmuSelect_Disabled:tmp = _CMU_LFCLKSEL_LFA_DISABLED;break;case cmuSelect_LFXO:/* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_LFXO, true, true);tmp = _CMU_LFCLKSEL_LFA_LFXO;break;case cmuSelect_LFRCO:/* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_LFRCO, true, true);tmp = _CMU_LFCLKSEL_LFA_LFRCO;break;case cmuSelect_HFCLKLE: #if defined( CMU_MAX_FREQ_HFLE )/* Set HFLE wait-state and divider */freq = SystemCoreClockGet();setHfLeConfig(freq, CMU_MAX_FREQ_HFLE); #endif/* Ensure HFCORE to LE clocking is enabled */BUS_RegBitWrite(&CMU->HFCORECLKEN0, _CMU_HFCORECLKEN0_LE_SHIFT, 1);tmp = _CMU_LFCLKSEL_LFA_HFCORECLKLEDIV2;break;#if defined( CMU_LFCLKSEL_LFAE_ULFRCO )case cmuSelect_ULFRCO:/* ULFRCO is always enabled */tmp = _CMU_LFCLKSEL_LFA_DISABLED;lfExtended = 1;break; #endifdefault:/* Illegal clock source for LFA/LFB selected */EFM_ASSERT(0);return;}/* Apply select */if (selRegId == CMU_LFACLKSEL_REG){ #if defined( _CMU_LFCLKSEL_LFAE_MASK )CMU->LFCLKSEL = (CMU->LFCLKSEL& ~(_CMU_LFCLKSEL_LFA_MASK | _CMU_LFCLKSEL_LFAE_MASK))| (tmp << _CMU_LFCLKSEL_LFA_SHIFT)| (lfExtended << _CMU_LFCLKSEL_LFAE_SHIFT); #elseCMU->LFCLKSEL = (CMU->LFCLKSEL & ~_CMU_LFCLKSEL_LFA_MASK)| (tmp << _CMU_LFCLKSEL_LFA_SHIFT); #endif}else{ #if defined( _CMU_LFCLKSEL_LFBE_MASK )CMU->LFCLKSEL = (CMU->LFCLKSEL& ~(_CMU_LFCLKSEL_LFB_MASK | _CMU_LFCLKSEL_LFBE_MASK))| (tmp << _CMU_LFCLKSEL_LFB_SHIFT)| (lfExtended << _CMU_LFCLKSEL_LFBE_SHIFT); #elseCMU->LFCLKSEL = (CMU->LFCLKSEL & ~_CMU_LFCLKSEL_LFB_MASK)| (tmp << _CMU_LFCLKSEL_LFB_SHIFT); #endif}break;#if defined( _CMU_LFCLKSEL_LFC_MASK )case CMU_LFCCLKSEL_REG:switch(ref){case cmuSelect_Disabled:tmp = _CMU_LFCLKSEL_LFA_DISABLED;break;case cmuSelect_LFXO:/* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_LFXO, true, true);tmp = _CMU_LFCLKSEL_LFC_LFXO;break;case cmuSelect_LFRCO:/* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_LFRCO, true, true);tmp = _CMU_LFCLKSEL_LFC_LFRCO;break;default:/* Illegal clock source for LFC selected */EFM_ASSERT(0);return;}/* Apply select */CMU->LFCLKSEL = (CMU->LFCLKSEL & ~_CMU_LFCLKSEL_LFC_MASK)| (tmp << _CMU_LFCLKSEL_LFC_SHIFT);break; #endif #endif#if defined( CMU_DBGCLKSEL_DBG ) || defined( CMU_CTRL_DBGCLK )case CMU_DBGCLKSEL_REG:switch(ref){ #if defined( CMU_DBGCLKSEL_DBG )case cmuSelect_AUXHFRCO:/* Select AUXHFRCO as debug clock */CMU->DBGCLKSEL = CMU_DBGCLKSEL_DBG_AUXHFRCO;break;case cmuSelect_HFCLK:/* Select divided HFCLK as debug clock */CMU->DBGCLKSEL = CMU_DBGCLKSEL_DBG_HFCLK;break; #endif#if defined( CMU_CTRL_DBGCLK )case cmuSelect_AUXHFRCO:/* Select AUXHFRCO as debug clock */CMU->CTRL = (CMU->CTRL & ~(_CMU_CTRL_DBGCLK_MASK))| CMU_CTRL_DBGCLK_AUXHFRCO;break;case cmuSelect_HFCLK:/* Select divided HFCLK as debug clock */CMU->CTRL = (CMU->CTRL & ~(_CMU_CTRL_DBGCLK_MASK))| CMU_CTRL_DBGCLK_HFCLK;break; #endifdefault:/* Illegal clock source for debug selected */EFM_ASSERT(0);return;}break; #endif#if defined( USB_PRESENT )case CMU_USBCCLKSEL_REG:switch(ref){case cmuSelect_LFXO:/* Select LFXO as clock source for USB, can only be used in sleep mode *//* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_LFXO, true, true);/* Switch oscillator */CMU->CMD = CMU_CMD_USBCCLKSEL_LFXO;/* Wait until clock is activated */while((CMU->STATUS & CMU_STATUS_USBCLFXOSEL)==0){}break;case cmuSelect_LFRCO:/* Select LFRCO as clock source for USB, can only be used in sleep mode *//* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_LFRCO, true, true);/* Switch oscillator */CMU->CMD = CMU_CMD_USBCCLKSEL_LFRCO;/* Wait until clock is activated */while((CMU->STATUS & CMU_STATUS_USBCLFRCOSEL)==0){}break;#if defined( CMU_STATUS_USBCHFCLKSEL )case cmuSelect_HFCLK:/* Select undivided HFCLK as clock source for USB *//* Oscillator must already be enabled to avoid a core lockup */CMU->CMD = CMU_CMD_USBCCLKSEL_HFCLKNODIV;/* Wait until clock is activated */while((CMU->STATUS & CMU_STATUS_USBCHFCLKSEL)==0){}break; #endif#if defined( CMU_CMD_USBCCLKSEL_USHFRCO )case cmuSelect_USHFRCO:/* Select USHFRCO as clock source for USB *//* Ensure selected oscillator is enabled, waiting for it to stabilize */CMU_OscillatorEnable(cmuOsc_USHFRCO, true, true);/* Switch oscillator */CMU->CMD = CMU_CMD_USBCCLKSEL_USHFRCO;/* Wait until clock is activated */while((CMU->STATUS & CMU_STATUS_USBCUSHFRCOSEL)==0){}break; #endifdefault:/* Illegal clock source for USB */EFM_ASSERT(0);return;}break; #endifdefault:EFM_ASSERT(0);break;} #if defined( CMU_MAX_FREQ_HFLE )/* Get to assert wait-state config. */getHfLeConfig(); #endif }?
轉(zhuǎn)載于:https://www.cnblogs.com/Ph-one/p/7597602.html
總結(jié)
以上是生活随笔為你收集整理的EFM32G232F64时钟树的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 51单片机与AVR(SPI)单片机驱动D
- 下一篇: multisim怎么设置晶体管rbe_m