標(biāo)題: STM32學(xué)習(xí)-時(shí)鐘系統(tǒng) [打印本頁(yè)]
作者: xiongxiao 時(shí)間: 2015-5-27 16:26
標(biāo)題: STM32學(xué)習(xí)-時(shí)鐘系統(tǒng)
上圖是STM32的時(shí)鐘樹。
從樹上我們可以看到,STM32的時(shí)鐘有兩個(gè)來(lái)源——內(nèi)部時(shí)鐘和外部時(shí)鐘。按時(shí)鐘頻率來(lái)分,又分為高速時(shí)鐘和低速時(shí)鐘。所以STM32的時(shí)鐘有四個(gè)來(lái)源——高速外部時(shí)鐘信號(hào)(HSE)、低速外部時(shí)鐘信號(hào)(LSE)、高速內(nèi)部時(shí)鐘信號(hào)(HSI)和低速內(nèi)部時(shí)鐘信號(hào)(LSI),圖中分別用藍(lán)色的①~④標(biāo)注。
①HSE高速外部時(shí)鐘:由外部4~16MHz的晶體或有源晶振提供,通常采用8MHz,ST三合一板上的也是8MHz。
②LSI低速外部時(shí)鐘:外部晶體提供,主要是給實(shí)時(shí)時(shí)鐘(RTC),一般為32.768kHz。
③HSI高速內(nèi)部時(shí)鐘:由內(nèi)部RC振蕩器產(chǎn)生的8MHz時(shí)鐘,但不夠穩(wěn)定。
④LSI低速內(nèi)部時(shí)鐘:內(nèi)部RC振蕩器產(chǎn)生的供給RTC的時(shí)鐘,頻率在30kHz~60kHz之間,通常約40kHz。
時(shí)鐘在STM32內(nèi)部最終是供給四大塊,圖中用紅色橢圓圈出——USB的48MHz時(shí)鐘、系統(tǒng)時(shí)鐘SYSCLK、實(shí)時(shí)時(shí)鐘模塊RTC、獨(dú)立看門狗的時(shí)鐘IWDGCLK。其中最主要的,也是最大頭是系統(tǒng)時(shí)鐘SYSCLK,它可以是內(nèi)部或外部高速時(shí)鐘直接接過來(lái),也可以內(nèi)、外部高速時(shí)鐘是PLL倍頻后提供的,系統(tǒng)時(shí)鐘再分別供給Cortex內(nèi)核、SDIO、AHB總線、DMA、APB1、APB2等。。
我們通常是采用外部8MHz高速時(shí)鐘(HSE),所以著重說(shuō)HSE。我們以前面的GPIO上的時(shí)鐘為例,由ST的Datasheet可知,GPIO是在APB2高速外設(shè)總線上的,圖中綠色的線就是時(shí)鐘的流程,我們一步步地來(lái)看。
8MHz外部晶體(或晶振)輸入后,先經(jīng)過一個(gè)開關(guān)PLLXTPRE(HSE divider for PLL entry),此開關(guān)決定對(duì)HSE進(jìn)行2分頻再輸入到PLL或直接到PLL。我們選擇不分頻。
這樣時(shí)鐘又到了第二個(gè)開關(guān)PLLSRC(PLL entry clock source),此開關(guān)決定PLL的時(shí)鐘來(lái)源,是內(nèi)部高速時(shí)鐘二分頻的時(shí)鐘還是PLLXTPRE的輸出。我們選擇后者,這時(shí)的時(shí)鐘在進(jìn)入PLL前還是8MHz,因?yàn)樵赑LLXTPRE我們沒有分頻。
到了PLL倍頻器,由PLLMUL決定倍頻系統(tǒng)數(shù),可以選擇2~16倍頻輸出,但記住,PLL輸出頻率最高72MHz,所以我們選擇9倍頻,這樣PLL輸出就是最高72MHz的PLLCLK時(shí)鐘了。這時(shí)的PLLCLK為USB提供時(shí)鐘。
開關(guān)SW來(lái)決定SYSCLK的時(shí)鐘來(lái)源,前面已經(jīng)提到,這里我們由PLLCLK做為SYSCLK的來(lái)源,這樣系統(tǒng)時(shí)鐘SYSCLK就是72MHz了。
在供給外設(shè)前,先經(jīng)過AHB預(yù)分頻,我們選擇不分頻;在供給GPIO前,還要再經(jīng)過APB2預(yù)分頻,因?yàn)锳PB2為高速外設(shè),所以我們選擇不分頻,這樣GPIO的時(shí)鐘就是72MHz了。注意,低速外設(shè)APB1最高頻率為36MHz,所以在使用APB1的外設(shè)時(shí),要注意設(shè)置好分頻系統(tǒng)。還要注意,要使用外設(shè),先要對(duì)外設(shè)時(shí)鐘進(jìn)行使能,見圖中黃色云形框。這是因?yàn)镾TM32采用了低功耗的設(shè)計(jì),對(duì)不使用的外設(shè),其時(shí)鐘不使能,以達(dá)到降低功耗的效果。
時(shí)鐘的設(shè)置在程序中是怎么來(lái)實(shí)現(xiàn)的呢?這里我們以前面GPIO的程序來(lái)一步步分析。當(dāng)然,前面的程序是基于ST庫(kù)的,其實(shí)也就是分析ST的官方庫(kù)了。
我們看到main()函數(shù)中的第一行代碼是調(diào)用了一個(gè)函數(shù):
SystemInit();
這個(gè)函數(shù)是在system_stm32f10x.c中:
void SystemInit (void)
{
/* Reset the RCC clock configuration to the default reset state(for debug purpose) */
/* Set HSION bit */
RCC->CR |= (uint32_t)0x00000001;
/* Reset SW, HPRE, PPRE1, PPRE2, ADCPRE and MCO bits */
#ifndef STM32F10X_CL
RCC->CFGR &= (uint32_t)0xF8FF0000;
#else
RCC->CFGR &= (uint32_t)0xF0FF0000;
#endif /* STM32F10X_CL */
/* Reset HSEON, CSSON and PLLON bits */
RCC->CR &= (uint32_t)0xFEF6FFFF;
/* Reset HSEBYP bit */
RCC->CR &= (uint32_t)0xFFFBFFFF;
/* Reset PLLSRC, PLLXTPRE, PLLMUL and USBPRE/OTGFSPRE bits */
RCC->CFGR &= (uint32_t)0xFF80FFFF;
#ifdef STM32F10X_CL
/* Reset PLL2ON and PLL3ON bits */
RCC->CR &= (uint32_t)0xEBFFFFFF;
/* Disable all interrupts and clear pending bits */
RCC->CIR = 0x00FF0000;
/* Reset CFGR2 register */
RCC->CFGR2 = 0x00000000;
#elif defined (STM32F10X_LD_VL) || defined (STM32F10X_MD_VL) || (defined STM32F10X_HD_VL)
/* Disable all interrupts and clear pending bits */
RCC->CIR = 0x009F0000;
/* Reset CFGR2 register */
RCC->CFGR2 = 0x00000000;
#else
/* Disable all interrupts and clear pending bits */
RCC->CIR = 0x009F0000;
#endif /* STM32F10X_CL */
#if defined (STM32F10X_HD) || (defined STM32F10X_XL) || (defined STM32F10X_HD_VL)
#ifdef DATA_IN_ExtSRAM
SystemInit_ExtMemCtl();
#endif /* DATA_IN_ExtSRAM */
#endif
/* Configure the System clock frequency, HCLK, PCLK2 and PCLK1 prescalers */
/* Configure the Flash Latency cycles and enable prefetch buffer */
SetSysClock();
#ifdef VECT_TAB_SRAM
SCB->VTOR = SRAM_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal SRAM. */
#else
SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal FLASH. */
#endif
}
我們可以看到,程序前面一系統(tǒng)Reset或Disable,最后調(diào)用了(紅色標(biāo)記出來(lái))
SetSysClock();
SetSysClock()函數(shù)也位于system_stm32f10x.c源文件中:
static void SetSysClock(void)
{
#ifdef SYSCLK_FREQ_HSE
SetSysClockToHSE();
#elif defined SYSCLK_FREQ_24MHz
SetSysClockTo24();
#elif defined SYSCLK_FREQ_36MHz
SetSysClockTo36();
#elif defined SYSCLK_FREQ_48MHz
SetSysClockTo48();
#elif defined SYSCLK_FREQ_56MHz
SetSysClockTo56();
#elif defined SYSCLK_FREQ_72MHz
SetSysClockTo72();
#endif
/* If none of the define above is enabled, the HSI is used as System clock
source (default after reset) */
}
因?yàn)槲覀兪褂玫氖?2MHz時(shí)鐘,那肯定我們定義了宏SYSCLK_FREQ_72MHz,所以才調(diào)用的函數(shù)SetSysClockTo72()。我們繼續(xù)跟蹤,會(huì)發(fā)現(xiàn)在源文件開頭,我們的確定義了:
#define SYSCLK_FREQ_72MHz 72000000
在SetSysClockTo72()函數(shù)中詳細(xì)地對(duì)72MHz進(jìn)行設(shè)置,這里就不一步步分析。
前面已經(jīng)提到,在使用外設(shè)時(shí),要使能相應(yīng)的外設(shè)時(shí)鐘,例如,在使用GPIOB進(jìn)行流水燈實(shí)驗(yàn)時(shí),程序中調(diào)用了下面的庫(kù)函數(shù),對(duì)外設(shè)時(shí)鐘進(jìn)行設(shè)置:
RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB, ENABLE);
另外,STM32還可以把時(shí)鐘輸出,如圖左下角的咖啡色方框里面,可以由MCO決定,PLL時(shí)鐘二分頻或HIS或HSE或系統(tǒng)時(shí)鐘作為主時(shí)鐘輸出。
歡迎光臨 (http://m.zg4o1577.cn/bbs/) |
Powered by Discuz! X3.1 |
主站蜘蛛池模板:
成年人免费网站
|
亚洲一区二区三区免费在线
|
日本二区在线观看
|
久久99精品久久久久久
|
亚洲精品v日韩精品
|
色婷婷av久久久久久久
|
国产专区在线
|
亚洲成av人片在线观看
|
五月综合色啪
|
无码一区二区三区视频
|
情侣av
|
精品欧美乱码久久久久久1区2区
|
久久精品高清视频
|
久久久久久中文字幕
|
亚洲国产成人精品女人久久久
|
一区中文字幕
|
日韩快播电影
|
欧美日韩综合精品
|
91久色
|
久久久免费观看视频
|
亚洲夜夜爽
|
日韩影院在线
|
伊人狼人影院
|
干出白浆视频
|
欧美区日韩区
|
日韩高清一区二区
|
午夜精品久久久
|
www.亚洲精品
|
国产一区二区在线免费观看
|
欧美久久久久久久
|
精品一区久久
|
日本视频中文字幕
|
欧美888
|
亚洲成人精品
|
久久久精品一区
|
午夜精品久久久久久久星辰影院
|
欧美色综合一区二区三区
|
国产内谢|
精品国产乱码久久久久久88av
|
操到爽
|
天堂色|