0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

给大家分享一个非常简单的IO输入输出框架

FPGA之家 来源:鱼鹰谈单片机 作者:鱼鹰谈单片机 2021-07-23 10:45 次阅读

在一个嵌入式系统中,可能存在许多输入或输出的IO口,输入有霍尔传感器红外对管等,输出有LED电源控制开关等。

如果说硬件可以一次成型,那么随便一份代码都可以完成IO的配置工作,但研发阶段的产品,硬件各种修改是难免的,每一次 IO 的修改,对于底层开发人员来说,可能都是一次挑战。

因为一旦有某一个 IO 配置错误,或者原来的配置没有修改正确(比如一个 IO 在原来的硬件适配中是输入,之后的硬件需要修改成输出),那么你很难查出来这是什么问题,因为这个时候不仅硬件修改了,软件也修改了,你需要先定位到底是软件问题还是硬件问题,所以一个好用的 IO 的配置框架就显得很有必要了。

有道友会说,不如使用 CubeMx 软件进行开发吧。

1、这个软件适用于 ST 单片机,以前还能用,现在,除非你家里有矿,不然谁用的起STM32?基本上都国产化了(虽然有些单片机号称兼容,但到底还是有些差异的)。

2、公司原本的代码就是使用标准库,只是因为IO 的变化,你就需要把整个库换掉吗?时间上允许吗?你确定修改后不会出现大问题?

3、国产化的芯片可没有所谓的标准库和HAL库供你选择,每一家都有各自的库,如果你的产品临时换方案怎么办?

4、HAL 效率问题。

今天鱼鹰介绍一个简单实用的框架,可用于快速增加或修改IO配置,甚至修改底层库。

假设有3个 LED 作为输出、3 个霍尔传感器作为输入:

输入配置代码:

#define GPIOx_Def GPIO_TypeDef*#define GPIOMode_Def GPIOMode_TypeDef

typedef struct{ GPIOx_Def gpio; uint16_t msk; GPIOMode_Def pull_up_down; } bsp_input_pin_def;

#define _GPIO_PIN_INPUT(id, pull, gpiox, pinx) [id].gpio = (GPIOx_Def)gpiox, [id].msk = (1 《《 pinx), [id].pull_up_down = (GPIOMode_Def)pull#define GPIO_PIN_INPUT(id, pull, gpiox, pinx) _GPIO_PIN_INPUT(id, pull, gpiox, pinx)

#define bsp_pin_get_port(gpiox) ((uint16_t)((GPIO_TypeDef *)gpiox)-》IDR)#define bsp_pin_get_value(variable,id) do{ bsp_pin_get_port(bsp_input_pin[id].gpio) & bsp_input_pin[id].msk ? variable |= (1 《《 id) : 0;} while(0)

#define BSP_GPIO_PUPD_NONE GPIO_Mode_IN_FLOATING#define BSP_GPIO_PUPD_PULLUP GPIO_Mode_IPU#define BSP_GPIO_PUPD_PULLDOWN GPIO_Mode_IPD

typedef enum{ PIN_INPUT_HALL_0 = 0, // 输入 IO 定义 PIN_INPUT_HALL_1, PIN_INPUT_HALL_2, PIN_INPUT_MAX}bsp_pin_input_id_def;

static const bsp_input_pin_def bsp_input_pin [PIN_INPUT_MAX] = { GPIO_PIN_INPUT(PIN_INPUT_HALL_0, BSP_GPIO_PUPD_NONE, GPIOA, 0), GPIO_PIN_INPUT(PIN_INPUT_HALL_1, BSP_GPIO_PUPD_NONE, GPIOB, 8), GPIO_PIN_INPUT(PIN_INPUT_HALL_2, BSP_GPIO_PUPD_NONE, GPIOE, 9), };

// 单个 IO 初始化函数 void bsp_pin_init_input(GPIOx_Def gpiox, uint32_t msk, GPIOMode_TypeDef pull_up_down){ uint32_t temp;

assert_param((msk & 0xffff0000) == 0 && gpiox != 0);

temp = ((uint32_t) gpiox - (uint32_t) GPIOA) / ( (uint32_t) GPIOB - (uint32_t) GPIOA);

/* enable the led clock */ RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA 《《 temp, ENABLE);

