单片机中的通用LED驱动

前言

项目中需要用到很多的LED灯,存在不同的闪烁方式,比如单闪,双闪,快闪,慢闪等等,我需要一个有如下特性的LED驱动

  • 方便的增加不同闪烁模式
  • 可以切换闪烁模式
  • 增加LED数目不会有太多的改动
  • 方便移植,要有良好的硬件对接接口

好,那就开整吧。
PS:本文中的程序源码只做演示,可运行的代码文末有链接

数据结构分析

首先考虑一颗LED的相关数据结构。
显然构建LED结构体应该有on,off接口,如下

typedef struct{void (*init)(void);	//初始化EDvoid (*on)(void);	//打开LEDvoid (*off)(void);	//关闭LED
}led_t;

LED闪烁是亮灭的交替,我们可以关注其中的两个参数,

  • LED亮起时长,标记为ontime
  • LED闪烁周期,标记为cycle
    在这里插入图片描述
    将这两个参数抽象为led_mode_t结构体
typedef struct {uint16_t cycle;		//LED闪烁周期uint16_t ontime;	//LED亮起的时长
}led_mode_t;

一颗LED可能会有很多闪烁模式,不同LED闪烁模式数量不同,所以当我们将led_mode_t集成到led_t的时候,应该采用指针形式,在运行的时候申请该结构的内存。由此,丰富led_t结构体如下

typedef struct{void (*init)(void);	//初始化EDvoid (*on)(void);	//打开LEDvoid (*off)(void);	//关闭LEDled_mode_t *mode;		//LED闪烁模式具体的数据uint8_t mode_count;		//LED闪烁模式个数uint8_t mode_current;	//当前闪烁模式编码
}led_t;

OK,LED结构体初步搭建完毕,接下来假设我们有4颗LED,每一颗闪烁的时间参数如下

  • 快闪:200ms亮起,200ms熄灭,周期400ms
  • 慢闪:500ms亮起,500ms熄灭,周期1000ms
  • 单闪:100ms亮起,900ms熄灭,周期1000ms
  • 双闪:30ms亮起, 70ms熄灭,30ms亮起, 870ms熄灭,周期1000ms

前三个都好说,最后一个需要简单分析一下。双闪可以看作两种闪烁模式的切换。第一种30ms亮起, 70ms熄灭,周期100ms。第二种30ms亮起, 870ms熄灭,周期900ms。
在这里插入图片描述

程序框架搭建

所以,初始化该情况下的代码如下

typedef struct {uint16_t cycle;		//LED闪烁周期uint16_t ontime;	//LED亮起的时长
}led_mode_t;typedef struct{void (*init)(void);	//初始化EDvoid (*on)(void);	//打开LEDvoid (*off)(void);	//关闭LEDled_mode_t *mode;		//LED闪烁模式具体的数据uint8_t mode_count;		//LED闪烁模式个数uint8_t mode_current;	//当前闪烁模式编码
}led_t;led_t led_array[4];void led0_init(void){}
void led1_init(void){}
void led2_init(void){}
void led3_init(void){}
void led0_on(void){}
void led1_on(void){}
void led2_on(void){}
void led3_on(void){}
void led0_off(void){}
void led1_off(void){}
void led2_off(void){}
void led3_off(void){}void bsp_led_init(void)
{//初始化函数指针led_array[0].on = led0_on;led_array[0].off = led0_off;led_array[0].init = led0_init;led_array[0].mode_count = 1;led_array[1].on = led1_on;led_array[1].off = led1_off;led_array[1].init = led1_init;led_array[1].mode_count = 1;	led_array[2].on = led2_on;led_array[2].off = led2_off;led_array[2].init = led2_init;led_array[2].mode_count = 1;led_array[3].on = led3_on;led_array[3].off = led3_off;led_array[3].init = led3_init;led_array[3].mode_count = 2;for(uint8_t i = 0; i < sizeof(led_array)/led_array[0]; i++){led_array[i].mode = malloc(sizeof(led_mode_t) * led_array[i].mode_count);memset(led_array[i].mode, 0, sizeof(led_mode_t) * led_array[i].mode_count);}//初始化mode时间参数led_array[0].mode[0].ontime = 200;	led_array[0].mode[0].cycle 	= 400;led_array[1].mode[0].ontime = 500;led_array[1].mode[0].cycle 	= 1000;led_array[2].mode[0].ontime = 100;led_array[2].mode[0].cycle 	= 1000;led_array[3].mode[0].ontime = 30;led_array[3].mode[0].cycle 	= 100;led_array[3].mode[1].ontime = 30;led_array[3].mode[1].cycle 	= 900;
}

