STM芯片的所有片上外設(shè)都需要手動設(shè)置時鐘,三種不同的時鐘源可被用來驅(qū)動系統(tǒng)時鐘(SYSCLK):
HSI振蕩器時鐘:由內(nèi)部8MHz的RC振蕩器產(chǎn)生,可直接作為系統(tǒng)時鐘或在2分頻后作為PLL輸入。HSI RC振蕩器能夠在不需要任何外部器件的條件下提供系統(tǒng)時鐘。它的啟動時間比HSE晶體振蕩器短。然而,即使在校準(zhǔn)之后它的時鐘頻率精度仍較差。(所以通常不用與提供SYSCLK)
HSE振蕩器時鐘:高速外部時鐘信號,由HSE外部晶體/陶瓷諧振器(較常用)或者HSE用戶外部時鐘兩種方式產(chǎn)生
PLL時鐘:時鐘源輸入,內(nèi)部PLL可以用來倍頻HSI RC的輸出時鐘或HSE晶體輸出時鐘,一旦PLL被激活,這些參數(shù)就不能被改動。
二級時鐘源:當(dāng)不被使用時,任一個時鐘源都可被獨立地啟動或關(guān)閉,由此優(yōu)化系統(tǒng)功耗。
LSI 時鐘:40kHz低速內(nèi)部RC,可以用于驅(qū)動獨立看門狗和通過程序選擇驅(qū)動RTC。RTC用于從停機/待機模式下自動喚醒系統(tǒng)
LSE 時鐘:32.768kHz低速外部晶體也可用來通過程序選擇驅(qū)動RTC(RTCCLK)。
配置時鐘流程:
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()
源代碼:
void RCC_Configuration(void)
{
ErrorStatus HSEStartUpStatus;
RCC_DeInit();
RCC_HSEConfig(RCC_HSE_ON);
HSEStartUpStatus = RCC_WaitForHSEStartUp();
if(HSEStartUpStatus == SUCCESS)
{
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_16); // PLL的輸入時鐘 = HSE時鐘頻率;RCC_PLLMul_16——PLL輸入時鐘x 16 ;HSE =4MHZ,所以PLLCLK=64MHZ
RCC_HCLKConfig(RCC_SYSCLK_Div1); //RCC_SYSCLK_Div1——AHB時鐘 = 系統(tǒng)時鐘(SYSCLK) = 64MHZ(外部晶振4HMZ)
RCC_PCLK1Config(RCC_HCLK_Div2); //RCC_HCLK_Div2——APB1時鐘 = HCLK/2 = 32MHZ(外部晶振4HMZ)
RCC_PCLK2Config(RCC_HCLK_Div1); //RCC_HCLK_Div1——APB2時鐘 = HCLK = 64MHZ(外部晶振4HMZ)
//注:AHB主要負(fù)責(zé)外部存儲器時鐘。
// APB2負(fù)責(zé)AD,I/O,高級TIM,串口1
// APB1負(fù)責(zé)DA,USB,SPI,I2C,CAN,串口2,3,4,5,普通TIM
FLASH_SetLatency(FLASH_Latency_2); //FLASH_Latency_1 1延時周期
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); // 預(yù)取指緩存使能
RCC_PLLCmd(ENABLE);
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)
{
}
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
while(RCC_GetSYSCLKSource() != 0x08)
{
}
}//end if(HSEStartUpStatus == SUCCESS)
// Enables or disables the High Speed APB(APB2) peripheral clock,APB2外設(shè)時鐘使能
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB |
RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD | RCC_APB2Periph_GPIOE, ENABLE);
//Enables or disable the High Speed APB(APB1) peripheral clock,APB1外設(shè)時鐘使能
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
}