GPIO端口引脚(位域)打包技巧笔记

1. 结构体定义:位域打包

c

复制代码
typedef struct
{
    uint8_t num:4,   /*!< 引脚号 (0~15) */
            port:4;  /*!< 端口号 (0~15) */
} gpio_port_pin_t;
  • 使用位域将两个4位信息压缩到一个字节中。
  • 内存布局:低位4位存放 num,高位4位存放 port(取决于编译器,通常如此)。
  • 结构体大小 = 1字节。

2. 典型用法:从 uint8_t 解析出端口和引脚

c

复制代码
void gpio_ana_func1_init(uint8_t pin)
{
    // 将 pin 的地址强制转换为结构体指针
    gpio_port_pin_t *x = (gpio_port_pin_t *)&pin;

    // 使用 x->port 和 x->num 访问对应字段
    SYSC_AWO->IO[x->port].AE |= 1 << (16 + x->num);
}

关键点

  • &pin 取得变量 pin 的地址(uint8_t*)。
  • 强制转换为 gpio_port_pin_t*,将同一个字节重新解释为结构体。
  • 之后通过 x->portx->num 即可获得编码时存入的高4位和低4位。

3. 调用约定

通常调用者将端口和引脚编码为一个字节:

c

复制代码
// 假设端口=2, 引脚=5 → 编码为 (2<<4) | 5 = 0x25
gpio_ana_func1_init(0x25);

函数内部通过上述转换自然拆解出 port=2, num=5

4. 注意事项

  • 必须使用 **&**取地址(gpio_port_pin_t *)pin 是错误的,它会将 pin 的数值当作内存地址,导致非法访问。
  • 内存布局一致性:位域的内存分布可能因编译器而异,但在大多数嵌入式环境中此用法稳定。
  • 对齐安全uint8_t 与结构体大小均为1字节,对齐无问题。
  • 此技巧适用于函数内临时解析,不应用于长期存储或跨模块传递。

5. 优点

  • 代码简洁,无需手动移位和掩码。
  • 提高可读性,直接使用 x->portx->num 表达意图。
相关推荐
是翔仔呐2 小时前
第13章 SPI通信协议全解:底层时序、4种工作模式与W25Qxx Flash芯片读写实战
c语言·开发语言·stm32·单片机·嵌入式硬件·学习·gitee
悠哉悠哉愿意2 小时前
【单片机复习笔记】第十六届省赛复盘
笔记·单片机·嵌入式硬件
IT方大同2 小时前
RT_thread(RTOS实时操作系统)线程的创建与切换
c语言·开发语言·嵌入式硬件
是翔仔呐2 小时前
第14章 CAN总线通信全解:底层原理、帧结构与双机CAN通信实战
c语言·开发语言·stm32·单片机·嵌入式硬件·学习·gitee
沐欣工作室_lvyiyi3 小时前
基于单片机的智能浴室(论文+源码)
stm32·单片机·嵌入式硬件·智能浴室
二本咕咕-机械转码3 小时前
STM32是怎么跑起来的?启动流程 + 时钟树一次讲透(面试高频)
stm32·嵌入式硬件·面试
撩妹小狗3 小时前
揭秘舵机精准控制的PWM奥秘
单片机·嵌入式硬件
Heartache boy4 小时前
野火STM32_HAL库版课程笔记-TIM通道输出应用之PWM实现呼吸灯
笔记·stm32·单片机·嵌入式硬件
桌面运维家4 小时前
解决Windows 10打印机脱机:端口、驱动、网络故障排除
windows·stm32·单片机