×
嵌入式 > 嵌入式开发 > 详情

STM32开发板例程讲解之二:GPIO的描述和配置

发布时间:2020-08-26 发布时间:
|
上一讲创建了一个stm32工程,从本讲开始将深入stm32内核与外设讲解。

首先介绍stm32的GPIO,这是入门的起点,也是最容易上手的部分。

一、GPIO的综合描述

stm32每一个GPIO端口拥有2个32bits的configuration寄存器(GPIOx_CRL,GPIOx_CRH),2个32bits的数据寄存器(GPIOx_IDR,GPIOx_ODR),1个32bits的set/reset寄存器(GPIOx_BSRR),1个16bits的reset寄存器(GPIOx_BRR)和1个32bits的Lock寄存器(GPIOx_LCKR)。

(一)每一个IO引脚都可以使用软件配置为以下几种模式:

1.浮空输入
2.带上拉输入
3.带下拉输入
4.模拟输入
5.开漏输出——(此模式可实现hotpower说的真双向IO)
6.推挽输出
7.复用功能的推挽输出
8.复用功能的开漏输出
模式7和模式8需根据具体的复用功能决定。

每一个IO引脚都可以单独编程,但是每一个IO寄存器只能32bits访问(半字或者字节访问都被禁止)。

(二)专门的寄存器(GPIOx_BSRR和GPIOx_BRR)实现对GPIO口的原子操作,即回避了设置或清除I/O端口时的“读-修改-写”操作,使得设置或清除I/O端口的操作不会被中断处理打断而造成误动作。

(三)每个GPIO口都可以作为外部中断的输入,便于系统灵活设计。

(四)I/O口的输出模式下,有3种输出速度可选(2MHz、10MHz和50MHz),这有利于噪声控制。

(五)所有I/O口兼容CMOS和TTL,多数I/O口兼容5V电平。

(六)大电流驱动能力:GPIO口在高低电平分别为0.4V和VDD-0.4V时,可以提供或吸收8mA电流;如果把输入输出电平分别放宽到1.3V和VDD-1.3V时,可以提供或吸收20mA电流。

(七)具有独立的唤醒I/O口。

(八)很多I/O口的复用功能可以重新映射。

(九)GPIO口的配置具有上锁功能,当配置好GPIO口后,可以通过程序锁住配置组合,直到下次芯片复位才能解锁。此功能非常有利于在程序跑飞的情况下保护系统中其他的设备,不会因为某些I/O口的配置被改变而损坏——如一个输入口变成输出口并输出电流。

二、GPIO的结构


三、GPIO的配置

(一)GPIO模式选择和速度匹配

(1) 浮空输入_IN_FLOATING ——浮空输入,可以做KEY识别,RX1。

(2)带上拉输入_IPU——IO内部上拉电阻输入。

(3)带下拉输入_IPD—— IO内部下拉电阻输入。

(4) 模拟输入_AIN ——应用ADC模拟输入,或者低功耗下省电。

(5)开漏输出_OUT_OD ——IO输出0接GND,IO输出1,悬空,需要外接上拉电阻,才能实现输出高电平。当输出为1时,IO口的状态由上拉电阻拉高电平,但由于是开漏输出模式,这样IO口也就可以由外部电路改变为低电平或不变。可以读IO输入电平变化,实现C51的IO双向功能。

(6)推挽输出_OUT_PP ——IO输出0-接GND, IO输出1 -接VCC,读输入值是未知的。

(7)复用功能的推挽输出_AF_PP ——片内外设功能(I 2C的SCL,SDA)

(8)复用功能的开漏输出_AF_OD——片内外设功能(TX1,MOSI,MISO.SCK)

GPIO输出的速度匹配:

GPIO_Speed_10MHz 最高输出速率10MHz

GPIO_Speed_2MHz 最高输出速率2MHz

GPIO_Speed_50MHz 最高输出速率50MHz

I/O口的输出模式下,有3种输出速度可选(2MHz、10MHz和50MHz),这个速度是指I/O口驱动电路的响应速度而不是输出信号的速度,输出信号的速度与程序有关(芯片内部在I/O口的输出部分安排了多个不同响应速度的输出驱动电路,用户可以根据自己的需要选择合适的驱动电路)。通过选择速度来选择不同的输出驱动模块,达到最佳的噪声控制和降低功耗的目的。高频的驱动电路,噪声也高,当不需要高的输出频率时,请选用低频驱动电路,这样非常有利于提高系统的EMI性能。当然如果要输出较高频率的信号,但却选用了较低频率的驱动模块,很可能会得到失真的输出信号。

有一点是关键,即GPIO的引脚速度跟应用匹配(推荐10倍以上)。比如:
1)对于串口,假如最大波特率只需115.2k,那么用2M的GPIO的引脚速度就够了,既省电也噪声小。
2)对于I2C接口,假如使用400k波特率,若想把余量留大些,那么用2M的GPIO的引脚速度或许不够,这时可以选用10M的GPIO引脚速度。
3)对于SPI接口,假如使用18M或9M波特率,用10M的GPIO的引脚速度显然不够了,需要选用50M的GPIO的引脚速度。

