历史上的今天
返回首页

历史上的今天

今天是:2025年02月12日(星期三)

正在发生

2019年02月12日 | 【stm32f407】GPIO原理以及跑马灯的应用

2019-02-12 来源:eefocus

1.    GPIO介绍

1)介绍:GPIO:(GeneralPurpose Input Output (通用输入/输出)简称为GPIO) 


基本结构:



stm32f407VGT6这款单片机上共有PA‐PE共5×16共80个复用IO口,


每个通用I / O端口有4个32位配置寄存器(GPIOx_MODERGPIOx_OTYPER GPIOx_OSPEEDR GPIOx_PUPDR),两个32位数据寄存器(GPIOx_IDR和GPIOx_ODR),一个32位的置位/复位寄存器(GPIOx_BSRR),32位锁定寄存器(GPIOx_LCKR)和两个32位的备用功能选择寄存器(GPIOx_AFRH GPIOx_AFRL),如图:



2)GPIO的工作模式:


2.    GPIO的八种工作模式详解


浮空输入_IN_FLOATING

带上拉输入_IPU

带下拉输入_IPD

模拟输入_AIN

开漏输出_OUT_OD

推挽输出_OUT_PP

开漏复用输出_AF_OD

推挽复用输出_AF_PP

4输入 + 2输出 + 2复用输出,一共是8种模式,以下是八种模式的工作原理:


GPIO浮空输入_IN_FLOATING模式工作原理


以上截图就是浮空输入模式的原理图,


图中阴影的部分在浮空输入模式下是处于不工作状态的,尤其是下半部分的输出电路实际上这时的输出电路与输入的端口处于隔离状态。


黄色的高亮部分显示了数据传输通道,外部的电平信号通过左边编号1的I/O端口进入STM32内部经过编号2的施密特触发器整形以后送入编号为3的“输入数据寄存器”,在“输入数据寄存器”的另一端(编号4),CPU通过内部的数据总线可以随时读出I/O端口的电平变化的状态。


GPIO带上拉输入_IPU模式工作原理:


上图是STM32的GPIO带上拉输入模式的原理图。


与前面介绍的浮空输入模式相比,仅仅是在数据通道上面,接入了一个上拉电阻,根据STM32的数据手册,这个上拉电阻阻值介于30K~50K欧姆。


同样,CPU可以随时在“输入数据寄存器”的另一端,通过内部的数据总线读出I/O端口的电平变化的状态。


GPIO带下拉输入_IPD模式工作原理:



对于输入下拉模式的输入,是在数据通道的下部,接入了一个下拉电阻。

根据STM32的数据手册,这个下拉电阻阻值也是介于30K~50K欧姆。


对于要加上拉或下拉电阻:

1.当作单片机作为输入时,假设我们直接在IO端口接一个按键到地(或电源)。

因为按键按,于不按管脚都是悬空的。单片机就很难检测按键是否按下。

所以人为的接一个上拉(或下拉)。以确定未按下的时候IO输入电平的状态

2.可以提高芯片的抗干扰能

3.当单片机的IO口作输出时,如果不接上拉电阻只能提供灌电流。无法输出电流驱动外接设备。这时也需要考虑上拉电阻。这样才可以使IO输出高电平


GPIO模拟输入_AIN模式工作原理:



如果把STM32配置为模拟输入模式时,工作原理就比较简单了,信号从左边编号为1的端口进从右边编号为2的一端直接进入STM32单片机的AD模块。


细心的朋友可以看到数据通道中上拉、下拉电阻和施密特触发器,这时均处于关断的状态,“输入数据寄存器”就不能反映IO端口上的电平变化的状态了,换句话说,也就是在模拟输入状态下,CPU不能通过“输入数据寄存器”读到IO端口变化的数据了


以上分析的是GPIO模块IO引脚的输入模式的工作原理,下面介绍一下GPIO输出模式的工作原理


