文章记录一段旅程–使用数码管显示它能显示的任何内容。
前言
数码管的特点是比较亮,隔老远就能看得清清楚楚,虽然大部分情况它被用来显示数字,但是如果合理拼凑也能显示出各种各样有意思的东西,比如进度条。当然我希望它能显示任何它能力范围内的东西。

一、硬件特性
其实这个例子我并不是很想研究硬件,但是呢来都来了,就简单看一看吧。
首先是74HC245芯片,这是一个三态8线双向收发器,也就是数据既可以从A端流向B端也可以反着来,不过它在这个电路的主要是用来驱动数码管的,虽然单片机IO不见得不能直驱数码管,但是需要精心地去设计驱动方式,比如要避免同时点亮所有数码管等等,加了这个东西可以提高系统的稳定裕度,省心。

然后是74HC138译码器,这是一个三八译码器,8421码转十进制选择线,下图中A管脚的权值为1,B的权值为2,C的权值为4。研究一种特例,CBA=101,换成10进制是5,那么芯片会让“Y5非”标识的那个端口变成0,而其他端口变成1,恰好这个例子用的数码管是共阴的,所以对应的数码管会被选中。
这个芯片有个好处是正常工况下它永远只有一个输出管脚为0或者所有输出管脚为1,并且由于硬件配置问题,使得在任意时刻有且仅有一位数码管被选中,这可以从设计的角度限制系统功耗。

总之,要点亮某个数码管的某一段,只要让数码管的共阴位选线为0,阳极段选线为1就能点亮了。本质上这就是在点亮LED灯,可以参考下面软件驱动器小节的第二张图,手绘的那张。
二、软件驱动器
1.软件与数码管的接口
思路永远从简单的开始。先操作一位数字,也就是下图中红框框起来的基本单元,然后再想办法操作整个数码管。


上面画的是关于数码管的其中一位的详细信息。要操作它,可以先确定a到dp的值,然后给共阴端口送入逻辑0,当然也可以反着来。得益于研究led点阵驱动器的经验,注意到74HC138与74HC245的传输速度不见得是一样的,如果速度相差过大,那么段数据和位数据先送谁就要关注一下了。转化成代码如下所示。
typedef enum{
TUBE1 = 0,TUBE2,TUBE3,TUBE4,
TUBE5, TUBE6,TUBE7,TUBE8,
}tubeIndex_t;
void operateOneTube(tubeIndex_t tubeIndex, uint8_t content){
switch(tubeIndex){
case TUBE1:
#error "step1:操作数码管对应位的共阴端为0"
break;
}
#error "step2:为数码管该位的各阳极端口赋值"
}
操作一个数码管的底层代码,尽量不要做任何多余的动作,只需要指定让哪一个数码管显示什么样的内容即可,尽可能保证这个过程的纯粹性以确保最大限度的可复用性。
至于要显示什么数字、字母、图形,或者要不要闪烁是上层代码的事情,这些都可以通过逻辑或算法获得数据流,然后让数据流在不同的时间流入operateOneTube函数最终显示出各种各样的内容。
接下来是要操作整排数码管。动态扫描很好用,直接用这个方法。在点亮一位数码管的基础上进行动态扫描以点亮多个数码管,需要增加延时稳定显示以及消隐操作,这几乎是每一个单片机操作数码管教程都会提的事情。
void operateOneTube(tubeIndex_t tubeIndex, uint8_t content){
#error "step1:操作数码管对应位的共阴端为0"
#error "step2:为数码管该位的各阳极端口赋值"
#error "step3(延时):延时以稳定显示,增加数码管亮度"
#error "step4(消隐):清除单片机用于驱动数码管阳极的端口上的值,防止下一位数码管显示当前内容"
}
操作全部的8位数码管。建立一个缓冲区unsigned char tubeBuf[8],缓冲区共8个字节,每一个字节用于存放指定位的数码管数据。数码管要刷新显示的时候就来缓存里面取数据,其他应用要调用数码管时就向缓存里面写数据,利用缓存,数码管所有的行为都被抽象为随时间变化的数据流,这其中包括完全关闭数码管(全0数据流)。
//数码管数据缓冲,有多少个字符就要有多少缓冲区
unsigned char tubeBuf[8] = {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};
void displayAnythingByAllTube(unsigned char disBuff[]){
operateOneTube(TUBE1,disBuff[0]);
operateOneTube(TUBE2,disBuff[1]);
operateOneTube(TUBE3,disBuff[2]);
operateOneTube(TUBE4,disBuff[3]);
operateOneTube(TUBE5,disBuff[4]);
operateOneTube(TUBE6,disBuff[5]);
operateOneTube(TUBE7,disBuff[6]);
operateOneTube(TUBE8,disBuff[7]);
}
突然想起一件事情,那就是上面说的缓存的并发访问问题。假如将数码管刷新函数displayAnythingByAllTube放在一个定时器中断里面以固定数码管刷新率,然后主程序的任意位置可以操作这个缓存,如果主程序操作这个缓存到一半时发生了数码管刷新中断呢?显示的数据就有可能出现乱码,这个事件并不是每时每刻一定会发生,偶发性bug往往是把人搞得焦头烂额的元凶。所以当前想到这个事情就写在提醒自己一下,具体要怎么解决我暂时也不在乎。或许可以使用二级缓存以及互斥锁机制来处理这个事情,然而事情开始变得复杂起来了。
2.多路数据块复用器
使用这个东西主要是想让数码管的某一位闪烁起来,比如在设置参数的时候,这样搞可以指示目前在设置数字的哪一位。

