GPIO:通用输入输出口
有八种输入输出模式
一个端口可以有多个输入但只能有一个输出,故配置输入不能输出,配置输出可以输入
模式名称 | 性质 | 特征 |
---|---|---|
浮空输入 | 数字输入 | 可读取引脚电平,若引脚悬空,则电平不确定,最常用 |
上拉输入 | 数字输入 | 可读取引脚电平,内部连接上拉电阻,悬空时默认高电平,输入信号较弱,容易受到干扰且不稳定时选择 |
下拉输入 | 数字输入 | 可读取引脚电平,内部连接下拉电阻,悬空时默认低电平,不常用 |
模拟输入 | 模拟输入 | GPIO无效,引脚直接接入内部ADC |
开漏输出 | 数字输出 | 可输出引脚电平,高电平为高阻态(无驱动能力),低电平接VSS |
推挽输出 | 数字输出 | 可输出引脚电平,高电平接VDD(有驱动能力),低电平接VSS |
复用开漏输出 | 数字输出 | 由片上外设控制,高电平为高阻态,低电平接VSS |
复用推挽输出 | 数字输出 | 由片上外设控制,高电平接VDD,低电平接VSS |
引脚电平有3V3(引脚定义无写FT),5V(引脚定义有写FT)
作用
- 输出模式:控制端口输出高低电平用以驱动元器件,模拟通信协议输出时序等
- 输入模式:读取端口高低电平或电压用以读取按键输入、外设电平信号输入以及ADC电压采集,模拟通信协议接收数据等
GPIO_PIN可以使用按位或同时设置多个引脚,时钟也是可以
GPIO的输出(操作步骤)
使用RCC开启GPIO的时钟
RCC外设中的库函数,常用的只有以下三个函数,在keil内可以右键进入定义查看参数类型
void RCC_AHBPeriphClockCmd(uint32_t RCC_AHBPeriph, FunctionalState NewState);//RCC_AHB外设时钟控制
void RCC_APB2PeriphClockCmd(uint32_t RCC_APB2Periph, FunctionalState NewState);//RCC_APB1外设时钟控制
void RCC_APB1PeriphClockCmd(uint32_t RCC_APB1Periph, FunctionalState NewState);//RCC_APB2外设时钟控制
使用GPIO_Init函数初始化GPIO
GPIO外设中的库函数常用介绍
void GPIO_DeInit(GPIO_TypeDef* GPIOx);//复位
void GPIO_AFIODeInit(void);//复位AFIO外设
void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef* GPIO_InitStruct);//用结构体参数初始化GPIO口
void GPIO_StructInit(GPIO_InitTypeDef* GPIO_InitStruct);//将结构体变量赋一个默认值
uint8_t GPIO_ReadInputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);//读取输入数据寄存器某一端口的输入值,返回值为uint8_t,代表端口高低电平
uint16_t GPIO_ReadInputData(GPIO_TypeDef* GPIOx);//读取整个输入数据寄存器的输入值,返回值为uint16_t,每一位戴代表一个端口值
uint8_t GPIO_ReadOutputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);//读取输出数据寄存器某一端口的输出值
uint16_t GPIO_ReadOutputData(GPIO_TypeDef* GPIOx);//读取整个输出数据寄存器的输出值
//以上四个读取函数,输出模式用输出,输入模式用输入
void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);//设置指定端口为高电平
void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);//设置指定端口为低电平
void GPIO_WriteBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin, BitAction BitVal);//根据第三个参数设定指定端口
void GPIO_Write(GPIO_TypeDef* GPIOx, uint16_t PortVal);//可以同时对16个端口进行写入
//以上四个写入函数
使用输出或输入函数控制GPIO
即第二步的后八个函数以及其他操作等
GPIO_Mode_AIN = 0x0,//模拟输入
GPIO_Mode_IN_FLOATING = 0x04,//浮空输入
GPIO_Mode_IPD = 0x28,//下拉输入
GPIO_Mode_IPU = 0x48,//上拉输入
GPIO_Mode_Out_OD = 0x14,//开漏输出
GPIO_Mode_Out_PP = 0x10,//推挽输出
GPIO_Mode_AF_OD = 0x1C,//复用开漏
GPIO_Mode_AF_PP = 0x18//复用推挽
GPIO的输入
模块化编程
解释
当有多个外设联合驱动时,都放在主函数内会过于混乱,不方便管理且移植,故像这种驱动代码一般将他封装起来,单独放在.c与.h中,即在工程内建一个hardware的文件夹存放硬件驱动。
步骤
-
先按三个箱子添加新的文件,再魔法棒C/C++内添加文件夹路径,然后在右栏右键Hardware添加.c和.h文件,表示某一个外设驱动的存放位置,如下图为LED的驱动程序的.c/.h文件。
-
在.c文件中右键,添加STM32头文件;
-
在.h中添加以下代码的格式
#ifndef __LED_H//__(2个)文件名_(1个)H
#define __LED_H//__(2个)文件名_(1个)H
//中间写.c文件内定义的函数名
#endif
其他
AO模拟电压输出
DO数字电压输出
ctrl+alt+空格 --> 跳出补全搜索框
库函数的使用方法
- 打开对应.h文件看一下有什么库函数,右键到定义查看具体内容
- 官方库函数用户手册或帮助文档
- 搜索引擎搜索
与C的联系的知识
- 与51不同,int在这里是32位,而不是16位
- 因为经常对不上,所以改变了他们的名字,使之更浅显易懂且对得上
关键字 | 位数 | 表示范围 | stdint关键字 | ST关键字 |
---|---|---|---|---|
char | 8 | -128 ~ 127 | int8_t | s8 |
unsigned char | 8 | 0 ~ 255 | uint8_t | u8 |
short | 16 | -32768 ~ 32767 | int16_t | s16 |
unsigned short | 16 | 0 ~ 65535 | uint16_t | u16 |
int | 32 | -2147483648 ~ 2147483647 | int32_t | s32 |
unsigned int | 32 | 0 ~ 4294967295 | uint32_t | u32 |
long | 32 | -2147483648 ~ 2147483647 | ||
unsigned long | 32 | 0 ~ 4294967295 | ||
long long | 64 | -(2^64)/2 ~ (2^64)/2-1 | int64_t | |
unsigned long long | 64 | 0 ~ (2^64)-1 | uint64_t | |
float | 32 | -3.4e38 ~ 3.4e38 | ||
double | 64 | -1.7e308 ~ 1.7e308 |
杂
- 电路中一个电容一端接地,极大可能为滤波作用,可以去掉
- 上拉下拉即指端口电压大小,变大为上拉,变小为下拉,在电路中用分压定理判断
- 接地一般叫下接xx,接3V3/5V则叫上接xx
- 单片机都有上拉模式,但是下拉模式不确定,所以设计电路时一般使用上拉模式
- 外设配置不知道配置什么模式,可以查看数据手册的GPIO中的外设GPIO配置