代码很长,主要长度占用在以下三部分

  • 每一个led都有init,on,off函数
  • 初始化函数指针
  • 初始化mode时间参数

第一部分暂时不做优化,这样会方便我按照顺序说下去吧
第二第三部分,显然程序中有很多重复的代码,我们可以使用可变参数宏来优化

#define INIT_PTR(__index, __onptr, __offptr, __initptr, __modecount)	\led_array[__index].on = __onptr;									\led_array[__index].off = __offptr;									\led_array[__index].init = __initptr;								\led_array[__index].mode_count = __modecount;						#define INIT_MODE(__index, __mode, __ontime, __cycle)	\led_array[__index].mode[__mode].ontime = __ontime;	\led_array[__index].mode[__mode].cycle 	= __cycle;	

使用这两个宏之后,第二第三部分的代码被优化为如下,看起来少了好多

void bsp_led_init(void)
{INIT_PTR(0, led0_on, led0_off, led0_init, 1);INIT_PTR(1, led1_on, led1_off, led1_init, 1);INIT_PTR(2, led2_on, led2_off, led2_init, 1);INIT_PTR(3, led3_on, led3_off, led3_init, 2);for(uint8_t i = 0; i < sizeof(led_array)/led_array[0]; i++){led_array[i].mode = malloc(sizeof(led_mode_t) * led_array[i].mode_count);memset(led_array[i].mode, 0, sizeof(led_mode_t) * led_array[i].mode_count);}INIT_MODE(0, 0, 200, 400);INIT_MODE(1, 0, 500, 1000);INIT_MODE(2, 0, 100, 100);INIT_MODE(3, 0, 30, 100);INIT_MODE(3, 1, 30, 900);
}

OK,我们已经设定了各个LED的闪烁模式,对接了初始化,亮起,熄灭的函数,是时候让他跑起来了。
假设我们有一个bsp_led_tick函数,该函数每1ms调用一次。我们在该函数中定义一个递增的变量tick,比较其它和LED灯ontime的大小。tick比ontime小则LED亮起,比ontime大则LED熄灭。为了循环往复的工作,我们采用的比较值应该是tick对周期的求余而不是tick本身。示例如下

static void bsp_led_tick(void){
#define i_CYCLE_LENGTH  (led_array[i].mode[led_array[i].mode_current].cycle)
#define i_ON_TIME (led_array[i].mode[led_array[i].mode_current].ontime)static uint64_t tick;for(uint8_t i = 0; i <  i < sizeof(led_array)/led_array[0]; i++){if(tick % i_CYCLE_LENGTH < i_ON_TIME)led_array[i].on();elseled_array[i].off();}tick++;
}

我们使用了两个宏i_CYCLE_LENGTH ,i_ON_TIME 来减少代码长度,增加可读性。
可以看出,切换闪烁模式的话直接修改led_array[i].mode_current即可。
到此,我们的驱动框架就很清晰了。
接下来我会指出该框架的问题,并逐一修改。

问题修复

问题1:LED无法关闭
上面代码中LED一直会闪烁无法关闭
解决办法
LED关闭可以认为是一种模式。其中ontime为0,周期为任意值(0除外)
所以,我们可以占用mode[0],将其所有LED的mode[0]初始化为ontime=0, cycle=1,用户设置mode_current为0的时候调用该模式,关闭LED。注意,此时用户设置的闪烁模式需要从mode[1]开始

那么,同样的,如果需要LED常亮呢?我认为LED熄灭是大部分项目中必要的,而常亮却不一定,所以在代码中不做常亮模式的预先设置,如果用户需要的话可以另外设置一个mode,其中的ontime大于cycletime(cycletime不可以为0)即可