(二)在STM32中如何配置片内外设使用的IO端口

①配置输入的时钟;②初始化后即被激活(开启);③如果使用该外设的输入输出管脚,则需要配置相应的GPIO端口(否则该外设对应的输入输出管脚可以做普通GPIO管脚使用);④再对外设进行详细配置。

对应到外设的输入输出功能有下述三种情况:
①外设对应的管脚为输出:需要根据外围电路的配置选择对应的管脚为复用功能的推挽输出或复用功能的开漏输出。
②外设对应的管脚为输入:则根据外围电路的配置可以选择浮空输入、带上拉输入或带下拉输入。
③ADC对应的管脚:配置管脚为模拟输入。

如果把端口配置成复用输出功能,则引脚和输出寄存器断开,并和片上外设的输出信号连接。将管脚配置成复用输出功能后,如果外设没有被激活,那么它的输出将不确定。

(三)通用IO端口(GPIO)初始化:

1、RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|B|C,ENABLE):使能APB2总线外设时钟;
2、RCC_APB2PeriphResetCmd(RCC_APB2Periph_GPIOA|B|C,DISABLE):释放GPIO复位;

3、配置各个PIN端口(模拟输入_AIN、输入浮空_IN_FLOATING、输入上拉_IPU、输入下拉_IPD、开漏输出_OUT_OD、推挽式输出_OUT_PP、推挽式复用输出_AF_PP、开漏复用输出_AF_OD)和匹配速度。
4、GPIO初始化完成

四、GPIO实例

#include "stm32f10x.h"

#include "stm32_eval.h"

GPIO_InitTypeDef GPIO_InitStructure;

void RCC_Configuration(void);

void Delay(__IO uint32_t nCount);

int main(void)

{

RCC_Configuration();

#if 0

// 配置所有未使用GPIO引脚为输入模式(浮空输入),这样可以降低功耗,并且提高器件的抗EMI/EMC 的性能

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB |

RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD |

RCC_APB2Periph_GPIOE, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_All;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_Init(GPIOB, &GPIO_InitStructure);

GPIO_Init(GPIOC, &GPIO_InitStructure);

//armfly :注释掉的原因是当代码在外部存储器运行时,GPIOD,E,F,G部分IO用于FSMC,因此对这些IO不能重置,否则导致取指异常

// GPIO_Init(GPIOD, &GPIO_InitStructure);

// GPIO_Init(GPIOE, &GPIO_InitStructure);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB |

RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD |

RCC_APB2Periph_GPIOE, DISABLE);

#ifdef USE_STM3210E_EVAL

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF | RCC_APB2Periph_GPIOG, ENABLE);

// armfly :由于代码在外部SRAM运行时,GPIOD,E,F,G部分IO用于FSMC,

因此对这些IO不能重置,否则导致取指异常

//GPIO_Init(GPIOF, &GPIO_InitStructure);

//GPIO_Init(GPIOG, &GPIO_InitStructure);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF | RCC_APB2Periph_GPIOG, DISABLE);

#endif

#endif

// 初始化STM3210X-EVAL板上的4个LED,即初始化LED对应的IO引脚

STM_EVAL_LEDInit(LED1);

STM_EVAL_LEDInit(LED2);

STM_EVAL_LEDInit(LED3);

STM_EVAL_LEDInit(LED4);

while (1)

{

STM_EVAL_LEDOn(LED1);

// 原始值 = 0xAFFFF,但是当代码在外部RAM运行时,效率很低,会延迟10秒以上,而到代码在内部RAM或内部Flash执行时,速度很快,小于100ms

Delay(0xAFFFF);

// Turn on LD2 and LD3

STM_EVAL_LEDOn(LED2);

STM_EVAL_LEDOn(LED3);

// Turn off LD1

STM_EVAL_LEDOff(LED1);

// Insert delay

Delay(0xAFFFF);

// Turn on LD4

STM_EVAL_LEDOn(LED4);

//Turn off LD2 and LD3

STM_EVAL_LEDOff(LED2);

STM_EVAL_LEDOff(LED3);

//Insert delay

Delay(0xAFFFF);

//Turn off LD4

STM_EVAL_LEDOff(LED4);

}

}

void RCC_Configuration(void)

{

// 建立MCU系统,初始化嵌入式FLASH接口,初始化锁相环(附注1)和系统频率参数

SystemInit();

}

void Delay(__IO uint32_t nCount)

{

for(; nCount != 0; nCount--);

}

#ifdefUSE_FULL_ASSERT

// @briefReports the name of the source file and the source line number

//where the assert_param error has occurred.

//@paramfile: pointer to the source file name

// @paramline: assert_param error line source number

// @retval None

void assert_failed(uint8_t* file, uint32_t line)

{

// User can add his own implementation to report the file name and line number,

ex: printf("Wrong parameters value: file %s on line %d", file, line)

//Infinite loop

while (1)

{

}

}

#endif



『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
关于Erlang语言的功能及特性