GPIO开漏输出_OUT_OD模式工作原理


上图是GPIO开漏输出模式的工作原理图

当CPU在编号1端通过“位设置/清除寄存器”或“输出数据寄存器”写入数据后

该数据位将通过编号2的输出控制电路传送到编号4的I/O端口。

如果CPU写入的是逻辑“1 ”,则编号3的N-MOS管将处于关闭状态

此时I/O端口的电平将由外部的上拉电阻决定

如果CPU写入的是逻辑“0 ”,则编号3的N-MOS管将处于开启状态

此时I/O端口的电平被编号3的N-MOS管拉到了“地”的零电位。


在图中的上半部,施密特触发器处于开启状态

这意味着CPU可以在“输入数据寄存器”的另一端,随时可以监控I/O端口的状态

通过这个特性,还可以实现了虚拟的I/O端口双向通信:假如CPU输出逻辑“1 ”

由于编号3的N-MOS管处于关闭状态,I/O端口的电平将完全由外部电路决定

因此,CPU可以在“输入数据寄存器”读到外部电路的信号,而不是它自己输出的逻辑“1 ”


GPIO口的输出模式下,有3种输出速度可选(2MHz、10MHz和50MHz)

这个速度是指GPIO口驱动电路的响应速度,而不是输出信号的速度

输出信号的速度与程序有关(芯片内部在I/O口的输出部分安排了多个响应速度不同的输出驱动电路

用户可以根据自己的需要选择合适的驱动电路)。

通过选择速度来选择不同的输出驱动模块,达到最佳的噪声控制和降低功耗的目的。

高频的驱动电路,噪声很高  

当我们的项目不需要比较高的输出频率时,请选用低频驱动电路,这样非常有利于提高系统的EMI性能。

当然如果我们的项目要求输出较高频率的信号,但却选用了较低频率的驱动模块,很可能会得到比较失真的输出信号

GPIO推挽输出_OUT_PP模式工作原理



GPIO的推挽输出模式是在开漏输出模式的基础上,在“输出控制电路”之后,增加了一个P-MOS管

当CPU输出逻辑“1 ”时,编号3处的P-MOS管导通,而下方的N-MOS管截止,达到输出高电平的目的

当CPU输出逻辑“0 ”时,编号3处的P-MOS管截止,而下方的N-MOS管导通,达到输出低电平的目的

在这个模式下,CPU仍然可以从“输入数据寄存器”读到该IO端口电压变化的信号


GPIO开漏复用输出_AF_OD模式工作原理



GPIO的开漏复用输出模式与开漏输出模式的工作原理基本相同

不同的是编号为2的输入的源不同,它是和复用功能的输出端相连

此时的“输出数据寄存器”被输出通道给断开了。

从上面的这个图,我们还可以看到CPU同样可以从“输入数据寄存器”读取到外部IO端口变化的电平信号。


GPIO推挽复用输出_AF_PP模式工作原理



最后介绍一下GPIO推挽复用输出模式的工作原理

编号2“输出控制电路”输入是与复用功能的输出端相连

此时“输出数据寄存器”被从输出通道断开了,片上外设的输出信号直接与“输出控制电路”的输入端想连接。

我们将GPIO配置成复用输出功能后,假如相应的外设模块没有被激活,那么此时IO端口的输出将不确定。

其它部分原理与前面叙述的模式一样,包括对“输入数据寄存器”的读取方式也是一样的。跑马灯硬件



跑马灯接的是PD12,PD13,PD14,PD15 4个PIN脚


3.    跑马灯库函数编程

1)    需要的库函数文件:


头文件:stm32f4xx_gpio.h


源文件:stm32f4xx_gpio.c


2)    重要函数


1个初始化函数:


void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef*GPIO_InitStruct);


2个读取输入电平函数:


uint8_t GPIO_ReadInputDataBit(GPIO_TypeDef* GPIOx, uint16_tGPIO_Pin);


