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 表达意图。
相关推荐
Teleger3 小时前
在window上使用c++控制鼠标点击,实现的exe
c++·单片机·计算机外设
黑白园5 小时前
STM32F103ZET6移植-电机2804-驱动板SimpleFOC Mini实现速度开环_位置开环控制(一、硬件介绍及接线)
stm32·单片机·嵌入式硬件
星夜夏空995 小时前
STM32单片机学习(12)——串口通信相关概念
stm32·单片机·学习
Stream_Silver6 小时前
【 libusb4java实战:跨平台USB设备通信完全指南】
java·笔记·嵌入式硬件·microsoft
黑白园6 小时前
STM32F103ZET6移植-电机2804(星型接法)-驱动板SimpleFOC Mini实现速度开环_位置开环控制(四、功能演示)
stm32·单片机·嵌入式硬件
Jack_02206 小时前
基于51单片机的停车场刷卡进出计费设计
单片机·嵌入式硬件·51单片机
振浩微433射频芯片6 小时前
433射频方案在远距离工业遥控中的应用解析:从TM-03到RM521的成熟之道
网络·单片机·嵌入式硬件·物联网·智能家居
Hello_Embed6 小时前
libmodbus 移植到 STM32H5
笔记·stm32·单片机·嵌入式硬件·嵌入式·ai编程
REDcker6 小时前
嵌入式MCU内存布局详解 Flash SRAM Keil MAP与启动分散加载实践
单片机·嵌入式硬件
qdprobot7 小时前
【无标题】
人工智能·单片机·嵌入式硬件·51单片机·硬件工程·iot·mixly