1. LED 动态扫描流程
- 对于动态扫描的原理,网上的描述已经足够清晰,这里不再做过多的阐述,本章主要侧重与代码的实现逻辑
- 首先我们需要一个行为函数对使用到的 GPIO 进行初始化,并声明一个表示 LED、或数码管的各个位的数据结构,再实现一个更新位段的行为函数(简称更新函数)。
- 更新函数需要满足以下几个特点
- 要根据 LED、或数码管的各个位的数据来更新状态
- 每次只更新一个位段的状态,下次调用时更新下一个位段的状态
- 最后我们仅需每隔一段合适的时间间隔后执行更新函数(如定时器中断服务函数中执行),然后在需要更改 LED、或数码管的显示状态时,仅需修改存放 LED、或数码管的状态的数据结构中的数据即可
- 我们仅需实现以下内容
- GPIO 初始化
- 表各位段的数据结构
- 更新位段的行为函数
2. 代码实现
- 这里以寄存器开发的单片机的数码管的动态扫描为例
- 具体的实现以下代码有注释解析
#define PLACE_NUM 4
#define PLACE_0 PB1
#define PLACE_1 PB2
#define PLACE_2 PB3
#define PLACE_3 PB4
#define PARAGRAPH_A PC1
#define PARAGRAPH_B PC0
#define PARAGRAPH_C PB0
#define PARAGRAPH_D PC6
#define PARAGRAPH_E PC7
#define PARAGRAPH_F PC4
#define PARAGRAPH_G PA3
#define PARAGRAPH_DP PA2
volatile unsigned char nixie_tubu_status[PLACE_NUM] = {0xff, 0xff, 0x00, 0x0a, };
const unsigned char NIXIE_TUBE_NUMBER[10] = {0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F, };
void nixie_tubu_init(void)
{
PBS &= 0xe0;
PBT &= 0xe0;
PBLC0 &= 0xe0;
PBLC1 &= 0xe0;
PB |= 0x1e;
PB &= 0xfe;
PAS &= 0xf3;
PAT &= 0xf3;
PALC &= 0xf3;
PA &= 0xf3;
PCS &= 0x2c;
PCT &= 0x2c;
PCLC &= 0x2c;
PC &= 0x2c;
}
void nixie_tubu_update(void)
{
static unsigned char count = 0;
PB |= 0x1e;
PARAGRAPH_A = nixie_tubu_status[count] >> 0;
PARAGRAPH_B = nixie_tubu_status[count] >> 1;
PARAGRAPH_C = nixie_tubu_status[count] >> 2;
PARAGRAPH_D = nixie_tubu_status[count] >> 3;
PARAGRAPH_E = nixie_tubu_status[count] >> 4;
PARAGRAPH_F = nixie_tubu_status[count] >> 5;
PARAGRAPH_G = nixie_tubu_status[count] >> 6;
PARAGRAPH_DP = nixie_tubu_status[count] >> 7;
switch (count) {
case 0:
PLACE_0 = 0;
break;
case 1:
PLACE_1 = 0;
break;
case 2:
PLACE_2 = 0;
break;
case 3:
PLACE_3 = 0;
break;
}
count = ++count % PLACE_NUM;
}