stm32库函数GPIO_Init()解析

本文深入探讨了GPIO_Init函数的使用,包括引脚初始化、配置速度和模式的方法。通过详细解析函数内部逻辑,读者可以更好地理解如何在STM32微控制器中进行GPIO配置。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

作者:pipiTong

转自:http://blog.chinaunix.net/uid-27021180-id-3230782.html


GPIO_Init函数是IO引脚的初始化函数,进行个个引脚的初始化配置,主要接受两个参数,一个是配置引脚组(GPIO_TypeDef* GPIOx),一个是配置的参数( GPIO_InitTypeDef* GPIO_InitStruct),具体如下


点击(此处)折叠或打开

  1. void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef* GPIO_InitStruct)
  2. /*其中第一个参数为那组引脚,每组拥有16个引脚,每组都具有不同的寄存器配置地址,第二个参数是一个数据结构,也就是将基本配置信息放在这个数据结构里面,再将这个结构传入函数进行配置*/
  3. //其中数据机构可以表示为如下
  4. typedef struct
  5. {
  6.   uint16_t GPIO_Pin; //引脚号
  7.   GPIOSpeed_TypeDef GPIO_Speed; //配置速度
  8.   GPIOMode_TypeDef GPIO_Mode; //工作模式
  9. }GPIO_InitTypeDef;

  10. //其中配置模式和工作模式为GPIOSpeed_TypeDef和GPIOMode_TypeDef的枚举变量
为了方面的解析这个函数我们需要把几个常量的定义罗列一下

点击(此处)折叠或打开

  1. //首先是引脚定义
  2. #define GPIO_Pin_0 ((uint16_t)0x0001) /* Pin 0 selected */
  3. #define GPIO_Pin_1 ((uint16_t)0x0002) /* Pin 1 selected */
  4. #define GPIO_Pin_2 ((uint16_t)0x0004) /* Pin 2 selected */
  5. #define GPIO_Pin_3 ((uint16_t)0x0008) /* Pin 3 selected */
  6. #define GPIO_Pin_4 ((uint16_t)0x0010) /* Pin 4 selected */
  7. #define GPIO_Pin_5 ((uint16_t)0x0020) /* Pin 5 selected */
  8. #define GPIO_Pin_6 ((uint16_t)0x0040) /* Pin 6 selected */
  9. #define GPIO_Pin_7 ((uint16_t)0x0080) /* Pin 7 selected */
  10. #define GPIO_Pin_8 ((uint16_t)0x0100) /* Pin 8 selected */
  11. #define GPIO_Pin_9 ((uint16_t)0x0200) /* Pin 9 selected */
  12. #define GPIO_Pin_10 ((uint16_t)0x0400) /* Pin 10 selected */
  13. #define GPIO_Pin_11 ((uint16_t)0x0800) /* Pin 11 selected */
  14. #define GPIO_Pin_12 ((uint16_t)0x1000) /* Pin 12 selected */
  15. #define GPIO_Pin_13 ((uint16_t)0x2000) /* Pin 13 selected */
  16. #define GPIO_Pin_14 ((uint16_t)0x4000) /* Pin 14 selected */
  17. #define GPIO_Pin_15 ((uint16_t)0x8000) /* Pin 15 selected */
  18. #define GPIO_Pin_All ((uint16_t)0xFFFF) /* All pins selected */

  19. //其次是模式定义
  20. typedef enum
  21. { GPIO_Mode_AIN = 0x0, //模拟输入
  22. GPIO_Mode_IN_FLOATING = 0x04, //浮空输入模式, 默认
  23. GPIO_Mode_IPD = 0x28, //上拉/下拉输入模式
  24. GPIO_Mode_IPU = 0x48, //保留
  25. GPIO_Mode_Out_OD = 0x14, //通用开漏输出
  26. GPIO_Mode_Out_PP = 0x10, //通用推挽输出
  27. GPIO_Mode_AF_OD = 0x1C, //复用(开漏)输出
  28. GPIO_Mode_AF_PP = 0x18 //复用(推挽)输出
  29. }GPIOMode_TypeDef;

  30. //最后是速度定义
  31. typedef enum
  32. {
  33. GPIO_Speed_10MHz = 1,
  34. GPIO_Speed_2MHz,
  35. GPIO_Speed_50MHz
  36. }GPIOSpeed_TypeDef;
