你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

基于STM32F1的CAN通信之GPIO

[复制链接]
攻城狮Melo 发布时间:2023-10-23 15:41
一、什么是GPIO
GPIO(英语:General-purpose input/output),通用型之输入输出的简称,可以用来输入高低电平或者输出高低电平。这里的高电平指的是3.3V,低电平指的是0V。通常称GPIO为IO口,或者引脚。STM32F103ZET6有GPIOx_0~GPIOx_15,其中x = A,B,C,D,E,F,G。

二、GPIO的输入/输出模式

GPIO有多种输入输出模式,输入模式有
• 输入浮空 输入浮空指的是GPIO与外设之间既不接高电平,也不接低电平,呈高阻态。除了类似于在数据传输时将GPIO配置为输入浮空外,一般不配置为该模式。因为输入浮空状态的GPIO电压具有不确定性,可能是0V,也可能是VCC,或者是介于0V和VCC之间的某一个值。
• 输入上拉 输入上拉是通过一个上拉电阻,将GPIO拉至高电平状态,不受外接电路的影响。
• 输入下拉 输入下拉是通过一个下拉电阻,将GPIO拉至低电平,不受外接电路的影响。
• 模拟输入 模拟输入模式通常用在ADC采集,采集模拟信号。

输出模式有
• 开漏输出
• 推挽式输出
• 推挽式复用功能
• 开漏复用功能

对于一些输出模式这里就不再做详细介绍了,贴一篇大佬的文章深刻理解GPIO。这些模式在接下来的学习过程中会慢慢的介绍这些模式需要在什么时候使用,这里只需要知道就够了。


三、GPIO初始化配置
本专栏介绍的是使用库函数进行开发,很多内容都是库函数提供的,相对来讲非常方便。在初始化GPIO时有一个结构体,只需要对这个结构体进行配置即可。结构体中包括想要配置的GPIO引脚,GPIO速度,GPIO工作模式。

初始化GPIO的步骤主要有
• 定义GPIO结构体
• 开启时钟 GPIO工作需要提供时钟信号,在初始化结构体之前需要将时钟打开
• 配置结构体成员 GPIO_Pin是想要配置的IO,GPIO_Speed,通常写GPIO_Speed_50MHz,GPIO_Mode是IO的工作模式
• 写入配置

GPIO的工作模式在程序中有定义
  1. typedef enum
  2. { GPIO_Mode_AIN = 0x0,   // 模拟输入
  3.   GPIO_Mode_IN_FLOATING = 0x04,   // 输入浮空
  4.   GPIO_Mode_IPD = 0x28,   // 输入下拉
  5.   GPIO_Mode_IPU = 0x48,   // 输入上拉
  6.   GPIO_Mode_Out_OD = 0x14,   // 开漏输出
  7.   GPIO_Mode_Out_PP = 0x10,   // 推挽式输出
  8.   GPIO_Mode_AF_OD = 0x1C,   // 开漏复用功能
  9.   GPIO_Mode_AF_PP = 0x18   // 推挽式复用功能
  10. }GPIOMode_TypeDef;
复制代码

初始化GPIO的例程如下
  1. void Drv_Gpio_Init (void)
  2. {
  3.     GPIO_InitTypeDef GPIO_InitStructure;   // 定义结构体
  4.     // 开启时钟
  5.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB,ENABLE);


  6.     // 配置结构体
  7.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_8;
  8.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  9.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;
  10.     GPIO_Init(GPIOA, &GPIO_InitStructure);
  11.    
  12.     // 配置结构体
  13.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_2;
  14.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  15.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;
  16.     GPIO_Init(GPIOB, &GPIO_InitStructure);
  17. }
复制代码

这里初始化的是PA0,PA8和PB1,PB2。在开启时钟不必要写两遍相同的代码用来初始化GPIOA和GPIOB的时钟。写一句,用“ | ”同时开启两个GPIO的时钟。写Pin时也同理。

四、Boot引脚
这里简单介绍一下Boot引脚的配置,对于只是利用核心板编写程序的小伙伴来说Boot引脚的存在感较低,但是当我们需要绘制硬件电路图时,Boot引脚怎么连接就显得很重要。中文参考手册中介绍如下

微信图片_20231023154028.png