uint16_t GPIO_ReadInputData(GPIO_TypeDef* GPIOx);


2个读取输出电平函数:


uint8_t GPIO_ReadOutputDataBit(GPIO_TypeDef* GPIOx,uint16_t GPIO_Pin);


uint16_t GPIO_ReadOutputData(GPIO_TypeDef* GPIOx);


4个设置输出电平函数:


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, BitActionBitVal);


void GPIO_Write(GPIO_TypeDef* GPIOx, uint16_t PortVal);


3)    GPIO_Init初始化样例


GPIO_InitTypeDef GPIO_InitStructure;


RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);//使能GPIOF时钟


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13;//LED0和LED1对应IO口


GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;//普通输出模式


GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出


GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz


GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉


GPIO_Init(GPIOF, &GPIO_InitStructure);//初始化GPIOD12,13


4)    读取输入电平函数


2个读取输入电平函数:


uint8_t GPIO_ReadInputDataBit(GPIO_TypeDef* GPIOx, uint16_tGPIO_Pin);


作用:读取某个GPIO的输入电平。实际操作的是GPIOx_IDR寄存器。


例如:


GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_5);//读取GPIOA.5的输入电平


uint16_t GPIO_ReadInputData(GPIO_TypeDef* GPIOx);


作用:读取某组GPIO的输入电平。实际操作的是GPIOx_IDR寄存器。


例如:


GPIO_ReadInputData(GPIOA);//读取GPIOA组中所有io口输入电平


5)    读取输出电平函数


uint8_t GPIO_ReadOutputDataBit (GPIO_TypeDef* GPIOx,uint16_t GPIO_Pin);


作用:读取某个GPIO的输出电平。实际操作的是GPIO_ODR寄存器。


例如:


GPIO_ReadOutputDataBit(GPIOA, GPIO_Pin_5);//读取GPIOA.5的输出电平


uint16_t GPIO_ReadOutputData(GPIO_TypeDef* GPIOx);


作用:读取某组GPIO的输出电平。实际操作的是GPIO_ODR寄存器。


例如:


GPIO_ReadOutputData(GPIOA);//读取GPIOA组中所有io口输出电平


6)    4个设置输出电平函数


void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);


作用:设置某个IO口输出为高电平(1)。实际操作BSRRL寄存器


void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_tGPIO_Pin);


作用:设置某个IO口输出为低电平(0)。实际操作的BSRRH寄存器。


void GPIO_WriteBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin,BitAction BitVal);


void GPIO_Write(GPIO_TypeDef* GPIOx, uint16_t PortVal);


这两个函数不常用,也是用来设置IO口输出电平。


      以上就是整个GPIO库函数的使用,总结下GPIO库函数的步骤:


1)使能IO口时钟。调用函数RCC_AHB1PeriphClockCmd();


不同的外设调用的时钟使能函数可能不一样


如图:GPIO是在AHB1总线上

2)初始化IO口模式。调用函数GPIO_Init();


3)  3)操作IO口,输出高低电平。


    GPIO_SetBits();


    GPIO_ResetBits();


源码:


Led.h


#ifndef_LED_H_H_H

#define_LED_H_H_H

 

#include"stm32f4xx_gpio.h"

#include"stm32f4xx_rcc.h"

 

#defineLED_GREEN       GPIO_Pin_12

#defineLED_ORANGE      GPIO_Pin_13

#defineLED_RED         GPIO_Pin_14

#defineLED_BLUE        GPIO_Pin_15

 

#defineLED_ON 1

#defineLED_OFF 0

voidLED_Init(void);

voidLED_Operate(uint16_t GPIO_Pin,int32_t operate);

#endif

Led.c


#include"led.h"

 

voidLED_Init(void)