问题2:on off重复调用
满足if(tick % i_cycle < i_ontime )条件的时候,程序会重复调用led_array[i].on();即使此时LED已经打开了。
解决办法
在led_t中增加state成员,标记LED状态,已经打开的时候不要重复调用on,已经关闭的不要重复调用off

	···for(uint8_t i = 0; i < sizeof(led_array)/led_array[0]; i++){//增加关闭模式,此后mode[0]就被占用了,用户定义的闪烁模式要从mode[1]开始led_array[i].mode = malloc(sizeof(led_mode_t) * (led_array[i].mode_count)+1);memset(led_array[i].mode, 0, sizeof(led_mode_t) *  (led_array[i].mode_count)+1t);}INIT_MODE(0, 1, 200, 400);INIT_MODE(1, 1, 500, 1000);INIT_MODE(2, 1, 100, 100);INIT_MODE(3, 1, 30, 100);INIT_MODE(3, 2, 30, 900);···
static void bsp_led_tick(void){······//cycletime == 0, LED不再闪烁if(i_CYCLE_LENGTH == 0){if( (lled_array[i]..state == 1)){led_array[i].state = 0;led_array[i].off();}continue;}else{if(tick % i_CYCLE_LENGTH < i_ON_TIME){if( (led_array[i].state == 0)){//如果之前是关闭的,那么开启led_array[i].state = 1;led_array[i].on();}}else{if( led_array.led[i].state == 1){//如果之前是开启的,那么关闭led_array[i].state = 0;led_array[i].off();}}}tick++;
}

问题3:如何实现双闪
我们之前说过,双闪是两种闪烁模式的交替闪烁,那么如何实现交替切换模式呢?
很简单,我们只要再bsp_led_tick中判断tick是否增加到mode[].cycle即可。

//周期回调函数
if(i_CYCLE_LENGTH-1 == (tick % i_CYCLE_LENGTH))
{led_array[i].cycle_func();
}

cycle_func是led_t中增加的新成员,作为周期结束的回调函数。使用前需要初始化指向led3_cyclefun,在该函数中做模式切换

static void led3_cyclefun()
{if(led_array[3].mode_current== 1)led_array[3].mode_current = 2;else if(led_array[3].mode_current== 2)led_array[3].mode_current = 1;
}

有了周期回调函数,再复杂的LED显示效果我们都可以做出来,只需要再周期结束修改显示模式即可。
但是理想很丰满,现实很骨感,这个地方还有一小片乌云等待解决。
在这里插入图片描述
我没有按照led3设置mode时间值,按照图上的会直观一点。
可以看到,在第二个闪烁模式tick=2时间段,此时ontime=1,tick%cycletime=2%5=2,那么第二模式根本不会亮起。此时的流程实际上变成了如下图
在这里插入图片描述
为了解决这个问题,我们需要明确,模式切换之后tick起始点是不同的,不可以笼统的写作tick%cycletime,模式切换之后应该有自己的tick起始点tick_start,按照(tick-tick_start)%cycletime求得余数
问题4:同步
为了解决问题三,我们引入了起点tick_start机制,这回带来新的问题。
每一个LED都有自己的起点,在模式切换的时候更新为当前tick。模式切换是使用该驱动的人决定的,这会造成即使相同周期的LED灯闪烁相位的差异,看起来在乱闪。
解决办法
在led_t中引入sync成员,ticks_start只有配置了sync为0的时候才会更新为当前tick,否则设置为0。
这将会明确一个事实,sync配置为1的LED无法实现循环的闪烁模式切换。即类似于双闪这种循环的切换模式是不被允许的,单次的切换可以。
双闪如果要做同步该怎么办?应该可以在周期回调函数中做一些工作,留给大家思考。

优化

这部分主要优化的是代码体积,假设我有十个灯,每一个都需要on,off,cyclefunc,这也太恐怖了。所以,我们应该再led_t结构体之上再做一个结构体led_array_t。内容如下

typedef struct{led_t led[LED_COUNT];void (*on)(void *parameter);	//打开LEDvoid (*off)(void *parameter);	//关闭LEDvoid (* cycle_func)(void *parameter);	//cycle结束的回调函数
}led_array_t;

这里设置了parameter参数,调用函数的时候将led_t传入以标识身份,在函数内部判断我是哪一个LED,进行相应的操作
举例如下

static void bsp_led_on(void *parameter)
{led_t *p = (led_t *)parameter;switch(p-led_array.led){case 0:DRV_DIO_ChannelOutSet(DRV_DIO_ID_PIO_9);break;case 1:DRV_DIO_ChannelOutSet(DRV_DIO_ID_PIO_25);break;case 2:DRV_DIO_ChannelOutSet(DRV_DIO_ID_PIO_14);break;case 3:DRV_DIO_ChannelOutSet(DRV_DIO_ID_PIO_15);break;default:break;};
}

OK,内容到此结束了,这样看来写好一个LED并不容易。
下面提供了一个基于RTOS的源码,如果要修改为裸机的并不需要耗费很大功夫,相信你可以做到
程序源码在此:https://gitee.com/nwwhhh/led_flash_driver

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/11139.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【C++杂货铺】模板(文末有彩蛋哟)

文章目录 一、泛型编程二、函数模板2.1 函数模板的原理2.2 函数模板的实例化2.3 模板参数的匹配原则 三、类模板四、非类型模板参数五、模板的特化5.1 函数模板特化5.2 类模板特化 六、模板分离编译七、模板总结好书推荐&#x1f381;彩蛋 一、泛型编程 &#x1f4d6;实现一个…

Spring Boot单元测试入门指南

Spring Boot单元测试入门指南 JUnit是一个成熟和广泛应用的Java单元测试框架&#xff0c;它提供了丰富的功能和灵活的扩展机制&#xff0c;可以帮助开发人员编写高质量的单元测试。通过JUnit&#xff0c;开发人员可以更加自信地进行重构、维护和改进代码&#xff0c;同时提高代…

volley 学习笔记1--发送请求

一、概览 Volley 具有以下优势&#xff1a; 自动网络请求调度。 多个并发网络连接。 透明磁盘和具有标准 HTTP 缓存一致性的内存响应缓存。 支持请求优先级。 取消请求 API。您可以取消单个请求&#xff0c;也可以设置要取消的请求的时间段或范围。 可轻松自定义&#xff…

linux下i2c调试神器i2c-tools安装及使用

i2c-tools介绍 在嵌入式linux开发中&#xff0c;有时候需要确认i2c硬件是否正常连接&#xff0c;设备是否正常工作&#xff0c;设备的地址是多少等等&#xff0c;这里我们就需要使用一个用于测试I2C总线的工具——i2c-tools。 i2c-tools是一个专门调试i2c的开源工具&#xff…

Kafka 入门到起飞系列 - 消费者组管理、位移管理

消费者组 - Consumer Group 上文我们已经讲过消费者组了&#xff0c;我们知道消费组的存在可以保证一个主题下一个分区的消息只会被组内一个消费者消费&#xff0c;从而避免了消息的重复消费 什么是消费组 - Consumer Group&#xff1f; 消费者组是Kafka 提供的可扩展且具有容…

事后多重比较方法

一、案例介绍 由单因素方差分析案例中&#xff0c;为研究郁金对低张性缺氧小鼠存活时间的影响&#xff0c;将36只小鼠随机生成A、B以及 C 三组&#xff0c;每组12个&#xff0c;雌雄各半&#xff0c;分别以10g/kg、20g/kg、40g/kg三种不同剂量的郁金灌胃&#xff0c;各组小鼠均…

从原理到实践,分析 Redisson 分布式锁的实现方案(二)

上篇讲解了如何用 Redis 实现分布式锁的方案&#xff0c;它提供了简单的原语来实现基于Redis的分布式锁。然而&#xff0c;Redis作为分布式锁的实现方式也存在一些缺点。本文将引入Redisson来实现分布式锁。 一、Redisson是什么 Redisson是一个基于Redis的分布式Java框架。它提…

信息安全:网络安全体系 与 网络安全模型.

信息安全&#xff1a;网络安全体系 与 网络安全模型. 网络安全保障是一项复杂的系统工程&#xff0c;是安全策略、多种技术、管理方法和人员安全素质的综合。一般而言&#xff0c;网络安全体系是网络安全保障系统的最高层概念抽象&#xff0c;是由各种网络安全单元按照一定的规…

抖音seo短视频矩阵系统源代码开发技术分享

抖音SEO短视频矩阵系统是一种通过优化技术&#xff0c;提高在抖音平台上视频的排名和曝光率的系统。以下是开发该系统的技术分享&#xff1a; 熟悉抖音平台的算法 抖音平台的算法是通过分析用户的兴趣爱好和行为习惯&#xff0c;对视频进行排序和推荐。因此&#xff0c;开发人员…

Visitor设计模式访问元素方法的问题

Visitor设计模式访问元素方法的问题 GPT给出的答案寻找灵感前置声明Element层次的实例Visitor interface的声明Element interface的声明Element实际类的声明及实现实现一个Visitor客户端代码 实战测试结果 针对C来说&#xff0c;若要实现Visitor设计模式&#xff0c;则会面临循…

SAP安装笔记

1、准备安装介质&#xff0c;SWPM10SP25&#xff0c;51050829_NW750_JavaExport、SAP_HANA_CLIENT、kernel放到/sapcd/NetWeaver目录下 ​​​​​​​ 进入SWPM10SP25执行./sapinst安装 2、待出现 “Open your browser and paste the following URL address to access the G…

上门家政系统开发|上门预约家政小程序定制系统

随着人们生活水平的提高&#xff0c;对于家政服务的需求也越来越高。上门家政小程序的开发为家政服务商家提供了一个全新的经营和服务渠道。本文将介绍上门家政小程序适合的商家以及其优势。   1. 家政公司   家政公司是最直接受益于上门家政小程序开发的商家。通过开发家政…

企业博客资讯如何高效运营起来?

运营一个高效的企业博客资讯需要综合考虑多个因素&#xff0c;包括内容策划、发布频率、优化推广、互动反馈等。下面将从这些方面介绍如何高效运营企业博客资讯。 如何高效运营企业博客资讯 内容策划 首先&#xff0c;需要制定一个明确的内容策略。确定博客的定位和目标受众…

【C语言】指针进阶(二)

&#x1f490; &#x1f338; &#x1f337; &#x1f340; &#x1f339; &#x1f33b; &#x1f33a; &#x1f341; &#x1f343; &#x1f342; &#x1f33f; &#x1f344;&#x1f35d; &#x1f35b; &#x1f364; &#x1f4c3;个人主页 &#xff1a;阿然成长日记 …

【UE5 多人联机教程】04-加入游戏

效果 步骤 1. 新建一个控件蓝图&#xff0c;父类为“USC_Button_Standard” 控件蓝图命名为“UMG_Item_Room”&#xff0c;用于表示每一个搜索到的房间的界面 打开“UMG_Item_Room”&#xff0c;在图表中新建一个变量&#xff0c;命名为“Session” 变量类型为“蓝图会话结果…

MB5B在HDB上的性能调优

背景 MB5B是用于查询物料的收发以及现有库存。日常业务查询,通常会按照月份查看某片地区物料的库存以及收发状态。 调优思路 按照客户日常操作的习惯,得到日常操作的数据范围,选出数据量最为突出最有代表性的地区和物料;利用SE30分别运行不同数量级的数据,比如20个门店、…

利用sklearn 实现线性回归、非线性回归

代码&#xff1a; import pandas as pd import numpy as np import matplotlib import random from matplotlib import pyplot as plt from sklearn.preprocessing import PolynomialFeatures from sklearn.linear_model import LinearRegression# 创建虚拟数据 x np.array(r…

DAY3,Qt(完成闹钟的实现,定时器事件处理函数的使用)

1.完成闹钟的实现&#xff0c;到点播报文本框的内容&#xff1b; ---alarm.h---头文件 #ifndef ALARM_H #define ALARM_H#include <QWidget> #include <QTimerEvent> //定时器处理函数类 #include <QTime> //时间类 #include <QPushButton> //按钮…

蓝牙技术|智能照明市场蓬勃发展,蓝牙技术助力市场发展

照明控制系统在商业和工业领域的应用广泛。例如&#xff0c;智能办公楼、商场、工厂等场所&#xff0c;可以通过照明控制系统实现节能和舒适性的提升。预计将从2023年的74亿美元增长到2032年的108亿美元&#xff0c;复合年增长率(CAGR)为4.3%。 随着LED照明技术在市场上的逐渐普…

QT DAY3

1.思维导图 2.完成闹钟的实现 头文件 #include <QTextToSpeech> #include <QTextEdit> QT_BEGIN_NAMESPACE namespace Ui { class Widget; } QT_END_NAMESPACEclass Widget : public QWidget {Q_OBJECTQLineEdit *edit1new QLineEdit;// QTextEdit *edit2new QTe…