则相应的GPIO口如何配置
① 如何实现gpio口模式的配置
一、 STM32的输入输出管脚有下面8种(4输入 2输出 2复用输出)可能的配置:
① 浮空输入_IN_FLOATING
② 带上拉输入_IPU
③ 带下拉输入_IPD
④ 模拟输入_AIN
⑤ 开漏输出_OUT_OD
⑥ 推挽输出_OUT_PP
⑦ 复用功能的推挽输出_AF_PP
⑧ 复用功能的开漏输出_AF_OD
1.1 I/O口的输出模式下,有3种输出速度可选(2MHz、10MHz和50MHz),这个速度是指I/O口驱动电路的响应速度而不是输出信号的速度,输出信号的速度与程序有关(芯片内部在I/O口 的输出部分安排了多个响应速度不同的输出驱动电路,用户可以根据自己的需要选择合适的驱动电路)。通过选择速度来选择不同的输出驱动模块,达到最佳的噪声控制和降低功耗的目的。高频的驱动电路,噪声也高,当不需要高的输出频率时,请选用低频驱动电路,这样非常有利于提高系统的EMI性能。当然如果要输出较高频率的信号,但却选用了较低频率的驱动模块,很可能会得到失真的输出信号。
输出速度又称输出驱动电路的响应速度,可理解为:输出驱动电路的带宽,即一个驱动电路可以不失真地通过信号的最大频率。
如果一个信号的频率超过了驱动电路的响应速度,就有可能信号失真。如果信号频率为10MHz,而你配置了2MHz的带宽,则10MHz的方波很可能就变成了正弦波。就好比是公路的设计时速,汽车速度低于设计时速时,可以平稳地运行,如果超过设计时速就会颠簸,甚至翻车。
关键是: GPIO的引脚速度跟应用相匹配,速度配置越高,噪声越大,功耗越大。
带宽速度高的驱动器耗电大、噪声也大,带宽低的驱动器耗电小、噪声也小。使用合适的驱动器可以降低功耗和噪声。
GPIO的引脚速度跟应用匹配(推荐10倍以上)。比如:
1.1.1 对于串口,假如最大波特率只需115.2k,那么用2M的GPIO的引脚速度就够了,既省电也噪声小。
1.1.2 对于I2C接口,假如使用400k波特率,若想把余量留大些,那么用2M的GPIO的引脚速度或许不够,这时可以选用10M的GPIO引脚速度。
1.1.3 对于SPI接口,假如使用18M或9M波特率,用10M的GPIO的引脚速度显然不够了,需要选用50M的GPIO的引脚速度。
1.2 GPIO口设为输入时,输出驱动电路与端口是断开,所以输出速度配置无意义。
1.3 在复位期间和刚复位后,复用功能未开启,I/O端口被配置成浮空输入模式。
1.4 所有端口都有外部中断能力。为了使用外部中断线,端口必须配置成输入模式。
1.5 GPIO口的配置具有上锁功能,当配置好GPIO口后,可以通过程序锁住配置组合,直到下次芯片复位才能解锁。
二、GPIO的翻转速度指:输入/输出寄存器的0 ,1 值反映到外部引脚(APB2上)高低电平的速度.手册上指出GPIO最大翻转速度可达18MHz。通过简单的程序测试,用示波器观察到的翻转时间是综合的时间,包括取指令的时间、指令执行的时间、指令执行后信号传递到寄存器的时间(这其中可能经过很多环节,比如AHB、APB、总线仲裁等),最后才是信号从寄存器传输到引脚所经历的时间。如有上拉电阻,其阻值越大,RC延时越大,即逻辑电平转换的速度越慢,功耗越大。
三、在STM32中如何配置片内外设使用的IO端口
首先,一个外设经过 ①配置输入的时钟和 ②初始化后即被激活(开启);③如果使用该外设的输入输出管脚,则需要配置相应的GPIO端口(否则该外设对应的输入输出管脚可以做普通GPIO管脚使用);④再对外设进行详细配置。
对应到外设的输入输出功能有下述三种情况:
① 外设对应的管脚为输出:需要根据外围电路的配置选择对应的管脚为复用功能的推挽输出或复用功能的开漏输出。
② 外设对应的管脚为输入:则根据外围电路的配置可以选择浮空输入、带上拉输入或带下拉输入。
③ ADC对应的管脚:配置管脚为模拟输入。
如果把端口配置成复用输出功能,则引脚和输出寄存器断开,并和片上外设的输出信号连接。将管脚配置成复用输出功能后,如果外设没有被激活,那么它的输出将不确定。
四、 通用IO端口(GPIO)初始化
4.1 GPIO初始化
41.1 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | B | C, ENABLE):使能APB2总线外设时钟;
41.2 RCC_ APB2PeriphResetCmd (RCC_APB2Periph_GPIOA | B | C, DISABLE):释放GPIO复位。
4.2 置各个PIN端口(模拟输入_AIN、输入浮空_IN_FLOATING、输入上拉_IPU、输入下拉_IPD、开漏输出_OUT_OD、推挽式输出_OUT_PP、推挽式复用输出_AF_PP、开漏复用输出_AF_OD)。
4.3GPIO初始化完成。
五、 的GPIO操作函数
uint8_t GPIO_ReadInputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);//读GPIO某一位的输入
uint16_t GPIO_ReadInputData(GPIO_TypeDef* GPIOx);//读GPIO的输入
uint8_t GPIO_ReadOutputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);//读GPIO某一位的输出
uint16_t GPIO_ReadOutputData(GPIO_TypeDef* GPIOx);//读GPIO的输出
void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);//将GPIO的某个位置位
void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);//将GPIO的某个位复位
void GPIO_WriteBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin, BitAction BitVal);//写GPIO的某个位
void GPIO_Write(GPIO_TypeDef* GPIOx, uint16_t PortVal);//写GPIO
六、管脚的复用功能 重映射
1、复用功能:内置外设是与I/O口共用引出管脚(不同的功能对应同一管脚)
STM32 所有内置外设的外部引脚都是与标准GPIO引脚复用的,如果有多个复用功能模块对应同一个引脚,只能使能其中之一,其它模块保持非使能状态。
2、重映射功能:复用功能的引出脚可以通过重映射,从不同的I/O管脚引出,即复用功 能的引出脚位是可通过程序改变到其他的引脚上!
直接好处:PCB电路板的设计人员可以在需要的情况下,不必把某些信号在板上绕一大圈完成联接,方便了PCB的设计同时潜在地减少了信号的交叉干扰。
如:USART1: 0: 没有重映像(TX/PA9,RX/PA10); 1: 重映像(TX/PB6,RX/PB7)。
(参考AFIO_MAPR寄存器介绍)[0,1为一寄存器的bit值]
【注】 下述复用功能的引出脚具有重映射功能:
- 晶体振荡器的引脚在不接晶体时,可以作为普通I/O口
- CAN模块; - JTAG调试接口;- 大部分定时器的引出接口; - 大部分USART引出接口
- I2C1的引出接口; - SPI1的引出接口;
举例:对于STM32F103VBT6,47引脚为PB10,它的复用功能是I2C2_SCL和 USART3_TX,表示在上电之后它的默认功能为PB10,而I2C2的SCL和USART3的TX为它的复用功能;另外在TIM2的引脚重映射后,TIM2_CH3也成为这个引脚的复用功能。
(1)要使用STM32F103VBT6的47、48脚的USART3功能,则需要配置47脚为复用推挽输出或复用开漏输出,配置48脚为某种输入模式,同时使能USART3并保持I2C2的非使能状态。
(2)使用STM32F103VBT6的47脚作为TIM2_CH3,则需要对TIM2进行重映射,然后再按复用功能的方式配置对应引脚。
② stm32如果一个gpio口既需要输出也需要输入,应该怎么配置
STM32F407ZG集成8个16引脚GPIO端口,支持168MHz最高时钟频率,可承受5V电压。每个端口拥有16个GPIO引脚,功能独立配置,具备双向保护二极管与可选上拉或下拉电阻。输入/输出数据可单独读取或输出。端口配置模式包括输入浮点、输出、开漏/推挽及外部中断/事件。在复位时,I/O端口默认为输入浮点模式,提供微弱内部电阻。输出模式下,可选开漏或推挽类型,速度取决于VDD值。每个端口支持16条外部中断/事件线,具有23个边缘检测器用于生成中断请求。
为实现4个按键控制2个LED灯的实验,选择正点原子原理图中的4个按键与2个LED灯。按键配置为输入模式,使用上拉或下拉电阻保持断开状态,LED灯配置为输出模式,用于显示状态。通过STM32CubeMX配置GPIO端口,将按键与LED灯与具体引脚关联。选择外部晶振作为时钟源,配置JTAG调试接口,设置时钟频率至168MHz,完成系统配置。
使用STM32CubeIDE集成开发环境进行代码编写、编译与调试。该环境整合了STM32CubeMX配置工具和Eclipse IDE,提供一站式解决方案。实现按键触发LED灯的轮询Polling方式代码,检测按键按下状态,点亮相应LED灯,反之关闭,完成功能验证。
③ 请问如何根据硬件或是数据手册配置GPIO的输入输出模式
用的最多的推挽输出 模拟输入,用库的方式直接写就完了,用寄存器麻烦一点。现在已经很多资料了,库的方式很多,寄存器的正点原子的不错。希望帮到你。
④ 求一个stm32的GPIO配置函数,就是使用这个函数可以配置任何一个GPIO端口(如下)
voidconfig_gpio(u32PORTx,u32MODE,u32PINxs,PULL_STATstat)//PORTx=0,则代表GPIOA,类推。PINxs从0开始。MODE和stat见如下对照表,需要你自己实现PULL_STAT枚举型
{
/*IO配置对照,4bit一个引脚(y)
高2位(CNFy):
输入00=模拟,01=浮空,10=上/下拉,11=保留
输出[推挽][开漏]
[通用]0001
[复用]1011
低2位(MODEy):
输入00
输出01=10MHz,10=2MHz,11=50MHz
*/
GPIO_TypeDef*GPIOs[]={GPIOA,GPIOB,GPIOC,GPIOD,GPIOE,GPIOF,GPIOG};
MODE|=stat<<2;
if(PINxs>15)
{
PINxs-=8;
GPIOs[PORTx]->CRH=MODE<<(PINxs*4);
}
else
{
GPIOs[PORTx]->CRL=MODE<<(PINxs*4);
}
}
至于
voidset_PINxs(u32 PORTx,u32PINxs,PIN_VALval)
{
}
可以用位带实现,如下:
保存为system.h
#ifndef__SYS_H
#define__SYS_H
#include"stm32f10x.h"
//0,不支持ucos
//1,支持ucos
#defineSYSTEM_SUPPORT_UCOS 0 //定义系统文件夹是否支持UCOS
//位带操作,实现51类似的GPIO控制功能
//具体实现思想,参考<<CM3权威指南>>第五章(87页~92页).
//IO口操作宏定义
#defineBITBAND(addr,bitnum)((addr&0xF0000000)+0x2000000+((addr&0xFFFFF)<<5)+(bitnum<<2))
#defineMEM_ADDR(addr)*((volatileunsignedlong*)(addr))
#defineBIT_ADDR(addr,bitnum)MEM_ADDR(BITBAND(addr,bitnum))
//IO口地址映射
#defineGPIOA_ODR_Addr(GPIOA_BASE+12)//0x4001080C
#defineGPIOB_ODR_Addr(GPIOB_BASE+12)//0x40010C0C
#defineGPIOC_ODR_Addr(GPIOC_BASE+12)//0x4001100C
#defineGPIOD_ODR_Addr(GPIOD_BASE+12)//0x4001140C
#defineGPIOE_ODR_Addr(GPIOE_BASE+12)//0x4001180C
#defineGPIOF_ODR_Addr(GPIOF_BASE+12)//0x40011A0C
#defineGPIOG_ODR_Addr(GPIOG_BASE+12)//0x40011E0C
#defineGPIOA_IDR_Addr(GPIOA_BASE+8)//0x40010808
#defineGPIOB_IDR_Addr(GPIOB_BASE+8)//0x40010C08
#defineGPIOC_IDR_Addr(GPIOC_BASE+8)//0x40011008
#defineGPIOD_IDR_Addr(GPIOD_BASE+8)//0x40011408
#defineGPIOE_IDR_Addr(GPIOE_BASE+8)//0x40011808
#defineGPIOF_IDR_Addr(GPIOF_BASE+8)//0x40011A08
#defineGPIOG_IDR_Addr(GPIOG_BASE+8)//0x40011E08
//IO口操作,只对单一的IO口!
//确保n的值小于16!
#definePAout(n)BIT_ADDR(GPIOA_ODR_Addr,n)//输出
#definePAin(n)BIT_ADDR(GPIOA_IDR_Addr,n)//输入
#definePBout(n)BIT_ADDR(GPIOB_ODR_Addr,n)//输出
#definePBin(n)BIT_ADDR(GPIOB_IDR_Addr,n)//输入
#definePCout(n)BIT_ADDR(GPIOC_ODR_Addr,n)//输出
#definePCin(n)BIT_ADDR(GPIOC_IDR_Addr,n)//输入
#definePDout(n)BIT_ADDR(GPIOD_ODR_Addr,n)//输出
#definePDin(n)BIT_ADDR(GPIOD_IDR_Addr,n)//输入
#definePEout(n)BIT_ADDR(GPIOE_ODR_Addr,n)//输出
#definePEin(n)BIT_ADDR(GPIOE_IDR_Addr,n)//输入
#definePFout(n)BIT_ADDR(GPIOF_ODR_Addr,n)//输出
#definePFin(n)BIT_ADDR(GPIOF_IDR_Addr,n)//输入
#definePGout(n)BIT_ADDR(GPIOG_ODR_Addr,n)//输出
#definePGin(n)BIT_ADDR(GPIOG_IDR_Addr,n)//输入//voidNVIC_Configuration(void);#endif
使用时直接:
PAout(0)=0;
//或
a=PAin(0);
当然,你还可以使用STM32的库函数
⑤ 如何实现一次对多GPIO进行配置
以LED灯为例子:
void led(void)
{
/*定义一个GPIO_InitTypeDef类型的结构体*/
GPIO_InitTypeDef GPIO_InitStructure;
/*开启GPIOB的外设时钟*/
RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOA, ENABLE);
/*选择要控制的GPIOB引脚*/
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_8;
/*设置引脚模式为通用推挽输出*/
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
/*设置引脚速率为50MHz */
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
/*调用库函数,初始化GPIOB*/
GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_Init(GPIOA, &GPIO_InitStructure);
/* 让选择引脚输出高电平*/
GPIO_SetBits(GPIOB, GPIO_Pin_0 | GPIO_Pin_1);
}
其中GPIO有多个模式,需要按照不同功能和使用来配置。这里LED灯需要输出,所以我设置为通用推挽。