一、什么是寄存器
? STM32编程通常有两种编程方法,一种是寄存器编程;另一种是固件库编程,其中寄存器编程是基础,而固件库编程是在寄存器编程的基础上升级而来的一种易于学习和开发的方法,是学习STM32编程时需重点掌握的一种编程方法。而固件库编程对于项目开发固然简单和快速,但是从学习的角度出发,寄存器编程的方法也不能不掌握。其实我们在学习8位单片机的时候,像是MSC-51单片机,就是采用寄存器编程。
二、GPIO框图剖析
? GPIO是通用输入输出口端口的简称,简单就是来说就是STM32可控制引脚。将STM32芯片引脚与外部设备连接起来,即可实现与外部通信、控制以及数据采集的功能。
? 通过GPIO硬件结构框图,就可以从整体上深入了解GPIO外设及它的各种应用模式。从最右端看起,最右端就是代表STM32芯片引出的GPIO引脚,其余部件都位于芯片内部。
2.1 基本结构分析
? 下面我们按上图中的编号对GPIO端口的结构部件进行说明。
①保护二极管及上、下拉电阻 ? 引脚的两个保护二极管可以防止引脚外部过高或过低的电压输入,当引脚电压高于VDD时,上方的二极管导通,当引脚电压低于VSS时,下方的二极管导通,防止不正常电压引入芯片导致芯片烧毁。尽管有这样的保护,并不意味着STM32的引脚能直接外接大功率驱动器件,如直接驱动电机,要么电机不转,要么导致芯片烧坏,必须要加大功率及隔离电路驱动。
②P-MOS管和N-MOS管 ? GPIO引脚线路经过两个保护二极管后,向上流向“输入模式”结构,向下流向“输出模式”结构。先看输出模式部分,线路经过一个由P-MOS和N-MOS管组成的单元电路。这个结构使GPIO具有了“推挽输出”和“开漏输出”两种模式。
? 所谓推挽输出模式,是根据这两个MOS管的工作方式来命名的。在该结构中输入高电平时,经过反向后,上方的P-MOS导通,下方的N-MOS关闭,对外输出高电平;而在该结构中输入低电平时,经过反向后,N-MOS管导通,P-MOS关闭,对外输出低电平。当引脚高低电平切换时,两个MOS管轮流导通,P管负责灌电流,N管负责拉电流,使其负载能力和开关速度都比普通的方式有很大的提高。推挽输出的低电平为0V,高电平为3.3V,推挽输出模式时的等效电路见图。 ? 而在开漏输出模式时,上方的P-MOS管完全不工作。如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出接地,若控制输出为1(它无法直接输出高电平)时,则P-MOS管和N-MOS管都关闭,所以引脚既不输出高电平,也不输出低电平,为高阻态。正常使用时必须外部接上拉电阻,等效电路见图7-3。它具有“线与”特性,也就是说,若有很多个开漏模式引脚连接到一起时,只有当所有引脚都输出高阻态,才由上拉电阻提供高电平,此高电平的电压为外部上拉电阻所接的电源的电压。若其中一个引脚为低电平,那线路就相当于短路接地,使得整条线路都为低电平,0V。 ? ? 推挽输出模式一般应用在输出电平为0和3.3V而且需要高速切换开关状态的场合。在STM32的应用中,除了必须用开漏模式的场合,一般习惯使用推挽输出模式。
? 开漏输出一般应用在I2C、SMBUS通信等需要“线与”功能的总线电路中。除此之外,还用在电平不匹配的场合,如需要输出5V的高电平,就可以在外部接一个上拉电阻,上拉电源为5V,并且把GPIO设置为开漏模式,当输出高阻态时,由上拉电阻和电源向外输出5V的电平,具体见图。 ③输出数据寄存器 ? 前面提到的双MOS管结构电路的输入信号,是由GPIO“输出数据寄存器GPIOx_ODR”提供的,因此我们通过修改输出数据寄存器的值,就可以修改GPIO引脚的输出电平。而“置位/复位寄存器GPIOx_BSRR”可以通过修改输出数据寄存器的值,从而影响电路的输出。 ④复用功能输出 ? “复用功能输出”中的“复用”是指STM32的其他片上外设对GPIO引脚进行控制,此时GPIO引脚用作该外设功能的一部分,算是第二用途。从其他外设引出来的“复用功能输出信号”与GPIO本身的数据寄存器都连接到双MOS管结构的输入中,将图7-1中的梯形结构作为开关切换选择。 ? 例如我们使用USART串口通信时,需要用到某个GPIO引脚作为通信发送引脚,这个时候就可以把该GPIO引脚配置成USART串口复用功能,由串口外设控制该引脚,发送数据。 ⑤输入数据寄存器 ? 看GPIO结构框图的上半部分,GPIO引脚经过内部的上、下拉电阻,可以配置成上/下拉输入,然后再连接到肖特基触发器,信号经过触发器后,模拟信号转化为0、1的数字信号,然后存储在“输入数据寄存器GPIOx_IDR”中,通过读取该寄存器就可以了解GPIO引脚的电平状态。 ⑥复用功能输入 ?与“复用功能输出”模式类似,在“复用功能输入模式”时,GPIO引脚的信号传输到STM32其他片上外设,由该外设读取引脚状态。 同样,如我们使用USART串口通信时,需要用到某个GPIO引脚作为通信接收引脚,这个时候就可以把该GPIO引脚配置成USART串口复用功能,使USART可以通过该通信引脚的接收远端数据。 ⑦模拟输入输出 ?当GPIO引脚用于ADC采集电压的输入通道时,用作“模拟输入”功能,此时信号是不经过肖特基触发器的,因为经过肖特基触发器后信号只有0、1两种状态,所以ADC外设要采集到原始的模拟信号,信号源输入必须在肖特基触发器之前。类似地,当GPIO引脚用于DAC作为模拟电压输出通道时,此时作为“模拟输出”功能,DAC的模拟信号输出就不经过双MOS管结构,而是直接输出到引脚。
2.2GPIO工作模式
?在固件库中,GPIO总共有8种细分的工作模式,稍加整理可以大致归类为以下3类。 ①输入模式(模拟/浮空/上拉/下拉) ?在输入模式时,肖特基触发器打开,输出被禁止,可通过输入数据寄存器GPIOx_IDR读取I/O状态。其中输入模式可设置为上拉、下拉、浮空和模拟4种。上拉和下拉输入很好理解,默认的电平由上拉或者下拉决定。浮空输入的电平是不确定的,完全由外部的输入决定,一般接按键的时候用的是这个模式。模拟输入则用于ADC采集。 ②输出模式(推挽/开漏) ?在输出模式中,推挽模式时双MOS管轮流工作,输出数据寄存器GPIOx_ODR可控制I/O输出高低电平。开漏模式时,只有N-MOS管工作,输出数据寄存器可控制I/O输出高阻态或低电平。输出速度可配置,有2MHz、10MHz、50MHz几种选项。此处的输出速度即I/O支持的高低电平状态最高切换频率,支持的频率越高,功耗越大,如果功耗要求不严格,把速度设置成最大即可。 ?在输出模式时肖特基触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。 ③复用功能(推挽/开漏) ?复用功能模式中,输出使能,输出速度可配置,可工作在开漏及推挽模式,但是输出信号源于其他外设,输出数据寄存器GPIOx_ODR无效;输入可用,通过输入数据寄存器可获取I/O实际状态,但一般直接用外设的寄存器来获取该数据信号。 ?通过对GPIO寄存器写入不同的参数,就可以改变GPIO的工作模式。再强调一下,要了解具体寄存器,一定要查阅《STM32F10X-中文参考手册》中对应外设的寄存器说明。在GPIO外设中,控制端口高低控制寄存器CRH和CRL可以配置每个GPIO的工作模式和工作的速度,每4位控制一个IO,CRL控制端口的低8位(见图7-5),CRH控制端口的高8位(见图7-6),具体要看CRL和CRH的寄存器描述。
三、点亮ELD
3.1 LED原理
?下图是STM32芯片和LED的连接图,这是个RGB灯,有红绿蓝3个小灯组成,使用PWM控制时可以混成256种不同的颜色。 ?图中从3个LED的阳极引出连接到3.3V电源,阴极各经过1个限流电阻引入至STM32的3个GPIO引脚,所以我们只要控制这3个引脚输出高低电平,即可控制其所连接的LED的亮灭。?如果您的实验板STM32连接到LED的引脚或极性不一样,只需要修改程序到对应的GPIO引脚即可,工作原理都是一样的。
我们的目标是把GPIO的引脚设置成推挽输出模式,并且默认下拉,输出低电平,这样就能让LED亮起来了。
3.2文件准备
?在创建工程后,主要有三个文件夹:startup_stm32f10x_hd.s、stm32f10x.h以及main.c。
3.2.1 startup_stm32f10x_hd.s
?名为startup_stm32f10x_hd.s的文件中用汇编语言写好了基本程序,当STM32芯片上电启动的时候,首先会执行这里的汇编程序,从而建立起C语言的运行环境,所以我们把这个文件称为启动文件。该文件使用的汇编指令是Cortex-M3内核支持的指令,可参考《Cortex-M3权威指南》中指令集内容。 ?startup_stm32f10x_hd.s文件由官方提供,一般需要修改也是在官方的基础上修改,不会自己完全重写。该文件可以从ST固件库里面找到,找到该文件后把启动文件添加到工程中即可。不同型号的芯片以及不同编译环境下使用的汇编文件是不一样的,但功能相同。 ?对于启动文件这部分我们主要总结它的功能,不详细讲解里面的代码。其功能如下:
- 初始化堆栈指针SP。
- 初始化程序计数器指针PC。
- 设置堆、栈的大小。
- 初始化中断向量表。
- 配置外部SRAM作为数据存储器(这个由用户配置,一般的开发板没有外部SRAM)。
- 调用System Ini() 函数配置STM32的系统时钟。
- 设置C库的分支入口“__main”(最终用来调用main函数)。
3.2.2 stm32f10x.h
?这个文件用来存放STM32寄存器映射的代码
#define PERIPH_BASE ((unsigned int)0x40000000)
#define APB1PERIPH_BASE PERIPH_BASE
#define APB2PERIPH_BASE (PERIPH_BASE+0x10000)
#define AHBPERIPH_BASE (PERIPH_BASE+0x20000)
#define RCC_BASE (AHBPERIPH_BASE + 0x1000)
#define GPIOB_BASE (APB2PERIPH_BASE + 0x0c00)
#define RCC_APB2ENR *(unsigned int*)(RCC_BASE + 0x18)
#define GPIOB_CRL *(unsigned int*)(GPIOB_BASE + 0x00)
#define GPIOB_CRH *(unsigned int*)(GPIOB_BASE + 0x04)
#define GPIOB_ODR *(unsigned int*)(GPIOB_BASE + 0x0C)
3.2.2 main.c
①GPIO模式 ?首先我们把连接到LED的GPIO引脚PB0配置成输出模式,即配置GPIO的端口配置成地7寄存器CRL,见下图。CRL中包含0-7号引脚,每个引脚占用四个寄存器位。MODE为用来配置输出的速度,CNF位用来配置各种输入输出模式。
?在这里我们把PB0配置为通用推免输出,输出速度为10M。
GPIO_CRL &= ~(0x00 << (4*0));
GPIO_CRL |=(1<<4*0);
?在代码中,我们首先把控制PB0的端口位清零,然后向他赋值“0001”,从而将GPIOB0的引脚设置成输出模式,速度为10M。
?代码中使用了“&=~”和“|=”这种操作方法,这是为了避免影响寄存器中的其他位,因为寄存器不能按位读写,不能直接赋值“0x00 0001”,这样就会影响他的高四位引脚。 ②控制引脚输出电平 在输出模式时,对端口位设置/清除寄存器BSRR寄存器、端口位清除寄存器BRR和ODR寄存器写入参数即可控制引脚的电平状态,其中操作BSRR和BRR最终影响的都是ODR寄存器,然后通过ODR寄存器的输出来控制GPIO,见下图。为了一步到位,我们在这里直接操作ODR寄存器来控制GPIO的电平,具体见代码。
GPIOB_ODR &= ~(1<<0);
GPIOB_ODR |= (0<<0);
③开启时钟外设 设置完GPIO的引脚,控制了电平输出,其实还差最后一步。由于STM32的外设很多,为了降低功耗,每个外设都对应一个时钟。在芯片刚上电的时候这些时钟都是关闭的,如果想要外设工作,必须把相应的时钟打开。 STM32的所有外设的时钟由一个专门的外设来管理,叫RCC(reset and clockcontrol),。 所有的GPIO都挂载到APB2总线上,具体的时钟由APB2外设时钟使能寄存器(RCC_ APB2ENR)来控制。
RCC_APB2ENR |= (1<<3);
④完成
#include "stm32f10x.h"
int main (void)
{
RCC_APB2ENR |= ((1)<<3);
GPIO_CRL &= ~(0x00 << (4*0));
GPIOB_CRL |= ((1)<<(4*0));
GPIOB_ODR &= ~(1<<0);
}
void SystemInit(void)
{
}
3.3 点亮三个灯
①startup_stm32f10x_hd.s文件配置
#define PERIPH_BASE ((unsigned int)0x40000000)
#define APB2PERIPH_BASE (PERIPH_BASE + 0x10000)
#define AHBPERIPH_BASE (PERIPH_BASE + 0x20000)
#define GPIOB_BASE (APB2PERIPH_BASE + 0x0C00)
#define GPIOB_CRL *(unsigned int*)(GPIOB_BASE+0x00)
#define GPIOB_CRH *(unsigned int*)(GPIOB_BASE+0x04)
#define GPIOB_IDR *(unsigned int*)(GPIOB_BASE+0x08)
#define GPIOB_ODR *(unsigned int*)(GPIOB_BASE+0x0C)
#define GPIOB_BSRR *(unsigned int*)(GPIOB_BASE+0x10)
#define GPIOB_BRR *(unsigned int*)(GPIOB_BASE+0x14)
#define GPIOB_LCKR *(unsigned int*)(GPIOB_BASE+0x18)
#define RCC_BASE (AHBPERIPH_BASE + 0x1000)
#define RCC_APB2ENR *(unsigned int*)(RCC_BASE+0x18)
②main.c文件
#include "stm32f10x.h"
int main(void)
{
RCC_APB2ENR |= (1<<3);
GPIOB_CRL &= ~( 0x0F<< (4*0));
GPIOB_CRL |= (1<<4*0);
GPIOB_ODR &= ~(1<<0);
GPIOB_CRL &= ~( 0x0F<< (4*1));
GPIOB_CRL |= (1<<4*1);
GPIOB_ODR &= ~(1<<1);
GPIOB_CRL &= ~( 0x0F<< (4*5));
GPIOB_CRL |= (1<<4*5);
GPIOB_ODR &= ~(1<<5);
while(1);
}
void SystemInit(void)
{
}
|