{

  GPIO_InitTypeDef GPIO_InitStruct;

 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD,ENABLE);

  GPIO_InitStruct.GPIO_Pin = GPIO_Pin_12 |GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;

  GPIO_InitStruct.GPIO_Mode = GPIO_Mode_OUT;

  GPIO_InitStruct.GPIO_Speed =GPIO_Speed_50MHz;

  GPIO_InitStruct.GPIO_OType = GPIO_OType_PP;

  GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_UP;

  GPIO_Init(GPIOD,&GPIO_InitStruct);

}

voidLED_Operate(uint16_t GPIO_Pin,int32_t operate)

{

  if(LED_ON == operate)

  {

    GPIO_SetBits(GPIOD,GPIO_Pin);

  }

  else if(LED_OFF == operate)

  {

    GPIO_ResetBits(GPIOD,GPIO_Pin);

  } 

}

Main.c


#include"led.h"

 

voidUser_Delay(__IO uint32_t nCount)

{

  while(nCount--)

  {

  }

}

intmain(void)

{

   LED_Init();

   while(1)

   {

     LED_Operate(LED_GREEN,LED_ON);

     User_Delay(0x3FFFFF);

     LED_Operate(LED_GREEN,LED_OFF);

     LED_Operate(LED_ORANGE,LED_ON);

     User_Delay(0x3FFFFF);

     LED_Operate(LED_ORANGE,LED_OFF);

     LED_Operate(LED_RED,LED_ON);

     User_Delay(0x3FFFFF);

     LED_Operate(LED_RED,LED_OFF);

     LED_Operate(LED_BLUE,LED_ON);

     User_Delay(0x3FFFFF);

     LED_Operate(LED_BLUE,LED_OFF);

     User_Delay(0x3FFFFF);

   }

   

}

推荐阅读

史海拾趣

ACP Optoelectronic Technology Co Ltd公司的发展小趣事

ACP Optoelectronic Technology Co Ltd,在创立初期便专注于光电技术的研发。某年,公司研发团队成功开发出一种新型光电传感器,其性能远超当时市场上的同类产品。这一技术突破迅速吸引了行业内的关注,公司因此获得了大量订单,业绩迅速攀升。随着技术的不断完善和市场的不断拓展,ACP Optoelectronic Technology Co Ltd逐渐在光电领域崭露头角。

酷宅(Coolkit)公司的发展小趣事

酷宅科技始终将技术创新作为公司发展的核心驱动力。公司不断投入研发资源,推动物联网、云计算、大数据等前沿技术的研发和应用。通过技术创新,酷宅科技不仅提高了产品的智能化水平,还降低了生产成本,提高了生产效率。同时,公司还积极与高校、科研机构等合作,共同开展技术攻关和人才培养工作,为公司的长期发展提供坚实的技术支持。

GAPTEC Electronic GmbH & Co. KG公司的发展小趣事

背景:随着技术的成熟,Galaxy Microelectronics开始寻求与大型电子产品制造商的合作。

发展:2005年,Galaxy Microelectronics与一家领先的智能手机制造商达成合作协议,为其旗舰产品提供定制化的NAND闪存解决方案。这一合作不仅让Galaxy Microelectronics的产品迅速进入市场,还显著提升了其品牌知名度。随后,公司陆续与多家知名电子产品厂商建立合作关系,进一步巩固了其在闪存市场的地位。

上海晶岳(AFSEMI)公司的发展小趣事

为了保持技术领先和市场竞争力,晶岳电子一直注重研发团队的建设。目前,公司在上海、深圳及台湾均建有成熟的产品研发团队,核心功率器件及集成电路设计人员均拥有15年以上的设计研发经验。这些优秀的研发人员不仅为公司带来了源源不断的技术创新,也确保了公司在激烈的市场竞争中始终保持领先地位。此外,晶岳电子还积极与高校和研究机构合作,共同开展前沿技术的研发和应用。

CUI公司的发展小趣事