中文参考手册Boot引脚介绍



这里简单说一下本人经历得来的经验。如果想使用USB转TTL通过串口下载程序,Boot0需要通过KΩ级电阻接VCC,Boot1接地。下载完程序后再将Boot0接地。需要注意的是这个KΩ级别的电阻最好用10KΩ左右电阻,如果电阻太大会导致下载失败。

由于本人在使用自制板时只使用过串口下载,所以对于用调试器下载需要怎么配置Boot引脚,未测试。


五、一些特殊的GPIO
在使用GPIO时需要注意一些特殊的GPIO,否则你会疑惑,为什么一些引脚的高低电平无法控制。其实有些GPIO在上电后就有自己的默认设置,会稳定在高电平或者低电平。比如用作JTAG的几个GPIO——PB3,PB4,PA13,PA14和PA15。这几个GPIO在上电后就已经默认用作JTAG,即使用上面的GPIO初始化程序将这几个引脚初始化后,依旧无法控制(无法正常用程序拉高拉低)。

这个时候我们需要对这几个GPIO进行复用重映射,关掉其默认的功能,才能作为普通的GPIO使用。针对JTAG/SWD复用功能重映射,中文参考手册描述如下

微信图片_20231023154011.png

中文参考手册关于JTAG/SWD引脚的描述



图中说明这几个GPIO有几种模式,完全SWJ,表中的五个IO均不可作为普通IO使用。完全SWJ但是没有JNTRST,PB3和PB4可用,依此解读。上图是针对寄存器开发描述的内容,使用库函数开发时有封装好的函数,但是需要注意的是需要提前开启AFIO时钟(对于AFIO是指什么,大家可以自行了解)。使用库函数开发时针对特殊引脚进行重映射的操作步骤为

• 定义GPIO结构体
• 开启GPIO时钟和AFIO时钟
• 重映射引脚(根据所需情况设定引脚模式)
• 配置GPIO结构体初始化GPIO

库函数中提供了可以选择的三种引脚模式
  1. GPIO_Remap_SWJ_NoJTRST          // 完全SWJ(恢复引脚的默认功能)
  2. GPIO_Remap_SWJ_JTAGDisable      // 关闭JTAG,启用SW-DP
  3. GPIO_Remap_SWJ_Disable          // 关闭JTAG-DP,关闭SW-DP
复制代码


提供了一个函数,可以进行重映射操作

  1. void GPIO_PinRemapConfig(uint32_t GPIO_Remap, FunctionalState NewState);
复制代码

当需要使用到上述的五个GPIO时,初始化程序需要修改,举例如下
  1. void Drv_Gpio_Init (void)
  2. {
  3.     GPIO_InitTypeDef GPIO_InitStructure;   // 定义结构体
  4.     // 开启时钟
  5.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB |
  6.                            RCC_APB2Periph_AFIO,ENABLE);


  7.     // 关闭JTAG-DP,关闭SW-DP
  8.     GPIO_PinRemapConfig(GPIO_Remap_SWJ_Disable,ENABLE);
  9.    
  10.     // 配置结构体
  11.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
  12.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  13.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;
  14.     GPIO_Init(GPIOA, &GPIO_InitStructure);
  15.    
  16.     // 配置结构体
  17.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_3;
  18.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  19.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;
  20.     GPIO_Init(GPIOB, &GPIO_InitStructure);
  21. }
复制代码

STM32F103ZET6芯片GPIO资源较为丰富,平时开发时尽量不要使用这几个特殊引脚。


六、点亮LED

学习完GPIO后我们就可以利用GPIO进行一些简单的操作,比如点亮LED,使用蜂鸣器,用IO驱动小黄电机,检测按键等。

点亮LED比较简单,但是也有一些需要注意的点

• 根据硬件电路确定LED是高电平点亮还是低电平点亮
• 初始化GPIO后要先将所有的LED


6.1 硬件电路
下图所示电路中,LED右侧接3.3V,左侧接IO口。此时如果想点亮LED,需要将对应的IO电平拉低。相反如果右侧接的是地,如果想要点亮LED,需要将对应IO电平拉高。

微信图片_20231023154005.png

LED电路