GPIO_InitTypeDef GPIO_InitStruct;

GPIO_InitStruct.GPIO_Mode = (GPIOMode_Def)pull_up_down; GPIO_InitStruct.GPIO_Pin = msk; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_2MHz;

GPIO_Init((GPIO_TypeDef*)gpiox, &GPIO_InitStruct);}

// 所有 IO 初始化void gpio_input_init(){ bsp_input_pin_def *info;

info = (bsp_input_pin_def *)&bsp_input_pin;

for(int i = 0; i 《 sizeof(bsp_input_pin)/sizeof(bsp_input_pin[0]); i++) { bsp_pin_init_input(info-》gpio, info-》msk, info-》pull_up_down); info++; } }

// 最多支持 32 个 IO 输入uint32_t bsp_input_all(void){ uint32_t temp = 0;

bsp_pin_get_value(temp, PIN_INPUT_HALL_0); bsp_pin_get_value(temp, PIN_INPUT_HALL_1); bsp_pin_get_value(temp, PIN_INPUT_HALL_2);

return temp;}

// 读取单个 IO 状态uint32_t bsp_input_level(bsp_pin_input_id_def id){ return (bsp_pin_get_port(bsp_input_pin[id].gpio) & bsp_input_pin[id].msk) ? 1 : 0;}

typedef enum{ HW_HAL_LEVEL_ACTIVE = 0, // 可直接修改为 0 或 1,另一个枚举值自动修改为相反值 HW_HAL_LEVEL_NO_ACTIVE = !HW_HAL_LEVEL_ACTIVE,}hw_input_hal_status_def;

typedef struct { hw_input_hal_status_def hal_level0; uint8_t hal_level1; uint8_t hal_level2;}bsp_input_status_def;

bsp_input_status_def bsp_input_status;

int main(void){ USRAT_Init(9600);//必须,进入调试模式后点击全速运行

gpio_input_init();

while(1) { uint32_t temp = bsp_input_all();

bsp_input_status.hal_level0 = (hw_input_hal_status_def)((temp 》》 PIN_INPUT_HALL_0) & 1); bsp_input_status.hal_level1 = ((temp 》》 PIN_INPUT_HALL_1) & 1); bsp_input_status.hal_level2 = ((temp 》》 PIN_INPUT_HALL_2) & 1); } }

调试的时候,我们可以很方便的查看每个 IO 的状态是怎样的,而不用管 0 或 1 到底代表什么意思:

输出配置代码:

#define GPIOx_Def GPIO_TypeDef*#define GPIOMode_Def GPIOMode_TypeDef

typedef struct{ GPIOx_Def gpio; uint32_t msk; uint32_t init_value; } bsp_output_pin_def;

#define _GPIO_PIN_OUT(id, gpiox, pinx, init) [id].gpio = gpiox, [id].msk = (1 《《 pinx), [id].init_value = init#define GPIO_PIN_OUT(id, gpiox, pinx, init) _GPIO_PIN_OUT(id, gpiox, pinx, init)

#define _bsp_pin_output_set(gpiox, pin) (gpiox)-》BSRR = pin#define bsp_pin_output_set(gpiox, pin) _bsp_pin_output_set(gpiox, pin)

#define _bsp_pin_output_clr(gpiox, pin) (gpiox)-》BRR = pin#define bsp_pin_output_clr(gpiox, pin) _bsp_pin_output_clr(gpiox, pin)

typedef enum{ PIN_OUTPUT_LED_G, PIN_OUTPUT_LED_R, PIN_OUTPUT_LED_B, PIN_OUTPUT_MAX}bsp_pin_output_id_def;

static const bsp_output_pin_def bsp_output_pin [PIN_OUTPUT_MAX] = { GPIO_PIN_OUT(PIN_OUTPUT_LED_G, GPIOA, 0, 0), GPIO_PIN_OUT(PIN_OUTPUT_LED_R, GPIOF, 15, 0), GPIO_PIN_OUT(PIN_OUTPUT_LED_B, GPIOD, 10, 0),};

void bsp_pin_init_output(GPIOx_Def gpiox, uint32_t msk, uint32_t init){ uint32_t temp;

assert_param((msk & 0xffff0000) == 0 && gpiox != 0);

temp = ((uint32_t) gpiox - (uint32_t) GPIOA) / ( (uint32_t) GPIOB - (uint32_t) GPIOA);

/* enable the led clock */ RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA 《《 temp, ENABLE);

GPIO_InitTypeDef GPIO_InitStruct;

GPIO_InitStruct.GPIO_Mode = (GPIOMode_Def)GPIO_Mode_Out_PP; GPIO_InitStruct.GPIO_Pin = msk; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_2MHz;

GPIO_Init((GPIO_TypeDef*)gpiox, &GPIO_InitStruct);

if(init == 0) { bsp_pin_output_clr(gpiox, msk); } else { bsp_pin_output_set(gpiox, msk); }}