在电子行业的初期,CUI技术的应用受限于其导电性能和稳定性。然而,随着科研人员的不断努力,新型的CUI材料逐渐问世,它们不仅导电性能优良,而且能够抵御各种环境因素的影响。与此同时,市场对于小型化、高集成度的电子产品需求日益增长,CUI技术因其能够在狭小空间内实现高效电路连接而备受青睐。

康奈特(CNNT)公司的发展小趣事

康奈特一直致力于拓展全球市场。公司通过参加国际展会、与海外经销商建立合作关系等方式,不断提升品牌在国际市场的知名度和影响力。同时,公司还针对不同国家和地区的市场需求和文化差异,推出了定制化的产品和服务。这些举措不仅帮助公司成功进入了多个海外市场,也为公司带来了丰厚的利润回报。

问答坊 | AI 解惑

SMT环境下的PCB设计技术详细

1.引言 SMT工艺是利用钎料或焊膏在元件与电路板连接之间构成机械与电气两方面的连接,其主要优点在于尺寸小、重量轻、互连性好;高频电路的性能好,寄生阻抗显著降低;抗冲击力与振动性能好。采用SMT工艺时引线不需穿过电路板,可避免产生引线接受 ...…

查看全部问答>

DIY光影棒在编程阶段应注意的几点事项

在这次论坛组织的DIY光影棒活动中,网友们表现积极,踊跃参加,到目前各位参与的网友们电路板都已焊接完毕,进入了编程阶段。由于我们这次参加的网友们几乎都是第一次DIY光影棒,所以,在确定方案时所设计的一些控制方法在编程阶段出现了不容易实现 ...…

查看全部问答>

模拟电路 复习思考题

康华光 第五版 模拟电路  394页 8.4.2 怎么解释 谢谢了 课后复习思考题 怎么没有答案啊 很苦恼…

查看全部问答>

TI M3 Day -- Stellaris family 技术培训会

TI 的 Stellaris产品系列凭借行业标准 ARM Cortex-M3 内核的通用处理能力和高级通信功能(包括 10/100 以太网 MAC+PHY、CAN、USB OTG、USB 主机/器件、SSI/SPI、UART、I2S 和 I2C)来满足主流 32 位 MCU 应用的要求。Stellaris 系列非常适合于要求 ...…

查看全部问答>

有项目寻技术合作:视频服务器的开发

如题,有兴趣可与我联系: QQ:76487572…

查看全部问答>

还在实习,有个公司叫去复试,说复试通过马上要过去工作,但是现在的项目要4月中旬才能完成,怎么办好??

还在实习,有个公司叫去复试,说复试通过马上要过去工作,但是现在的项目要4月中旬才能完成,怎么办好?? 好不容易等来了一个回应,还叫我回去复试,复试通过了就马上工作了,,,心想还是把实习的项目做完吧,现在也在研究TCPMP, 做完了播放器 ...…

查看全部问答>

如何使windows 2003系统的驱动,响应PCI-E设备的MSI中断

在给PCI-E设备做WDM驱动,设备采用MSI中断,但据说windows 2003操作系统不支持MSI中断,如何能实现呢? 以下摘自MSDN: MSI 只在Windows Vista/2008以上的系统支持,在Win2003/XP下是不支持的。 Message-signaled interrupts (MSIs) were intro ...…

查看全部问答>

好东西大家分享

偶发现了一个专门的手机开发网站,小弟不敢独,所以拿出来与大家分享啦. http://www.imhongyu.com…

查看全部问答>

xp应该如何响应中断请求?

我在XP下应该怎么知道有串口中断产生,然后去读串口的数据, 是否应该要写个驱动呢, 那要写什么类型的驱动,谢啦…

查看全部问答>

很神奇啊,高手看看

我写了一个1602显示时钟,用的是DS12C887,神奇的是写年的时候,01年,02年,都可以,写为03年或者以后就所有数据出错,?????高手解释一下,那里的问题???,而且,写成2002年12月31日24:59:59让他自己跑,很顺利就进入03年了   ...…

查看全部问答>