[N32L4xx] 高效无隐患输出 IO 的方法

[复制链接]
 楼主| chenci2013 发表于 2025-4-23 17:09 | 显示全部楼层 |阅读模式

[i=s] 本帖最后由 chenci2013 于 2025-4-23 17:11 编辑 [/i]<br /> <br />

一般实现

void out_data(uint8_t byte) {
    if(byte & 1) {
        GPIOA->BSRR = ((uint16_t)byte << 1);
        // set
    } else {
        GPIOA->BRR = ((uint16_t)byte << 1);
        // reset
    }
    if(byte & 2) {
        GPIOA->BSRR = ((uint16_t)byte << 2);
        // set
    } else {
        GPIOA->BRR = ((uint16_t)byte << 2);
        // reset
    }
    if(byte & 4) {
        GPIOA->BSRR = ((uint16_t)byte << 3);
        // set
    } else {
        GPIOA->BRR = ((uint16_t)byte << 3);
        // reset
    }
}

高效实现 很简单,看下面的代码

//假设使用 PA0~7
void out_data(uint8_t byte)
{
GPIOA->BSRR = byte;  // set
byte = ~byte;
GPIOA->BRR = byte;  // reset
}

//假设使用 PA8~15

void out_data(uint8_t byte)
{
GPIOA->BSRR = ((uint16_t)byte << 8);  // set
byte = ~byte;
GPIOA->BRR = ((uint16_t)byte << 8);  // reset
}

这种操作并不会影响其他 IO 的输出,也防止了操作 ODR 寄存器可能造成的问题。

如果数据位是四位的又该如何控制呢?

//假设使用 PA1~5

void out_data(uint8_t byte)
{
byte &= 0x0f;    // 提取低 4 位数据
GPIOA->BSRR = ((uint16_t)byte << 1);  // set
byte = ~byte;
byte &= 0x0f;
GPIOA->BRR = ((uint16_t)byte << 1);  // reset
}

这样你就可以不用管到底该使用那个寄存器了。简单、方便、高效!!!

ccook11 发表于 2025-5-3 22:21 | 显示全部楼层
不同单片机的 IO 引脚可能具有不同的驱动能力、电气特性和复用功能。
sdlls 发表于 2025-5-3 23:18 | 显示全部楼层
合并多次小数据包为大数据块一次性发送,降低协议开销。
sdCAD 发表于 2025-5-4 00:51 | 显示全部楼层
实现自检功能,定期检查GPIO线路是否有短路或开路情况发生。
ulystronglll 发表于 2025-5-4 20:48 | 显示全部楼层
始终检查返回值,假设外设可能随时失效。
macpherson 发表于 2025-5-6 09:28 | 显示全部楼层
推挽输出(Push-Pull Output)具有较强的驱动能力,可以输出高电平和低电平,适用于需要较大电流驱动的场合。
elsaflower 发表于 2025-5-6 11:18 | 显示全部楼层
直接操作GPIO寄存器(如ODR、BSRR等)通常比通过库函数更高效,因为它们减少了函数调用的开销。
fengm 发表于 2025-5-6 13:04 | 显示全部楼层
避免数据溢出或丢失。              
i1mcu 发表于 2025-5-6 14:50 | 显示全部楼层
尽量减少在中断服务程序中进行 IO 操作,因为中断服务程序的执行时间会影响系统的实时性。若必须在中断中进行 IO 操作,要确保操作简单且快速。
uptown 发表于 2025-5-6 17:04 | 显示全部楼层
对于大量数据传输,使用DMA(直接内存访问)可以减少CPU的负担。
10299823 发表于 2025-5-6 18:48 | 显示全部楼层
对于需要精确延时的应用,可以利用硬件定时器而不是软件延时函数。
primojones 发表于 2025-5-6 20:32 | 显示全部楼层
采用批量操作的方式,例如在需要同时控制多个 IO 引脚时,使用寄存器的批量设置功能,而不是逐个引脚进行操作,这样可以减少指令执行时间,提高效率。
everyrobin 发表于 2025-5-6 22:14 | 显示全部楼层
防止外设无响应导致死循环。              
bestwell 发表于 2025-5-11 10:03 | 显示全部楼层
如果在中断服务程序(ISR)中更改GPIO状态,确保不会干扰主循环中的GPIO操作。
primojones 发表于 2025-5-11 11:56 | 显示全部楼层
选择合适的IO引脚              
everyrobin 发表于 2025-5-11 17:11 | 显示全部楼层
在硬件设计阶段,要考虑软件的实现方式和要求;在软件编程阶段,要充分利用硬件的特性和优势。
mollylawrence 发表于 2025-5-13 15:50 | 显示全部楼层
在开发阶段,可以通过串口打印调试信息来监控GPIO的状态变化,但应避免在最终产品中保留过多的日志输出以免影响性能。
jkl21 发表于 2025-5-13 17:57 | 显示全部楼层
考虑添加静电放电(ESD)保护二极管,特别是在易受静电影响的环境中工作时。
1988020566 发表于 2025-5-13 19:39 | 显示全部楼层
在PCB设计中,合理布局布线,确保信号线的长度和宽度适当,减少信号衰减和反射。
sanfuzi 发表于 2025-5-13 21:30 | 显示全部楼层
使用 DMA、硬件流控制、高速接口
您需要登录后才可以回帖 登录 | 注册

本版积分规则

124

主题

6644

帖子

4

粉丝
快速回复 在线客服 返回列表 返回顶部

124

主题

6644

帖子

4

粉丝
快速回复 在线客服 返回列表 返回顶部