001_配置时钟_stm32

不同开发板使用的外部晶振可能都是不同的,这样通过倍频器调出来的主频并不是我们想要的,会出现串口乱码,延时不准确,外部接口用不了等现象,因此需要进行修改

一些能获取当前时钟源及总线频率的程序

RCC_ClocksTypeDef RCC_CLK;//写在main()的最前面,定义要在赋值前面
RCC_GetClocksFreq(&RCC_CLK);//Get chip frequencies
printf("System Clock Source : %d\r\n", RCC_GetSYSCLKSource());
printf("APB1/PCLK1 : %dHZ\r\n", RCC_CLK.PCLK1_Frequency);
printf("APB2/PCLK2 : %dHZ\r\n", RCC_CLK.PCLK2_Frequency);
printf("SYSCLK     : %dHZ\r\n", RCC_CLK.SYSCLK_Frequency);
printf("HCLK       : %dHZ\r\n", RCC_CLK.HCLK_Frequency);

stm32时钟配置

在system_stm32f10x.c文件中配置宏定义之后,在此.c文件中的相应的函数才能使用,例如static void SetSysClockTo72(void)函数体,

HSE_VALUE 会在stm32f10x_rcc.c中赋值给RCC_Clocks->SYSCLK_Frequency,因此必须设置准确,按照真实的外置高速晶振来

static void SetSysClockTo72(void)这种函数体本质上就是对RCC_CFGR寄存器进行配置,在库函数里面关于CFGR主要是设置了HCLK、APB1和APB2的时钟频率,还有通过寄存器后两位选择HSE/HSI/PLL这三个其中一个作为系统时钟。

image-20221216184115162

有些工程文件是通过调用SetSysClock( )函数体来间接调用static void SetSysClockTo72(void)的,完成RCC的配置。如果没有调用,则修改这里的代码并没有用。此外,上述函数SetSysClock( )void SystemInit (void)中被间接调用

除此之外,还可以自己写一个void RCC_Configuration(void)函数体来配置RCC时钟,如下所示(这里会初始化RCC寄存器,这样之前stm32f10x_rcc.c启动文件中的代码则会失效,需要在RCC_Configuration中重新配置):

void RCC_Configuration(void){ //RCC时钟的设置  
	ErrorStatus HSEStartUpStatus;   
	RCC_DeInit();              /* RCC system reset(for debug purpose) RCC寄存器恢复初始化值*/   
	RCC_HSEConfig(RCC_HSE_ON); /* Enable HSE 使能外部高速晶振*/   
	HSEStartUpStatus = RCC_WaitForHSEStartUp(); /* Wait till HSE is ready 等待外部高速晶振使能完成*/   
	if(HSEStartUpStatus == SUCCESS){   
		/*设置PLL时钟源及倍频系数*/   
		RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9); //RCC_PLLMul_x(枚举2~16)是倍频值。当HSE=8MHZ,RCC_PLLMul_9时PLLCLK=72MHZ 
		//RCC_PLLConfig(RCC_PLLSource_HSE_Div2, RCC_PLLMul_9);		//HSE=16MHZ,16/2*9
		/*设置AHB时钟(HCLK)*/   
		RCC_HCLKConfig(RCC_SYSCLK_Div1); //RCC_SYSCLK_Div1——AHB时钟 = 系统时钟(SYSCLK) = 72MHZ(外部晶振8HMZ)   
		/*注意此处的设置,如果使用SYSTICK做延时程序,此时SYSTICK(Cortex System timer)=HCLK/8=9MHZ*/   
		RCC_PCLK1Config(RCC_HCLK_Div2); //设置低速AHB时钟(PCLK1),RCC_HCLK_Div2——APB1时钟 = HCLK/2 = 36MHZ(外部晶振8HMZ)   
		RCC_PCLK2Config(RCC_HCLK_Div1); //设置高速AHB时钟(PCLK2),RCC_HCLK_Div1——APB2时钟 = HCLK = 72MHZ(外部晶振8HMZ)   
		/*注:AHB主要负责外部存储器时钟。APB2负责AD,I/O,高级TIM,串口1。APB1负责DA,USB,SPI,I2C,CAN,串口2,3,4,5,普通TIM */  
		FLASH_SetLatency(FLASH_Latency_2); //设置FLASH存储器延时时钟周期数   
		/*FLASH时序延迟几个周期,等待总线同步操作。   
		推荐按照单片机系统运行频率:
		0—24MHz时,取Latency_0;   
		24—48MHz时,取Latency_1;   
		48~72MHz时,取Latency_2*/   
		FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //选择FLASH预取指缓存的模式,预取指缓存使能   
		RCC_PLLCmd(ENABLE);	//使能PLL
		while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); //等待PLL输出稳定   
		RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //选择SYSCLK时钟源为PLL
		while(RCC_GetSYSCLKSource() != 0x08); //等待PLL成为SYSCLK时钟源   
	}  
	/*开始使能程序中需要使用的外设时钟*/   
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB |   
	RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD | RCC_APB2Periph_GPIOE, ENABLE); //APB2外设时钟使能      

参考资料

1. STM32-时钟启动的两个寄存器(RCC_CFGR)(PLL_CFGR)的配置
2.正点原子STM32F1开发指南

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值