void bsp_output_init(){ bsp_output_pin_def *info;

info = (bsp_output_pin_def *)&bsp_output_pin; for(int i = 0; i 《 sizeof(bsp_output_pin)/sizeof(bsp_output_pin[0]); i++) { bsp_pin_init_output(info-》gpio, info-》msk, info-》init_value); info++; }}

void bsp_output(bsp_pin_output_id_def id, uint32_t value){ assert_param(id 《 PIN_OUTPUT_MAX);

if(value == 0) { bsp_pin_output_clr(bsp_output_pin[id].gpio, bsp_output_pin[id].msk); } else { bsp_pin_output_set(bsp_output_pin[id].gpio, bsp_output_pin[id].msk); }}

int main(void){ USRAT_Init(9600);//必须,进入调试模式后点击全速运行

bsp_output_init();

while(1) { bsp_output(PIN_OUTPUT_LED_G, 1); bsp_output(PIN_OUTPUT_LED_B, 0); bsp_output(PIN_OUTPUT_LED_R, 1); } }

这个框架有啥好处呢?

1、自动完成 GPIO 的时钟初始化工作,也就是说你只需要修改引脚即可,不必关心时钟配置,但对于特殊引脚(比如PB3),还是得另外配置才行。

2、应用和底层具体 IO 分离,这样一旦修改了 IO,应用代码不需要进行任何修改。

3、增加或删减 IO 变得很简单,增加 IO时,首先加入对应枚举,然后就可以添加对应的 IO 了。删除 IO时,只要屏蔽对应枚举值和引脚即可。

4、参数检查功能, IO 删除时,因为屏蔽了对应的枚举,所以编译时可以帮你发现问题,而增加 IO 时,它可以帮你在运行时检查该 IO是否进行配置了,可以防止因为失误导致的问题。

a82570d2-e10c-11eb-9e57-12bb97331649.png

5、更改库时可以很方便,只需要修改对应的宏即可,目前可以顺利在 GD32 和 STM32 库进行快速更换。

6、对于输入 IO 而言,可以方便的修改有效和无效状态,防止硬件修改有效电平。对于输出 IO 而言,可以设定初始 IO 电平状态。

7、代码简单高效,尽可能的复用代码,增加一个 IO 只需要很少的空间。

8、缺点就是,只对同种配置的 IO 可以这样用。

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 嵌入式系统
    +关注

    关注

    41

    文章

    3593

    浏览量

    129475

原文标题:简单实用IO输入输出框架

