一、STM32时钟设置函数移植
1.时钟模块回顾
一个疑问
前面代码并没有设置时钟为什么可以直接使用。
2.时钟树
data:image/s3,"s3://crabby-images/ba3f7/ba3f7c82a4c781ad8819ae6c696cb5efed818be1" alt=""
3.时钟树分析
1.内部晶振(HSI)
内部晶振不稳定,当我们上电后,会自动产生振动,自动产生时钟,但是晶振不稳定。
不经过PPLMUL,默认使用8MHZ。所以如果我们想要72MHZ,则需要使用外部晶振
data:image/s3,"s3://crabby-images/c33ff/c33ff0d2fe097acaa251ee22a0010dd346e529de" alt=""
2.外部晶振(HSE)
当接上外部晶振,当接通电源之后,不用软件操作,会自动产生振动。可以进行分频等操作。
data:image/s3,"s3://crabby-images/73dc0/73dc0e6ce2ae5bdcf4466f3b4f01f8b0360e5df0" alt=""
从外部接上外部晶振的时候,我们需要等待一段时间,让其稳定后,才开始工作。(所以要进行判断)
data:image/s3,"s3://crabby-images/dc29a/dc29a8782d0391a57ed12f00c94e72f8c501eb73" alt=""
3.PLLMUL
当上电后,经过他时,要等待一段时间,让其稳定后,才可以开始工作。(所以我们有一个寄存器专门用来判断其是否准备好开始工作,当我们去读取到其准备好了才可以进行下一步)
data:image/s3,"s3://crabby-images/2154a/2154abf4ff8f50ca71887256fbb07986b9b0cbe8" alt=""
data:image/s3,"s3://crabby-images/582f7/582f70609e139dba08c05bfab877f09453919605" alt=""
二、代码移植
cpp
#ifndef __CLOCK_H__
#define __CLOCK_H__
#include "gpio.h"
// 寄存器宏定义
// RCC寄存器基地址为0x40021000
#define RCC_BASE 0x40021000 // RCC部分寄存器的基地址
#define RCC_CR (RCC_BASE + 0x00) // RCC_CR的地址
#define RCC_CFGR (RCC_BASE + 0x04)
#define FLASH_ACR 0x40022000
// 用C语言来访问寄存器的宏定义
#define rRCC_CR (*((volatile unsigned int *)RCC_CR))
#define rRCC_CFGR (*((volatile unsigned int *)RCC_CFGR))
#define rFLASH_ACR (*((volatile unsigned int *)FLASH_ACR))
// 函数作用:时钟源切换到HSE并且使能PLL,将主频设置为72MHz
void Set_SysClockTo72M(void);
#endif
1.复位RCC_CR寄存器
data:image/s3,"s3://crabby-images/62d01/62d016140782081ced61a4678ba4f4ebb95702ab" alt=""
#define rRCC_APB2ENR (*((unsigned int *)RCC_APB2ENR))
RCC->CR就相当于rRCC_APB2ENR
data:image/s3,"s3://crabby-images/c7028/c702801ab4431fe0445a04bf9132e31ad69dd1ee" alt=""
cpp
//复位RCC_CR寄存器
rRCC_CR=0x00000083;
2.开启外部时钟(就是开启外部晶振)
data:image/s3,"s3://crabby-images/e8f3f/e8f3f959f6c3d81e2a6a44039c52c6b0e52a9358" alt=""
&:将某一些位置0
|:将某一些位置1
data:image/s3,"s3://crabby-images/d5199/d51998b67d39205c9aa3874178f398f316bc29cb" alt=""
data:image/s3,"s3://crabby-images/a334d/a334dc2c08ca334abcbbfac07a88ef2a0ce6dc64" alt=""
data:image/s3,"s3://crabby-images/0df9c/0df9c0963231c99429729b3b381b94b6107a779f" alt=""
cpp
//开启外部时钟(外部晶振)
//第一步:先置0【将bit16清零】
rRCC_CR &= ~(1<<16);//关闭HSEON
//第二步:在置1
rRCC_CR |= (1<<16);//打开HSEON,让HSE开始工作
3.检测外部时钟开启是否成功(HSEREDY)
do while十分适合检测是否超时!!!!!!!
data:image/s3,"s3://crabby-images/b751d/b751d74fcf53833ff0c3d6c67dc7883dbf6c8a41" alt=""
data:image/s3,"s3://crabby-images/c35ab/c35aba9fdfd0fc70b4386ee8c81e9869b8e500bd" alt=""
data:image/s3,"s3://crabby-images/63191/631915b8841ba44ffb043d9326085942c3d7705b" alt=""
cpp
do{
//检测HSEREAY(bit17)是否为1,1表示准备好
Rcc_CR_HSE_Ready=rRCC_CR&(1<<17);//取出bit17
faultTime++;
}while((faultTime<0x0fffffff) && (Rcc_CR_HSE_Ready==0))
//跳出do-while 1)要么超时2)要么准好了
4.当准备好进入下一步
data:image/s3,"s3://crabby-images/13599/13599901a8684138e6da7ca3fb83a025dfd8d2af" alt=""
5.Flash的设置
data:image/s3,"s3://crabby-images/5cb55/5cb55292bb6c2227c5ed46b877f1b67471b415f8" alt=""
cpp
rFLASH_ACR |= 0x10;
rFLASH_ACR &= (~0x03);
rFLASH_ACR |= (0x02);
6.对其进行预分频
data:image/s3,"s3://crabby-images/96805/968052dd0bc7473a72c7d7aa00cbe641c88bda3e" alt=""
data:image/s3,"s3://crabby-images/dbb42/dbb42aecee791d3fb19a9fe0cfbc951bdf3ce64a" alt=""
data:image/s3,"s3://crabby-images/d3a50/d3a5037cecdc8f552147a65e8201a41afda706a6" alt=""
cpp
//HPRE【AHB】:对应bit4-bit7:不分频(000)
//PPRE1【APB1】:对应bit8-bit10:进行二分频(100)
//PPRE2【APB2】:对应bit11-bit13:不分频(000)
//AHB和APB2未分频,APB1被2分频
//所以最终:AHB和APB2都是72MHZ,APB1是36MHZ
//第一步:先置0
rRCC_CFGR=(~((0x0f<<4) | (0x07<<8) | (0x07<<11)));
//等价于:rRCC_CFGR=(~(0x3ff<<4));
//第二步:置1
rRCC_CFGR=(((0x0<<4) | (0x04<<8) | (0x0<<11)));
7.设置SHE为输入时钟,同时HSE不分频
data:image/s3,"s3://crabby-images/9ffb2/9ffb2a7b78b6dc6793591f1d192ee345e6213221" alt=""
data:image/s3,"s3://crabby-images/1b4d3/1b4d39ebce7d163e9bbff71d27fa3843b65ef4f9" alt=""
data:image/s3,"s3://crabby-images/f7c80/f7c80e7b5b930763baed95f9cedbcd47864f5bc9" alt=""
data:image/s3,"s3://crabby-images/dbae8/dbae8820a2b8d1bfa72ecc2d7232a60a67ef26ed" alt=""
cpp
//设置为输入时钟:bit16
//设置为不分频:bit17
//第一步:先置0
rRCC_CFGR &=(~((1<<16) | (1<<17)));
//第二步:置1
rRCC_CFGR |= ((1<<18) | (0<<17));
8.设置PLL倍频系数
因为我们在开发板上接上的外部晶振就是8MHZ,如果我们想要在内部使用72MHZ,则需要在内部进行分频率(9倍)
data:image/s3,"s3://crabby-images/6d79e/6d79ea30353be6d1ab52d70f5196d52feefb8e5f" alt=""
data:image/s3,"s3://crabby-images/25a54/25a549bc68f4da58fa848f0f651eb81de5d08db9" alt=""
data:image/s3,"s3://crabby-images/f9cd6/f9cd673a1c6dcfcb25f030defcb30871073f5263" alt=""
data:image/s3,"s3://crabby-images/42107/42107cb471442b87338e0363695976585eb4c47a" alt=""
cpp
//9分频:0111:0x07
rRCC_CFGR &=(~(0x0f<<18));//清零bit18-bit21
rRCC_CFGR |= (0x07<<18);//设置为9倍频
9.打开使能
data:image/s3,"s3://crabby-images/1e5cd/1e5cdaf3a6263ab0265c590d6cd55b354f7d7b1f" alt=""
data:image/s3,"s3://crabby-images/dc3e8/dc3e8bba73ac8ebf87ef8f2532d0e59d49fdc909" alt=""
cpp
//七、打开PLL开关
rRCC_CR |= (1<<24);
10.等待开启PLL开启成功
data:image/s3,"s3://crabby-images/eff49/eff49902b8dce2e2b0da84dd09f19e2f69225348" alt=""
data:image/s3,"s3://crabby-images/81425/81425ca7f30c623c0e70b28d14c6d1572af1d803" alt=""
cpp
//八、等待开启PLL开启成功
do{
Rcc_CR_PLL_Ready=rRcc_CR & (1<<25);//检测第25位是否为1
faultTime++;
}while((faultTime<0x0fffffff) && (Rcc_CR_PLL_Ready==0))
data:image/s3,"s3://crabby-images/b1b8d/b1b8d2e0dcbdf80c0030812e45970fde5dd65196" alt=""
11.将PLL作为SYSCLK的时钟来源
data:image/s3,"s3://crabby-images/4f02f/4f02fe2f584c8f1e143143943b8eceb8e99043b5" alt=""
data:image/s3,"s3://crabby-images/29dc9/29dc983f4c826fbfa4a27b2e379d96ed560a1da6" alt=""
data:image/s3,"s3://crabby-images/6286e/6286e6f6b7c56169f25a992bcf0af95034a17ec7" alt=""
data:image/s3,"s3://crabby-images/a7d36/a7d36a83130f865be1b86217be8261a038700fbb" alt=""
cpp
//到这里说明PLL已经稳定,可以用了,下面可以切换成外部时钟了
rRCC_CFGR &=(~(0x03)<<0);
rRCC_CFGR |=(0x10<<0);
data:image/s3,"s3://crabby-images/a3e5b/a3e5b545d214f2ce3479c7b1fccc1f400fcee9fe" alt=""
12. 判断切换成PLL是否成功
data:image/s3,"s3://crabby-images/e5911/e5911d9ef79353505a6f3791f29b0346cd3e5cc1" alt=""
cpp
do{
RCC_CF_SWS_PLL=rRCC_CFGR & (0x03<<2);//读出bit2-bit3
faultTime++;
//0x02<<2:表示此时转换成PLL
}while((faultTime<0x0fffffff) && (Rcc_CR_PLL_Ready!=(0x02<<2)))
13.此时PLL转换成功
data:image/s3,"s3://crabby-images/70084/70084c07938bfdaa5eda7f59bdcbea8453325f98" alt=""
14.完整代码
cpp
#include "clock.h"
void Set_SysClockTo72M(void){
//检测外部晶振是否准备好
unsigned int Rcc_CR_HSE_Ready=0;
//等待开启PLL开启成功
unsigned int Rcc_CR_PLL_Ready=0;
//判断切换成PLL是否成功
unsigned int RCC_CF_SWS_PLL=0;
unsigned int faultTime=0;//判断等待是否超时
//一、复位RCC_CR寄存器
rRCC_CR = 0x00000083;
//二、开启外部时钟(外部晶振)
//第一步:先置0【将bit16清零】
rRCC_CR &= ~(1<<16);//关闭HSEON
//第二步:在置1
rRCC_CR |= (1<<16);//打开HSEON,让HSE开始工作
//三、检测外部时钟开启是否成功
do{
//检测HSEREAY(bit17)是否为1,1表示准备好
Rcc_CR_HSE_Ready=rRCC_CR&(1<<17);//取出bit17
faultTime++;
}while((faultTime<0x0fffffff) && (Rcc_CR_HSE_Ready==0));
//跳出do-while 1)要么超时2)要么准好了
//判断是超时还是准备好
//注意点:不能直接使用"Rcc_CR_HSE_Ready"因为rRCC_CR是需要读一次寄存器
//但是读出的结果可能还未改变,所以一定不能直接使用
if((rRCC_CR&(1<<17))!=0)//rRCC_CR&(1<<17)==1
{//这里HSE就ready,下面再去配置PLL并且等待他ready
//四、对其进行预分频
//HPRE【AHB】:对应bit4-bit7:不分频(000)
//PPRE1【APB1】:对应bit8-bit10:进行二分频(100)
//PPRE2【APB2】:对应bit11-bit13:不分频(000)
//AHB和APB2未分频,APB1被2分频
//所以最终:AHB和APB2都是72MHZ,APB1是36MHZ
//第一步:先置0
rRCC_CFGR=(~((0x0f<<4) | (0x07<<8) | (0x07<<11)));
//等价于:rRCC_CFGR=(~(0x3ff<<4));
//第二步:置1
rRCC_CFGR=(((0x0<<4) | (0x04<<8) | (0x0<<11)));
//五、设置SHE为输入时钟,同时HSE不分频
//选择HSE作为PLL输入并且HSE不分频
//设置为输入时钟:bit16
//设置为不分频:bit17
//第一步:先置0
rRCC_CFGR &=(~((1<<16) | (1<<17)));
//第二步:置1,bit16
rRCC_CFGR |= ((1<<18) | (0<<17));
//六、设置PLL倍频系数
//9分频:0111:0x07
rRCC_CFGR &=(~(0x0f<<18));//清零bit18-bit21
rRCC_CFGR |= (0x07<<18);//设置为9倍频
//七、打开PLL开关
rRCC_CR |= (1<<24);
//八、等待开启PLL开启成功
do{
Rcc_CR_PLL_Ready=rRCC_CR & (1<<25);//检测第25位是否为1
faultTime++;
}while((faultTime<0x0fffffff) && (Rcc_CR_PLL_Ready==0));
if((rRCC_CR & (1<<25)) == (1<<25)){
//到这里说明PLL已经稳定,可以用了,下面可以切换成外部时钟了
//九、切换成PLL
rRCC_CFGR &=(~(0x03)<<0);
rRCC_CFGR |=(0x10<<0);
//十、判断切换成PLL是否成功
do{
RCC_CF_SWS_PLL=rRCC_CFGR & (0x03<<2);//读出bit2-bit3
faultTime++;
//0x02<<2:表示此时转换成PLL
}while((faultTime<0x0fffffff) && (Rcc_CR_PLL_Ready!=(0x02<<2)));
//十一、此时PLL转换成功
if((rRCC_CFGR & (0x03<<2))==(0x02<<2)){
//到这里我们的时钟整个就设置好了,可以结束了
}else{
//到这里说明PLL输出作为PLL失败
while(1);
}
}
else{
//到这里说明PLL启动时出错了,PLL不能稳定工作
while(1);
}
}else{//超时,或者未准备好,此时HSE不可以使用
while(1);
}
}
三、问题解决
1.我们想要让led快速闪3下,然后换成72MHZ的频率接着闪
cpp
void delay(){
unsigned int i=0,j=0;
for(i=0;i<1000;i++){
for(j=0;j<2000;j++){
}
}
}
void led_init(){
rRCC_APB2ENR = 0x00000008;
rGPIOB_CRH = 0x33333333;
rGPIOB_ODR = 0x0000ff00;//全灭
}
void led_flash(void){
unsigned int i=0;
for(i=0;i<3;i++){
rGPIOB_ODR = 0x00000000;//全亮
delay();
rGPIOB_ODR = 0x0000ff00;//全灭
delay();
}
}
void main(void){
led_init();
led_flash();
Set_SysClockTo72M();
led_flash();
}
但是实际上并无法实现,只能在闪烁完3次后就熄灭。
2.问题解决
led初始化时,默认是全亮的
1.degger方法
把点亮led灯的函数加到clock中去,看看代码运行到哪里不会亮
2.判断超时变量的初始化
因为我们多次使用到超时变量,则每一个进入do-while循环之前要重新置0
data:image/s3,"s3://crabby-images/4eca2/4eca22d43f8d80cd8cb3803c7d9f80ca876724a8" alt=""
data:image/s3,"s3://crabby-images/83418/83418731109bee1efca1c8e1522571c53db01669" alt=""
3.出错点
data:image/s3,"s3://crabby-images/7c946/7c9467b9ec1d7c408e59560817fca3479ca0f617" alt=""