4.配置系统时钟思路及方法

 前言:

         比起之前用过的三星的猎户座4412芯片,STM32F4的系统时钟可以说是小巫见大巫,首先我们需要清晰时钟产生的原理:几乎大多数的芯片都是由晶振产生一个比较低频的频率,然后通过若干个PLL得到单片机能承受的频率(作为主频),再通过其他手段将PLL出来的频率降频分给其他外设使用。一个时钟树一般先对复杂,我们先调出主频(及编程好时钟源、PLL倍频这一部分)其他的之后再说,如此编程才不会太复杂。

        实际上这个一般厂家会给一个配置文件的,但是如果要自己做些超频之类的操作,就要彻底掌握时钟树的配置了,见人见智,追求技术的这个内容是逃不掉的。

        编程思路:1.PLL倍频因子配置 2..PLL时钟源激活和切换(上电后单片机会选择一个默认的时钟源,可能是晶振也可能是内部RC电路产生的频率) 3.切换系统时钟

时钟资源概览:

        下面先看一下我们这个F4的系统时钟资源,查看手册可知系统复位后是默认选择HSI这个内部RC电路产生的时钟作为这个单片机的系统时钟,但是我们要的是PLL产生的时钟。

        下面看手册PLL配置的说明(可以把时钟树截图出来作参考,不过主要编程还是靠手册的文字描述),可知RCC_PLLCFGR 可以用来配置PLL (PLLI2S可以先不管,先搞出主频再说),那就配置它吧。寄存器就不放出来了,自己看手册,这里给出寄存器各个位的配置值及解释

RCC_PLLCFGR寄存器配置

可见:
PLL = VCO / PLLP
VCO = PLL时钟源*(PLLN / PLLM)
一共涉及P / N / M 三个因子,以及PLL时钟源。
PLL就是我们要选的系统时钟,PLL = 168M,如果PLLP选的是2,那么VCO就得是168*2 = 336 。VCO是336,我们的PLL时钟源如果选的是外部晶振(探索者这个开发板上晶振是8M),那么PLLN/PLLM就得等于 VCO / PLL时钟源 = 336 / 8 = 42 ,所以PLLN除以PLLM必须是等于42,PLLN配置为336(可以在192和432这个数值间任意取),则PLLM配置为 336 / 42 = 8 .分频因子就搞定了。
下面开始配置 RCC_PLLCFGR
bit[5:0]:    设置PLLM为8,即0x8<<0
bit[14:6]:  设置PLLN为336,即336<<6
bit[17:16]:设置PLLP为2,即0<<16
bit[22]:     设置PLL和PLLI2S时钟源,设为1<<22,选择HES晶振
bit[27:24]:设置USB OTG之类的时钟,随便设一个,0x7<<24
其他使用默认值,编程如下:
RCC->PLLCFGR = 0x24003010 ;//复位值
RCC->PLLCFGR = 0x7<<24 | 1<<22 | 0<<16 | 336<<6 | 0x8<<0 ;
这样PLL就配置好了,PLL的时钟源我们选择的是HES,它还没有激活,所以 在PLL配置这一步之前,还需要将HSE激活

RCC_CR寄存器配置

查看手册HSE部分,可知要换时钟源要操作RCC_CR寄存器,时钟中断我们不需要。
同样,配置寄存器RCC_CR:(真真苦力活~)
bit[0]:    HSI的开关,这个呢暂时还不能关闭,要让HSE生效之后才能关闭,不然单片机一个时钟都没有没法工作。
bit[1]:    HSI的状态 1是ok,0是不ok
bit[16]:  HSE的开关,置为1<<16,打开
bit[17]:  HSE的状态,它就绪之后才能配置PLL,配置完PLL才能切换系统时钟(HSI时钟---->>PLL时钟)
bit[18]:  HSE时钟旁路,这个要关掉,因为我们要用的是HSE晶振,配置为0<<18
bit[24]:  PLL的开关,这个还没配置好之前要关掉
bit[25]:PLL是否稳定的标志
其他不管,RCC->CR复位时的默认值是 让HSI正常工作的,其他都是0,所以不改动它原有的,在它原有的值基础上进行幅值
1.配置使得HSE开始工作:
RCC->CR | = 1<<16;
u16 retry=0;//这个只是提供短暂延时的变量
while(((RCC->CR&(1<<17))==0)&&(retry<0X1FFF))retry++;//跳出循环后说明HSE ok了
if(retry==0X1FFF)status=1;    //当然如果超过了一定时间也会跳出,表示HSE无法就绪

