對STM32系統(tǒng)時鐘和分頻的一點理解
系統(tǒng)時鐘和分頻
首先來手冊里的一段話。
三種不同的時鐘源可被用來驅(qū)動系統(tǒng)時鐘(SYSCLK)
HSI振蕩器時鐘
HSE振蕩器時鐘
PLL時鐘
一般用的是PLL時鐘,后面有證據(jù)。
我們可以通過庫函數(shù)獲取各時鐘值
void RCC_GetClocksFreq(RCC_ClocksTypeDef* RCC_Clocks)
在我的系統(tǒng)里,把時鐘值打印信息如下:
SYSCLK:0x44aa200 //72000000,72MHz
HCLK:0x44aa200 //72000000,72MHz
PCLK1:0x2255100 //36000000,36MHz
PCLK2:0x44aa200 //72000000,72MHz
ADCCLK:0x2255100 //36000000,36MHz
RCC->CFGR:0x001D040A //PLL輸出作為系統(tǒng)時鐘
可推測幾個預(yù)分頻值為
AHB prescaler = 1
APB1 prescaler = 2
APB2 prescaler = 1
ADC prescaler = 2
根據(jù)讀取RCC->CFGR寄存器值為:0x001D040A,上面推測完全正確。
CFGR寄存器的SWS段也說明:PLL輸出作為系統(tǒng)時鐘。
TIM2使用PCLK1,但注意時鐘樹里有這一段
見附圖
已知APB1 prescaler=2,故TIM2CLK = PCLK1*2 = 72MHz.
所以被TIM2分頻的時鐘大小是72MHz。
我的程序也證明了這點
TIM_TimeBaseInitTypeDef tim2_InitStruct;
TIM_DeInit(TIM2);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);//Enable Timer2 clock.
NVIC_ConfigurationForTimer2();
// PCLK1=36MHz, PSC=36000-1, CK_CNT=36MHz/(PSC+1)=1000
// ARR=2000, 1s/1000 * 2000 = 2s.
tim2_InitStruct.TIM_Prescaler = 36000-1;
tim2_InitStruct.TIM_Period = 2000-1;
tim2_InitStruct.TIM_CounterMode = TIM_CounterMode_Up;
tim2_InitStruct.TIM_ClockDivision = TIM_CKD_DIV1;
tim2_InitStruct.TIM_RepetitionCounter = 0;
TIM_TimeBaseInit(TIM2, &tim2_InitStruct);
TIM_ClearFlag(TIM2, TIM_FLAG_Update);
TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);//Enables the specified TIM interrupts.
這段配置原本以為定時時間是2s,實際只有1s??戳耸謨圆爬斫庠颉?/p>
摘自:http://blog.csdn.net/marike1314/article/details/5673522
STM32 RCC復(fù)位與時鐘配置,我首先忽略掉復(fù)位,首先學(xué)習(xí)時鐘配置,復(fù)位以后用到再學(xué)習(xí)
STM32有多個時鐘源,分別是
HSI:上電默認(rèn)啟動,因精度不高所以先不采用,以后如果需要再使用
HSE:外部高速時鐘,系統(tǒng)時鐘一般采用它,經(jīng)過PLL倍頻作為系統(tǒng)同時鐘
LSE:外部低速時鐘,一般專門用于RTC,等到RTC模塊時再使用
LSI:內(nèi)部低速時鐘,精度不高,一般用于IWDGCLK
時鐘系統(tǒng)框圖如下:
STM32中各個模塊都有自己的時鐘,當(dāng)使用相應(yīng)的模塊時首先記得把此模塊時鐘開啟
本次學(xué)習(xí)使用標(biāo)準(zhǔn)固件庫3.3.0
好了,看明白上圖咱就開始吧:
void RCC_Configuration(void)
{
ErrorStatus HSEStartUpStatus;
//SystemInit();//完全可以使用此函數(shù)配置,但是為了學(xué)習(xí)咱先不用
RCC_DeInit();//復(fù)位RCC模塊的寄存器,復(fù)位成缺省值
RCC_HSEConfig(RCC_HSE_ON); //開啟HSE時鐘,咱是用HSE的時鐘作為PLL的時鐘源
HSEStartUpStatus = RCC_WaitForHSEStartUp();//獲取HSE啟動狀態(tài)
if(HSEStartUpStatus == SUCCESS)//如果HSE啟動成功
{
FLASH_PrefetchBufferCmd(ENABLE);//開啟FLASH的預(yù)取功能
FLASH_SetLatency(FLASH_Latency_2); //FLASH延遲2個周期(這里我也不明白,先用吧)
RCC_HCLKConfig(RCC_SYSCLK_Div1); //配置HCLK,PCLK2,PCLK1,PLL
RCC_PCLK2Config(RCC_HCLK_Div1);
RCC_PCLK1Config(RCC_HCLK_Div2);
RCC_PLLConfig(RCC_PLLSource_HSE_Div1,RCC_PLLMul_9);
RCC_PLLCmd(ENABLE);//啟動PLL
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY)==RESET)
{}//等待PLL啟動完成
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //配置系統(tǒng)時鐘
while(RCC_GetSYSCLKSource() !=0x80) //檢查是否將HSE 9倍頻后作為系統(tǒng)時鐘
{}
}
}
設(shè)置時鐘流程:
1.將RCC寄存器重新設(shè)置為默認(rèn)值 RCC_DeInit
2.打開外部高速時鐘晶振HSE RCC_HSEConfig(RCC_HSE_ON);
3.等待外部高速時鐘晶振工作 HSEStartUpStatus = RCC_WaitForHSEStartUp();
4.設(shè)置AHB時鐘 RCC_HCLKConfig;
5.設(shè)置高速APB2時鐘RCC_PCLK2Config;
6.設(shè)置低速速APB1時鐘RCC_PCLK1Config
7.設(shè)置PLL RCC_PLLConfig
8.打開PLL RCC_PLLCmd(ENABLE);
9.等待PLL工作 while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)
10.設(shè)置系統(tǒng)時鐘 RCC_SYSCLKConfig
11.判斷是否PLL是系統(tǒng)時鐘 while(RCC_GetSYSCLKSource() != 0x08)
12.打開要使用的外設(shè)時鐘 RCC_APB2PeriphClockCmd()/RCC_APB1PeriphClockCmd()
至此我們就將STM32的系統(tǒng)時鐘配置好了,系統(tǒng)時鐘72MHz,APH 72MHz,APB2 72MHz,APB1 32MHz,USB 48MHz
其他至于ADC什么的先用不管,用到時再設(shè)置,本次只是大體先熟悉下STM32的時鐘配置流程,便于以后程序的編寫