6.2 拉高/拉低GPIO
如何将GPIO电平拉高拉低?库函数提供了封装好的函数
  1. // 设置为高电平
  2. void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);


  3. // 设置为低电平
  4. void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);
复制代码

6.3 程序设计
点亮LED很简单,只需要初始化相应的GPIO,输入模式设置为推挽式输出,然后设定电平即可。

这里给出点亮LED的例程,LED电路为一侧接3.3V,另一侧接GPIO。

这里给出的只是一些必要的函数,如果需要工程模板可私信联系。

需要注意的是,初始化GPIO时需要开启时钟,开启时钟之前需要确认GPIO挂载的总线。
  1. /*
  2. *==============================================================================
  3. *函数名称:Drv_Led_Init
  4. *函数功能:初始化LED的GPIO
  5. *输入参数:无
  6. *返回值:无
  7. *==============================================================================
  8. */
  9. void Drv_Led_Init (void)
  10. {
  11.     GPIO_InitTypeDef GPIO_InitStructure;   // 定义结构体
  12.     // 开启时钟
  13.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);


  14.     // 配置结构体
  15.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
  16.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  17.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  18.     GPIO_Init(GPIOA, &GPIO_InitStructure);


  19.     GPIO_SetBits(GPIOA,GPIO_Pin_1);   // 熄灭LED
  20. }


  21. /*
  22. *==============================================================================
  23. *函数名称:Med_Led_On
  24. *函数功能:点亮LED
  25. *输入参数:无
  26. *返回值:无
  27. 备注:如果有多个LED可以定义一个结构体或者使用swtich,增加一个输入变量
  28.       来确定开启哪个LED
  29. *==============================================================================
  30. */
  31. void Med_Led_On (void)
  32. {
  33.     GPIO_ResetBits(GPIOA,GPIO_Pin_1);   // 熄灭LED
  34. }
复制代码

七、GPIO的位带操作
对于什么是“位带”这里就不做解释了,大家可以自行搜索。这里只介绍如何使用。位带操作可以使操作GPIO变得更加简单。在模板程序的sys.h文件中已经定义好了各个IO的位带操作所需内容
  1. //IO口操作,只对单一的IO口!
  2. //确保n的值小于16!
  3. #define PAout(n)   BIT_ADDR(GPIOA_ODR_Addr,n)  //输出
  4. #define PAin(n)    BIT_ADDR(GPIOA_IDR_Addr,n)  //输入

  5. #define PBout(n)   BIT_ADDR(GPIOB_ODR_Addr,n)  //输出
  6. #define PBin(n)    BIT_ADDR(GPIOB_IDR_Addr,n)  //输入

  7. #define PCout(n)   BIT_ADDR(GPIOC_ODR_Addr,n)  //输出
  8. #define PCin(n)    BIT_ADDR(GPIOC_IDR_Addr,n)  //输入

  9. #define PDout(n)   BIT_ADDR(GPIOD_ODR_Addr,n)  //输出
  10. #define PDin(n)    BIT_ADDR(GPIOD_IDR_Addr,n)  //输入

  11. #define PEout(n)   BIT_ADDR(GPIOE_ODR_Addr,n)  //输出
  12. #define PEin(n)    BIT_ADDR(GPIOE_IDR_Addr,n)  //输入

  13. #define PFout(n)   BIT_ADDR(GPIOF_ODR_Addr,n)  //输出
  14. #define PFin(n)    BIT_ADDR(GPIOF_IDR_Addr,n)  //输入

  15. #define PGout(n)   BIT_ADDR(GPIOG_ODR_Addr,n)  //输出
  16. #define PGin(n)    BIT_ADDR(GPIOG_IDR_Addr,n)  //输入
复制代码

有了位带操作后,就不需要再使用前面介绍的拉高拉低函数来操作GPIO。比如需要拉d低PA1
  1. PAout(1) = 0;   // 拉低PA1
复制代码

有了位带操作后也可以对GPIO进行宏定义,用自己想要的名字来操作它。还是拿上面的拉低PA1举例
  1. #define LED   PAout(1)   // 将PA1宏定义为LED

  2. LED = 1;   // 拉高PA1
  3. LED = 0;   // 拉低PA1
复制代码


转载自: 二土电子
如有侵权请联系删除


收藏 评论0 发布时间:2023-10-23 15:41

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版