2.打开PLL并等待其稳定

RCC->CR|=1<<24;			//打开主PLL
while((RCC->CR&(1<<25))==0);//等待PLL准备好 

使PLL倍频出很高的频率

有了上面的东西,我们就可以使PLL倍频出很高的频率了,结合上面两个寄存器:

RCC->PLLCFGR = 0x7<<24 | 1<<22 | 0<<16 | 336<<6 | 0x8<<0 ;//配置PLL倍频因子RCC->CR | = 1<<16;//激活HSE晶振
u16 retry=0;//这个只是提供短暂延时的变量
while(((RCC->CR&(1<<17))==0)&&(retry<0X1FFF))retry++;//跳出循环后说明HSE ok了
if(retry==0X1FFF)status=1; else{//激活HSE完成了RCC->CR|=1<<24;			//打开主PLLwhile((RCC->CR&(1<<25))==0);//PLL稳定}

现在PLL理论上已经有了晶振倍频后的频率了,下面切换PLL作为系统时钟:

切换PLL作为系统时钟

查看手册,知道RCC_CFGR是管这个事的:再一波嘎嘎配置

bit[1:0]:0x2<<0        切换PLL作为系统时钟

bit[3:2]:这两个位可以读出是否切换完成,如果读出来是0x2就是切换成PLL成功

bit[7:4]:这四个位是配置AHB分频的,我记得是不分频的,设为0000 即0x0<<4

bit[12:10]: 这三个位是配置APB1分频的 设为4分频,即0x5<<10

bit[15:13]:这三个位是配置APB2分频的 设为2分频,即0x4<<13

bit[20:16]:这五个位是配置RTC分频的 可以先随便设一个设为HSE/2,即0x2<<16

其他不用管


 

RCC_CFGR = 0;//清零RCC_CFGR = 0x2<<16 | 0x4<<13 | 0x5<<10 | 0x0<<4 | 0x2<<0 ;//切换PLL为系统时钟并且设置其他分频while((RCC->CFGR&(3<<2))!=(2<<2));//等待主PLL作为系统时钟成功. 

这样,综合上面所有的代码就是:

    u16 retry=0;//这个只是提供短暂延时的变量u8 status=0;   //按照上面的分析思路,编程流程就是://1.配置PLL倍频因子RCC->PLLCFGR = 0x7<<24 | 1<<22 |0<<16 |336<<6 |0x8<<0 ;//2.激活HSE晶振RCC->CR |= 1<<16;while(((RCC->CR&(1<<17))==0)&&(retry<0X1FFF))retry++;//跳出循环后说明HSE ok了if(retry==0X1FFF)status=1;else{//激活HSE完成了//3.打开PLL等待PLL输出稳定RCC->CR|=1<<24;			while((RCC->CR&(1<<25))==0);//4.切换PLL输出为系统时钟RCC->CFGR = 0;//清零RCC->CFGR =0x2<<16 |0x4<<13 | 0x5<<10 | 0x0<<4 |0x2<<0;//切换PLL为系统时钟并且设置其他分频while((RCC->CFGR&(3<<2))!=(2<<2));//等待主PLL作为系统时钟成功.现在主频是168M了}

验证测试:

下面可以用串口来打印,验证是不是设置完成。

可见是ok了的~说明上面的系统时钟配置没问题。main函数

疑难杂症:

如果你的整个main函数是这样的:是配置完时钟后也是没法正常工作的

#include "sys.h"
#include "usart.h" 
#include "delay.h" u16 myconut;
//systick中断服务函数,使用OS时用到
void SysTick_Handler(void)
{	myconut++;if(myconut>=1000){myconut=0;printf("hello\r\n");}}int main(void)
{ u8 t=0;u16 retry=0;//这个只是提供短暂延时的变量u8 status=0;//按照上面的分析思路,编程流程就是://1.配置PLL倍频因子RCC->PLLCFGR = 0x7<<24 | 1<<22 |0<<16 |336<<6 |0x8<<0 ;//2.激活HSE晶振RCC->CR |= 1<<16;while(((RCC->CR&(1<<17))==0)&&(retry<0X1FFF))retry++;//跳出循环后说明HSE ok了if(retry==0X1FFF)status=1;else{//激活HSE完成了//3.打开PLL等待PLL输出稳定RCC->CR|=1<<24;			while((RCC->CR&(1<<25))==0);//4.切换PLL输出为系统时钟RCC->CFGR = 0;//清零RCC->CFGR =0x2<<16 |0x4<<13 | 0x5<<10 | 0x0<<4 |0x2<<0;//切换PLL为系统时钟并且设置其他分频while((RCC->CFGR&(3<<2))!=(2<<2));//等待主PLL作为系统时钟成功.现在主频是168M了}delay_init(168);		//初始化延时函数NVIC_SetPriorityGrouping(2);SysTick_Config(168000);//1ms中断一次NVIC_EnableIRQ(SysTick_IRQn);uart_init(84,115200);	//串口初始化为115200while(1){}
}

原因是缺少了这样几行关于CPU的代码:将它加在时钟配置代码的上方即可正常运行了

    	FLASH->ACR|=1<<8;		//指令预取使能.FLASH->ACR|=1<<9;		//指令cache使能.FLASH->ACR|=1<<10;		//数据cache使能.FLASH->ACR|=5<<0;		//5个CPU等待周期. 

        正点原子是把它放在时钟配置里的,我也不知道为啥,但是我觉得它和时钟配置是没什么关系的,应该是另一部分的知识。正点原子时钟配置中还有这样两句关于电源的代码,我实测去掉也是可以的,不过应该还是加上比较好,但是时钟配置的部分手册没有提到,我也就没有在上面说,以免它出现的很突兀。同样要加的话加在时钟配置代码之前即可。

		RCC->APB1ENR|=1<<28;	//电源接口时钟使能PWR->CR|=3<<14; 		//高性能模式,时钟可到168Mhz

完事了~系统时钟就是这样配置啦,这个算是简单的,像能跑linux的那种芯片,就得依靠厂家给的来写或者修改了,自己写的总有不到位的地方~

整个main.c代码如下:

#include "sys.h"
#include "usart.h" 
#include "delay.h" 
//ALIENTEK 探索者STM32F407开发板 实验0
//新建工程实验  
//技术支持:www.openedv.com
//广州市星翼电子科技有限公司
u16 myconut;
//systick中断服务函数,使用OS时用到
void SysTick_Handler(void)
{	myconut++;if(myconut>=1000){myconut=0;printf("hello\r\n");}}int main(void)
{ u8 t=0;//plln,pllm,pllp,pllq//Stm32_Clock_Init(336,8,2,7);//设置时钟,168Mhzu16 retry=0;//这个只是提供短暂延时的变量u8 status=0;//CPU相关的初始化FLASH->ACR|=1<<8;		//指令预取使能.FLASH->ACR|=1<<9;		//指令cache使能.FLASH->ACR|=1<<10;		//数据cache使能.FLASH->ACR|=5<<0;		//5个CPU等待周期. //电源相关的初始化RCC->APB1ENR|=1<<28;	//电源接口时钟使能PWR->CR|=3<<14; 		//高性能模式,时钟可到168Mhz//按照博客的分析思路,系统时钟配置的编程流程就是://1.配置PLL倍频因子RCC->PLLCFGR = 0x7<<24 | 1<<22 |0<<16 |336<<6 |0x8<<0 ;//2.激活HSE晶振RCC->CR |= 1<<16;while(((RCC->CR&(1<<17))==0)&&(retry<0X1FFF))retry++;//跳出循环后说明HSE ok了if(retry==0X1FFF)status=1;else{//激活HSE完成了//3.打开PLL等待PLL输出稳定RCC->CR|=1<<24;			while((RCC->CR&(1<<25))==0);//4.切换PLL输出为系统时钟RCC->CFGR = 0;//清零RCC->CFGR =0x2<<16 |0x4<<13 | 0x5<<10 | 0x0<<4 |0x2<<0;//切换PLL为系统时钟并且设置其他分频while((RCC->CFGR&(3<<2))!=(2<<2));//等待主PLL作为系统时钟成功.现在主频是168M了}delay_init(168);		//初始化延时函数NVIC_SetPriorityGrouping(2);SysTick_Config(168000);//1ms中断一次NVIC_EnableIRQ(SysTick_IRQn);uart_init(84,115200);	//串口初始化为115200while(1){}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/283870.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

模拟IC设计-学习笔记

写在前面&#xff1a;本文是我自己学习b站上up主(jrilee讲电路)的模集课程时记的笔记&#xff0c;原课程&#xff1a;模拟IC设计 第一讲 -- Device and modeling (I)_哔哩哔哩_bilibili 目录 器件基础 求输出电阻 MOS管的小信号模型&#xff0c;拉扎维书上常用的是π模型&am…

Flutter ios 使用ListView 。滚动时 AppBar 改变颜色问题

在Ios 中 列表滚动条向下滚动一段距离后 会导致 AppBar 颜色改变 可以给 AppBar 或者 AppBarTheme。 scrolledUnderElevation: 0.0 属性 全局&#xff1a; MaterialApp(theme: ThemeData(appBarTheme: AppBarTheme(scrolledUnderElevation: 0.0)) ) 局部&#xff1a; App…

硬件基础-电阻

电阻 1.品牌 厚声、风华&#xff0c;三星、罗姆、松下、KOA 2.分类 插件 碳膜电阻&#xff1a;精度-5 J 是在高阻&#xff0c;高压和高温应用中 属负温度系数电阻 金属膜&#xff1a;-1 F 贴片 电阻标识&#xff1a;&#xff08;含义&#xff1a;阻值大小和精度&a…

keithley 吉时利6221源表

特点 优势 10 14 Ω 输出阻抗 提供广泛的输出阻抗&#xff0c;确保负载中有稳定的电流源。 65000 点源内存 允许直接从电流源执行全面的测试电流扫描。 输出 0.1V 至 105V 的恒流电压&#xff0c;10mV 步长 防止潜在损坏对过电压敏感的设备。 源交流电源范围为 4pA 至…

AI聊天伴侣的语料采集大揭秘:OpenCV如何轻松识别聊天图片?

最近&#xff0c;负责元宇宙中AI聊天伴侣的语料数据采集&#xff0c;这些数据主要用于AI虚拟角色聊天的训练和测试。虽然语料获取有多种渠道&#xff0c;但由于部分数据涉及隐私&#xff0c;这里就不多说了&#xff08;感兴趣的朋友可以私聊我&#xff09;。今天&#xff0c;我…

权重衰减weight_decay

查了好几次了&#xff0c;一直忘&#xff0c;记录一下 使用L 2 范数的一个原因是它对权重向量的大分量施加了巨大的惩罚。这使得我们的学习算法偏向于在大量特征上均匀分布权重的模型。在实践中&#xff0c;这可能使它们对单个变量中的观测误差更为稳定。 相比之下&#xff0c…

linux gdb反汇编定位问题

日前解决一现网问题遇到补丁加载未生效现象&#xff0c;想要验证流程是否走进补丁代码&#xff0c;由于补丁函数和原函数名称一样&#xff0c;且修改代码较少&#xff0c;通过普通gdb方法难以看出是否走进补丁&#xff0c;但可用gdb反汇编方法来验证。 gdb该进程&#xff0c;之…

干货下载丨不分业态、不关注核心需求,怎么做得好项目管理?!

项目管理&#xff1a;装备制造业的破局利刃 对于装备制造行业而言&#xff0c;每一笔订单都是非标定制、小批量制造。这种特性决定了其行业企业普遍存在新品开发周期长、生产效率低、质量不稳定、交货期不稳定、成本预算难控制、非标品报价慢等问题。 如何提升企业的管理水平…

【TOP解刊】IEEE旗下中科院2区,无需版面费,IF持续上涨,见刊快,值得投吗?

工程综合类 • 好刊解读 今天小编带来IEEE旗下工程领域高分区好刊&#xff0c;如您有投稿需求&#xff0c;可作为重点关注&#xff01;后文有相关领域真实发表案例&#xff0c;供您投稿参考~ 01 期刊简介 IEEE Sensors Journal ✅出版社&#xff1a;IEEE ✅ISSN&#xff1a…

第15章 《乐趣》Page355~375 纹理,渲染器,代码简化版

运行效果&#xff1a;全屏了 简化之后的代码如下&#xff1a; //main.cpp #include <iostream> #include <SDL2/SDL.h> #include "sdl_initiator.hpp" #include "sdl_error.hpp" #include "sdl_window.hpp" #include "sdl_sur…

【日常总结】连接Mysql,打开数据表非常慢

问题 Navicat 连接mysql时&#xff0c;第二次打开非常慢 原因 Mysql服务器端会定时清理长时间不活跃空闲的数据库连接&#xff0c;以此优化数据库的性能。 解决方案 数据库右键---编辑连接--高级---保持连接间隔30秒 带来的问题 每次打开Navicat时&#xff0c;设置设置自动…

【论文解读】Kvazaar 2.0: Fast and Efficient Open-Source HEVC Inter Encoder

时间&#xff1a;2020 级别&#xff1a;SCI 机构&#xff1a;Tampere University 摘要&#xff1a;高效视频编码(HEVC)是当前多媒体应用中经济的视频传输和存储的关键&#xff0c;但解决其固有的计算复杂性需要强大的视频编解码器实现。本文介绍了Kvazaar 2.0 HEVC编码器&…