stm32的编程和stc89c51还是存在着很多思想上的不同的,在51单片机中,我们点亮LED灯,只用给对应IO高低电平就可以了,而stm32中,就一个简单的GPIO,我们也需要配置好几个寄存器,而且,就算配置好GPIO寄存器,我们还必须有一步不能缺少的操作,那就是开启对应GPIO的时钟。在stm32的编程中,开启对应时钟是必不可少的一个操作,而且,开启时钟的操作必须在配置相应寄存器之前,以GPIO为例,如果我们先配置GPIO的寄存器,不开启时钟或者先配置GPIO寄存器,最后再开启时钟,都不能正常驱动芯片工作。时钟就像人的脉搏一样,人必须有脉搏才能工作,同样,stm32也必须先给时钟,再配置寄存器达到相应的功能。
那么我是怎么知道要这样配置的呢?
第一,现在的教程比较成熟,能够知道这些内容;第二,在开发板的学习教程中,他也是通过ST的官方demo知道的,具体在
F4固件库\STM32F4xx_DSP_StdPeriph_Lib_V1.5.1\STM32F4xx_DSP_StdPeriph_Lib_V1.5.1\Project\STM32F4xx_StdPeriph_Examples\
这个目录下,有基于ST的参考代码,别人告诉我们要先开启时钟;第三,为什么要先开启对应外设时钟才能工作,51单片机就不需要啊。这个也是可以理解的,为了低功耗嘛,时钟相当于一把钥匙控制这个电路的开关,没用时钟这把钥匙,你什么操作也是白搭,对应时钟控制对应外设,这也无可厚非。
那么我们怎么查看具体外设到底要开启什么时钟呢?
查看《STM32F4xx中文参考手册1》2.3节,STM32F4xx 寄存器边界地址 ,可以看到对应外设的时钟。
可以看到我们的GPIOH是挂载在AHB1总线上,所以要先开启AHB1总线上的GPIOH时钟。
那么又在哪里查看这个呢?还是中文参考资料中,6.3.12 RCC AHB1 外设时钟使能寄存器 (RCC_AHB1ENR)
通过这里我们我们可以看到:
参考手册上说明,这个寄存器的第七位控制我们的GPIOH,置1,使能。
同样,配置GPIO寄存器时,也应该参考对应手册:
在知道这些流程之后,开始编写程序:
/*
使用寄存器的方法点亮LED灯
*/
#include "./stm32f4xx.h" /**
* 主函数
*/
int main(void)
{
/*开启 GPIOH 时钟,使用外设时都要先开启它的时钟*/
RCC_AHB1ENR |= (<<); //很明朗了,第七位要置1,使能时钟 /* LED 端口初始化 */
//根据上面的寄存器描述依次配置,这个技能是只要你去看,就会的东西
/*GPIOH MODER10清空*/
GPIOH_MODER &= ~( 0x03<< (*));
/*PH10 MODER10 = 01b 输出模式*/
GPIOH_MODER |= (<<*); /*GPIOH OTYPER10清空*/
GPIOH_OTYPER &= ~(<<*);
/*PH10 OTYPER10 = 0b 推挽模式*/
GPIOH_OTYPER |= (<<*); /*GPIOH OSPEEDR10清空*/
GPIOH_OSPEEDR &= ~(0x03<<*);
/*PH10 OSPEEDR10 = 0b 速率2MHz*/
GPIOH_OSPEEDR |= (<<*); /*GPIOH PUPDR10清空*/
GPIOH_PUPDR &= ~(0x03<<*);
/*PH10 PUPDR10 = 01b 上拉模式*/
GPIOH_PUPDR |= (<<*); /*PH10 BSRR寄存器的 BR10置1,使引脚输出低电平*/
GPIOH_BSRR |= (<<<<); /*PH10 BSRR寄存器的 BS10置1,使引脚输出高电平*/
//GPIOH_BSRR |= (1<<10); while(); } // 函数为空,目的是为了骗过编译器不报错
void SystemInit(void)//这个函数必须自己写一个加上,因为启动文件的关系,到后面会详细说这个问题
{
} /*********************************************END OF FILE**********************/
这个是我们自己用寄存器实现的例子,我的开发板有三个灯,分别接GPIOH10,11,12,所以更改了几次代码,这样也发现寄存器编程的一个问题,修改代码有点麻烦,而且还是有点冗余,虽然我们可以通过宏或者函数封装,但是,那样封装之后,他就是库函数版本了, 到后面,我们都会采用库函数的方式来学习,项目开发中肯定也是库函数首选。但是,我们经过自己配置寄存器这个步骤,知道了库函数底层其实也就是封装了配置好的寄存器而已,知道原理之后,我们还是应该才赢开发效率更高的库函数来进行应用。