【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时钟)

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

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

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

 

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
STM32F103R6是一款基于ARM Cortex-M3内核的微控制器,它具有丰富的外设和功能。在配置STM32F103R6的时钟时,需要设置系统时钟和外设时钟。 1. 系统时钟配置: - 首先,需要选择系统时钟源。可以选择内部时钟源(HSI)或外部时钟源(HSE)。 - 如果选择内部时钟源,可以使用默认的内部高速时钟HSI)作为系统时钟源。可以通过设置RCC_CFGR寄存器来配置HSI的分频系数。 - 如果选择外部时钟源,需要将外部时钟源连接到MCU的引脚上,并通过设置RCC_CFGR寄存器来配置外部时钟源的分频系数。 - 然后,需要选择系统时钟的分频系数。可以通过设置RCC_CFGR寄存器来配置分频系数,以得到所需的系统时钟频率。 2. 外设时钟配置: - 对于每个外设,都有一个时钟使能寄存器(RCC_APBxENR或RCC_AHBxENR),用于控制外设时钟的使能和禁用。 - 通过设置相应的时钟使能位,可以使能或禁用特定的外设时钟。 下面是一个示例代码,演示了如何配置STM32F103R6的时钟: ```c #include "stm32f10x.h" void SystemClock_Config(void) { // 选择系统时钟源为外部时钟源(HSERCC->CFGR |= RCC_CFGR_SW_HSE; // 配置外部时钟源的分频系数 RCC->CFGR |= RCC_CFGR_HPRE_DIV1; // AHB时钟不分频 RCC->CFGR |= RCC_CFGR_PPRE1_DIV2; // APB1时钟分频系数为2 RCC->CFGR |= RCC_CFGR_PPRE2_DIV1; // APB2时钟不分频 // 使能外部时钟源(HSERCC->CR |= RCC_CR_HSEON; while(!(RCC->CR & RCC_CR_HSERDY)); // 使能外设时钟 RCC->APB1ENR |= RCC_APB1ENR_TIM2EN; // 使能TIM2时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN; // 使能GPIOA时钟 } int main(void) { // 配置系统时钟 SystemClock_Config(); // 其他代码... return 0; } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值