下面的机构运行起来有一个前提条件,数码管刷新的速度和复用器运行的速度必须要远大于某个数字闪烁的速度,当然由于单片机的运行速度很快,而且高于30帧的刷新率人眼是瞧不出来的,这个条件可以自然得到保证。

在前几篇文章(蜂鸣器,led)里面讨论过的多路逻辑复用器的基础上。将复用器的输入输出参数修改为数据块,当然在c里面写程序时指定的是数据块起始地址指针。
typedef struct dataMUX //数据复用器
{
unsigned char selfActionSignal;
void* currentOutput;
enum
{
OUT_LINK_TO_CH1 = 0,
OUT_LINK_TO_CH2,
}linkCH;
void* inputChannel1;
void* inputChannel2;
void (*step)(struct dataMUX* multiplexer);
void (*processSignal)(unsigned char signal, unsigned char value, struct dataMUX* multiplexer);
}dataMUX_t;
然后是主循环和中断代码。由于代码比较冗长,所以只构造伪代码,其中最主要的还是协调各功能模块的运行时间,伪代码足够了。关于上面的多路复用器的运行机制与部分代码,请移步文章《【电子电路基础实验】无源蜂鸣器》。
int main(void){
#error "初始化定时器"
#error "启动定时器"
while(1){
#error "step1: 运行数据复用器并装载数码管显示缓存tubeBuf最长时间 = 920us"
#error "step2: 刷新数码管时间固定 = 15.6ms"
}
}
void T0_interrupt(void) interrupt 1 {
#error "step1: 重载T0定时器初值以及其他辅助代码段最长时间 = 89us"
#error "step2: 运行用于数字闪烁的虚拟定时器最长时间 = 75us"
}
闹了半天刷新数码管花的时间最多。由于前面讨论过的缓存并发访问的问题,我觉得最直接的方法是把装载tubeBuf的代码与刷新数码管的代码捆在一起,要么都放在中断里面,要么都放在主循环里面,在这个简单的例子里面当然放在哪里都一样,但是工程变得复杂的时候中断里面就不太合适,刷新一次数码管花的时间太长,如果在建的还是个高频采样系统,那这个事情就麻烦了。
就目前而言,针对上面伪代码的模块分配方式,我觉得下面的时间搭配是一种可行的办法,50ms一次定时器中断可以保证主循环中的数码管数据装载与刷新过程至少能不间断地执行两轮,而定时器中断服务函数至多消耗164us,即使打断数码管刷新函数,实测是看不出来数码管频闪的,并且50ms定时器中断仍然能保证能为数码管闪烁提供足够的时基。

另外,在采样系统中,数码管的刷新时间就要引起足够的重视,比如使用1kHz的采样率对一个信号持续采样32点用于信号处理,采样间隔要求稳定,显然刷新数码管和采样这两件事情需要精心地去协调,亦或者牺牲成本增加驱动芯片之类的东西来兼得。
3. 回到开头的进度条
得益于displayAnythingByAllTube函数接收显示内容的任意性,我们可以随意构造、拼凑数码管图形。在构造进度条时,先构造多个静态的显示数据,比如进度为0%时是什么样子,进度为25%时是什么样子,然后让这些数据汇入时间流就构造完成了完整的进度条。
//进度条段选表-高电平使能
const unsigned char code progressBarTubeData[9][8] = {
{0x39,0x09,0x09,0x09,0x09,0x09,0x09,0x0F}, //0%
{0x79,0x09,0x09,0x09,0x09,0x09,0x09,0x0F}, //12.5%
{0x79,0x49,0x09,0x09,0x09,0x09,0x09,0x0F}, //25%
{0x79,0x49,0x49,0x09,0x09,0x09,0x09,0x0F}, //37.5%
{0x79,0x49,0x49,0x49,0x09,0x09,0x09,0x0F}, //50%
{0x79,0x49,0x49,0x49,0x49,0x09,0x09,0x0F}, //62.5%
{0x79,0x49,0x49,0x49,0x49,0x49,0x09,0x0F}, //75%
{0x79,0x49,0x49,0x49,0x49,0x49,0x49,0x0F}, //87.5%
{0x79,0x49,0x49,0x49,0x49,0x49,0x49,0x4F}, //100%
};
受限于数码管的位数,本例的进度条最小刻度为12.5%。
本文介绍了数码管的基础知识,包括硬件特性如74HC245和74HC138芯片的作用,以及如何通过软件驱动器实现数码管的显示控制。数码管的显示通过选择合适的共阴或共阳端口和段选线实现,动态扫描和多路数据块复用器则用于提高显示效果和指示特定状态。此外,文章还探讨了在单片机系统中处理数码管缓存并发访问的问题和刷新频率对系统性能的影响。
1566

被折叠的 条评论
为什么被折叠?



