【STM32F103】RCC-使用HSE/HSI配置时钟

时间:2024-04-03 17:12:03

【STM32F103】RCC-使用HSE/HSI配置时钟

 

【STM32F103】RCC-使用HSE/HSI配置时钟

【STM32F103】RCC-使用HSE/HSI配置时钟

 

【STM32F103】RCC-使用HSE/HSI配置时钟

【STM32F103】RCC-使用HSE/HSI配置时钟

【STM32F103】RCC-使用HSE/HSI配置时钟

【STM32F103】RCC-使用HSE/HSI配置时钟

【STM32F103】RCC-使用HSE/HSI配置时钟

【STM32F103】RCC-使用HSE/HSI配置时钟

【STM32F103】RCC-使用HSE/HSI配置时钟

 以上为STM32的时钟树,通过系统的启动文件,系统自动调用函数进行时钟初始化,所以我们在实际应用时不需要去写时钟这块,但是如果我们有特殊需求或者想要修改内部时钟的属性,则需要了解时钟树。以下以一个实验例子来测试,修改系统时钟:

1、通过HSE(高速外部时钟)来配置系统时钟:

//HSE配置系统时钟
void HSE_SetSysClk( uint32_t RCC_PLLMul_x )
{
	ErrorStatus HSEStatus;
	
	// 把RCC 寄存器复位成复位值
	RCC_DeInit();	

	// 使能 HSE 
	RCC_HSEConfig(RCC_HSE_ON);
	
	HSEStatus = RCC_WaitForHSEStartUp();
	
	if( HSEStatus == SUCCESS )
	{
		// 使能预取指
		FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
		FLASH_SetLatency(FLASH_Latency_2);
		
		RCC_HCLKConfig(RCC_SYSCLK_Div1);
		RCC_PCLK1Config(RCC_HCLK_Div2);
		RCC_PCLK2Config(RCC_HCLK_Div1);
		
		// 配置 PLLCLK = HSE * RCC_PLLMul_x
    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_x);
		
    // 使能PLL
		RCC_PLLCmd(ENABLE);
		
		// 等待PLL稳定
		while( RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET );
		
    // 选择系统时钟
		RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
		
    while( RCC_GetSYSCLKSource() != 0x08 );
	}
	else
  {
		/* 如果HSE 启动失败,用户可以在这里添加处理错误的代码 */
	}
}

 2、HSI(高速内部时钟)来配置系统时钟

void HSI_SetSysClk( uint32_t RCC_PLLMul_x )
{
	__IO uint32_t HSIStatus = 0;
	
	// 把RCC 寄存器复位成复位值
	RCC_DeInit();	

	// 使能 HSI 
	RCC_HSICmd(ENABLE);
	
	HSIStatus = RCC->CR & RCC_CR_HSIRDY;
	
	if( HSIStatus == RCC_CR_HSIRDY )
	{
		// 使能预取指
		FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
		FLASH_SetLatency(FLASH_Latency_2);
		
		RCC_HCLKConfig(RCC_SYSCLK_Div1);
		RCC_PCLK1Config(RCC_HCLK_Div2);
		RCC_PCLK2Config(RCC_HCLK_Div1);
		
		// 配置 PLLCLK = HSE * RCC_PLLMul_x
    RCC_PLLConfig(RCC_PLLSource_HSI_Div2, RCC_PLLMul_x);
		
    // 使能PLL
		RCC_PLLCmd(ENABLE);
		
		// 等待PLL稳定
		while( RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET );
		
    // 选择系统时钟
		RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
		
    while( RCC_GetSYSCLKSource() != 0x08 );
	}
	else
  {
		/* 如果HSI 启动失败,用户可以在这里添加处理错误的代码 */
	}
}

 3、MCO GPIO初始化,对外提供时钟输出,有条件也可以用示波器来检查该引脚(PA8)

void MCO_GPIO_Config()
{
	GPIO_InitTypeDef  GPIO_InitStruct;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_8;
	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;
	GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	
	GPIO_Init(GPIOA, &GPIO_InitStruct);	
}

 4、设置软件延时

void Delay( uint32_t count )
{
	for(; count!=0; count--);
}

 5、主函数测试

int main(void)
{
	// 来到这里的时候,系统的时钟已经被配置成72M。
	
	//HSE_SetSysClk( RCC_PLLMul_9 );
	//HSI_SetSysClk( RCC_PLLMul_16 );
	
	MCO_GPIO_Config();
	RCC_MCOConfig(RCC_MCO_SYSCLK);
	//RCC_MCOConfig(RCC_MCO_HSI);
	
	LED_GPIO_Config();
	
	while(1)
	{
		
		LED_G(OFF);
		Delay(0xFFFFF);
		
		LED_G(ON);
		Delay(0xFFFFF);
	}
}

我们将之前的LED程序也放进去,因为没有示波器,通过检查LED闪烁的频率来查看MCO输出的频率,即系统时钟的频率是否得到了修改。

MCO=SYSCLK=72M时:(不需要修改程序,因为是启动文件自动调用时钟初始化获得72M时钟)

【STM32F103】RCC-使用HSE/HSI配置时钟

可以看到实验效果(不方便展示)是LED连续闪烁,速度很快。

MCO=HSI 且使用HSI时,时钟频率=4Mhz*RCC_PLLMul_x,这里我们取x=4,即频率=16MHz

【STM32F103】RCC-使用HSE/HSI配置时钟

观察到LED闪烁的速率较慢,实际用示波器还可以看的更精准,HSE就不展示了,效果一样,程序将在后续上传上来。