其中引脚定义很容以看出引脚0则16位的第0位置1 ,引脚为2则第2位置1,一次类推,所以如果要定义多个引脚只需要使用或逻辑运算(|)
其次模式定义也有他的规律,参考《stmf10xxx参考手册》可以得出知道存储的高低配置寄存器中每4位表达一个模式+速度,其中模式占高2位,速度占低2位,16个引脚就拥有4*16=64位来存储,所以这样定义就有它的道理,原因就是模式占高位例如10表示上拉/下拉输出模式,则在4位中占高2位,就应该是1000,低2位先用00表示,这样的话上拉/下拉输出模式就可以表示为1000=0x8,或者用0x28的第四位表示也可以,其它也是一样,就是里面的第五位1表示输出模式,0表示输入模式。
速度就直接用1,2,3来表示
具体的函数分析如下

点击(此处)折叠或打开

  1. void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef* GPIO_InitStruct)
  2. {
  3.  /*初始化各个变量*/
  4.   uint32_t currentmode = 0x00, currentpin= 0x00, pinpos= 0x00, pos= 0x00;
  5.   uint32_t tmpreg = 0x00, pinmask= 0x00;
  6.   //currentmode 用于存放临时的LCIR
  7.   //currentpin 用于存放配置的引脚位
  8.   //pinpos 用于存放当前操作的引脚号
  9.   //pos 存放当前操作的引脚位
  10.   //tmreg 当前的CIR
  11.   //pinmask


  12.   //判断参数
  13.   assert_param(IS_GPIO_ALL_PERIPH(GPIOx));
  14.   assert_param(IS_GPIO_MODE(GPIO_InitStruct->GPIO_Mode));
  15.   assert_param(IS_GPIO_PIN(GPIO_InitStruct->GPIO_Pin));
  16.   
  17.   //取出配置信息里面的模式信息并且取它的低4位
  18.   currentmode = ((uint32_t)GPIO_InitStruct->GPIO_Mode)&((uint32_t)0x0F);
  19.   if ((((uint32_t)GPIO_OInitStruct->GPIO_Mode)&((uint32_t)0x10))!= 0x00)//输出模式
  20.   {

  21.     //判断参数
  22.     assert_param(IS_GPIO_SPEED(GPIO_InitStruct->GPIO_Speed));
  23.     
  24.     //将速度信息放入currentmode低二位
  25.     currentmode |=(uint32_t)GPIO_InitStruct->GPIO_Speed;
  26.   }
  27.   if (((uint32_t)GPIO_InitStruct->GPIO_Pin&((uint32_t)0x00FF))!= 0x00)//引脚有定义
  28.   {
  29.     //当前的CRL保存
  30.     tmpreg = GPIOx->CRL;

  31.     //循环低八位引脚
  32.     for (pinpos= 0x00; pinpos< 0x08; pinpos++)
  33.     {
  34.       //当前是那个引脚,那个位置1
  35.       pos = ((uint32_t)0x01)<< pinpos;

  36.       //读取引脚信息里面的当前引脚
  37.       currentpin = (GPIO_InitStruct->GPIO_Pin)& pos;
  38.       if (currentpin== pos)    //如果当前引脚在配置信息里存在
  39.       {
  40.         pos = pinpos << 2;//pos=引脚号x2
  41.         pinmask = ((uint32_t)0x0F)<< pos;//1111<<引脚号x2,根据CRL 的结构很容易理解
  42.         tmpreg &=~pinmask;//当前应该操作的CRL位清0
  43.         tmpreg |=(currentmode<< pos);//设置当前操作的CRL 位
  44.         if (GPIO_InitStruct->GPIO_Mode== GPIO_Mode_IPD)// 端口置为高电平
  45.         {
  46.           GPIOx->BRR=(((uint32_t)0x01)<< pinpos);
  47.         }
  48.         else
  49.         {
  50.           if (GPIO_InitStruct->GPIO_Mode== GPIO_Mode_IPU)    // 端口清0
  51.           {
  52.             GPIOx->BSRR=(((uint32_t)0x01)<< pinpos);
  53.           }
  54.         }
  55.       }
  56.     }
  57.     GPIOx->CRL= tmpreg;
  58.   }
最后就是把配置好的CRL传入CRL寄存器。设置完毕


<think>好的,我现在要解决用户的问题:如何在AT32库函数中实现STM32GPIO_Mode_IPU等效功能。首先,我需要回忆STM32GPIO_Mode_IPU的定义。IPU是内部上拉输入模式,即GPIO配置为上拉输入,用于检测外部信号,当外部无信号时,引脚电平由上拉电阻拉至高电平。 接下来,我需要了解AT32GPIO配置方式。AT32STM32库函数可能有相似之处,但具体命名或结构可能不同。用户提到他们之前使用STM32库函数,现在需要迁移到AT32,所以需要找到对应的配置模式。 查阅AT32库函数文档,比如AT32gpio.h或相关用户手册,寻找输入模式的相关定义。STM32GPIO_Mode_IPU对应的是上拉输入模式,在AT32中可能类似,但枚举名称可能不同。例如,AT32可能使用GPIO_MODE_INPUT或GPIO_MODE_IPU这样的枚举值。 另外,需要注意AT32GPIO配置结构体是否与STM32一致。STM32中使用GPIO_InitTypeDef结构体,包含Pin、Mode、Speed等成员。AT32可能也有类似的结构体,但需要确认成员名称和可选值是否正确。 假设在AT32中,上拉输入模式对应的模式名为GPIO_MODE_INPUT_PULLUP,或者需要单独设置上拉电阻。有些库函数可能需要分别设置输入模式和上拉电阻,比如先设置模式为输入,再使能上拉。 例如,AT32可能有GPIO_InitStructure.GPIO_Mode = GPIO_MODE_INPUT,然后通过另一个函数或结构体成员来配置上拉,比如GPIO_InitStructure.GPIO_Pull = GPIO_PULL_UP。 如果用户之前使用的STM32代码是GPIO_Mode_IPU,那么在AT32中可能需要将模式设置为输入,并单独启用上拉。需要确认AT32库函数是否将模式和上下拉分开设置,或者是否有直接的组合模式。 另外,参考用户提供的引用内容,其中提到了STM32的复用功能配置,但当前问题与输入模式相关,可能不需要涉及复用功能。但需要注意,AT32GPIO配置可能在某些细节上与STM32不同,比如寄存器的位设置。 可能的解决方案步骤: 1. 查找AT32GPIO模式枚举,寻找输入上拉对应的模式。 2. 检查是否有单独的上拉/下拉配置选项。 3. 编写示例代码,初始化GPIO时设置正确的模式和上拉。 4. 参考AT32的官方文档或库函数头文件确认。 如果AT32库函数没有直接对应的GPIO_Mode_IPU,可能需要组合模式和上拉配置。例如,模式设为输入,然后使能上拉电阻。比如: 在STM32中: GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; 在AT32中可能需要: GPIO_InitStructure.GPIO_Mode = GPIO_MODE_INPUT; GPIO_InitStructure.GPIO_Pull = GPIO_PULL_UP; 或者,如果AT32的模式已经包含上拉选项: GPIO_InitStructure.GPIO_Mode = GPIO_MODE_INPUT_PULLUP; 需要具体查看AT32库函数定义。例如,在AT32F403A的库中,GPIO的模式定义可能有GPIO_MODE_INPUT和GPIO_PULL_UP等参数。 此外,用户可能需要参考AT32的参考手册或库函数示例代码,确认正确的配置方式。如果用户提供的引用中有关于AT32的初始化结构体,可能需要调整结构体成员的赋值。 总结步骤: - 确认AT32GPIO模式选项,是否有输入上拉模式。 - 调整GPIO初始化结构体的Mode和Pull成员。 - 提供示例代码,并建议查阅AT32的官方文档。</think>在AT32库函数中,STM32的`GPIO_Mode_IPU`(内部上拉输入模式)可通过以下方式实现: --- ### 等效功能实现方式 1. **模式选择**:AT32GPIO模式定义与STM32类似,但命名可能不同。上拉输入模式通常通过组合**输入模式**和**上拉电阻使能**实现。 2. **参数对应关系**: - STM32的`GPIO_Mode_IPU`对应AT32的: ```c GPIO_Mode_INPUT // 输入模式 GPIO_Pull_UP // 使能内部上拉电阻 ``` - 需分别设置**模式**和**上下拉配置**。 --- ### 替换代码示例 ```c // 定义GPIO初始化结构体 GPIO_InitType GPIO_InitStructure; // 配置引脚为输入模式,并启用上拉 GPIO_InitStructure.GPIO_Pins = GPIO_Pins_0; // 具体引脚号 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_INPUT; // 输入模式 GPIO_InitStructure.GPIO_Pull = GPIO_Pull_UP; // 使能上拉 GPIO_Init(GPIOA, &GPIO_InitStructure); // 初始化GPIOA ``` --- ### 关键差异说明 1. **分离模式与上下拉**: AT32将**输入/输出模式**和**上下拉电阻配置**分离,而STM32的部分模式(如`IPU`)是组合参数。需显式设置`GPIO_Pull`字段[^1]。 2. **寄存器映射兼容性**: AT32GPIO外设寄存器与STM32高度兼容,但需注意库函数接口的命名差异(如`GPIO_InitType`结构体)[^2]。 --- ### 参考文档建议 1. **AT32固件库手册**:查看`gpio.h`中的`GPIO_Mode`和`GPIO_Pull`枚举定义。 2. **应用笔记AN0012**:通常包含GPIO配置的迁移指南。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值