文章出处:【微信号:zhuyandz,微信公众号:FPGA之家】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    io口和串口的区别 单片机有多少io

    等。而串口,即串行通信接口(Serial Communication Interface),是种数据通信方式,通过条数据线按照顺序传送数据。IO口和串口在功能和用途上存在显著区别: 通信方式 :
    的头像 发表于 10-06 10:06 1440次阅读

    误差放大器的输入输出关系

    误差放大器(Error Amplifier)在电子测量和控制系统中扮演着至关重要的角色,其输入输出关系对于理解和设计这些系统至关重要。以下是对误差放大器输入输出关系的详细解析,包括其工作原理、输入输出特性、影响因素以及实际应用等
    的头像 发表于 09-11 15:32 1202次阅读

    隔离变压器输入输出可以随便接吗

    隔离变压器的输入输出接线并非可以随便接,而是需要遵循定的原则、步骤和注意事项,以确保其正常、安全、高效地运行。 、接线原则 电压匹配 :在接线之前,必须确保输入电源和
    的头像 发表于 09-06 11:07 858次阅读

    互阻放大电路对输入输出电阻的要求

    检测的领域中非常重要。互阻放大电路的设计和性能受到输入输出电阻的影响,因此对这些电阻的要求非常严格。 输入电阻 高输入电阻 :互阻放大电路需
    的头像 发表于 09-05 14:21 804次阅读

    寄存器的输入输出方式

    寄存器的输入输出方式是数字电路设计中至关重要的部分,它决定了数据如何在寄存器中进出以及处理的效率。下面将详细探讨寄存器的几种主要输入输出方式,包括并行输入输出、串行输入输出以及双向
    的头像 发表于 09-05 14:09 1304次阅读

    PLC运动控制中的输入输出设备

    在PLC(可编程逻辑控制器)运动控制系统中,输入输出设备扮演着至关重要的角色。这些设备不仅负责将外部信号传递给PLC,还负责将PLC的处理结果输出到外部执行机构,从而实现对机械设备运动的精确控制。以下是对PLC运动控制中输入输出
    的头像 发表于 09-03 10:52 840次阅读

    用OPA656搭了简单的跟随器,但是输入输出之间有漂移,为什么?

    我用OPA656搭了简单的跟随器,但是输入输出之间有漂移,差不多250mV
    发表于 09-03 07:26

    引入负反馈对输入输出电阻的影响

    度等。 1. 引言 在电子电路设计中,电阻是基本的参数,它影响着电路的电流和电压。输入电阻和输出电阻分别描述了电路对输入信号和
    的头像 发表于 07-30 09:43 1707次阅读

    PLC输入输出信号异常的原因分析

    在工业自动化领域中,PLC(可编程逻辑控制器)作为控制系统的核心部件,其性能的稳定性和可靠性对于整个生产线的正常运行具有至关重要的影响。然而,在实际应用中,PLC的输入输出信号异常问题时有发生,这不
    的头像 发表于 06-12 11:25 2237次阅读

    为什么可以将GPIO引脚同时配置为输入输出模式呢?

    系列高低电平。 那么对于GPIO6而言,在发送起始信号的时候,是输出模式,接收信号的时候是输入模式,也就是说需要将引脚同时配置为
    发表于 06-05 07:37

    自定义位宽输入输出截位模块的灵活配置方案

    可配置任意输入输出位宽截位模块
    的头像 发表于 04-25 11:36 402次阅读

    PLC的输入输出接口是否需要进行隔离保护?

    PLC(可编程逻辑控制器)的输入输出接口是否需要进行隔离保护,取决于具体的应用场景和需求。
    的头像 发表于 02-23 09:09 2073次阅读

    输入输出复用电路的定义 复用器的输出由什么控制

    输入输出复用电路是种特殊的电路设计,它允许单个物理端口在不同的时间或条件下执行输入输出操作。这种复用电路在多个领域都有应用,特别是在需要高效利用硬件资源和提高系统性能的场景中。
    的头像 发表于 02-18 14:11 909次阅读

    锁相环的输入输出相位致吗?

    锁相环是保证相位致,还是相位差致?锁相环的输入输出相位致吗? 锁相环(PLL)是种回路控制系统,用于保持
    的头像 发表于 01-31 15:45 1181次阅读

    输入输出电压差与效率的关系

    在开关稳压电源中,输入电压的范围是预知的,输出电压也是知道的,但是输入输出的电压差和转换效率的关系很多人 不清楚,有经验的工程师就会根据公式去推导出来输入输出电压差越小,转换效率越高。
    发表于 01-05 15:12 883次阅读
    <b class='flag-5'>输入输出</b>电压差与效率的关系