1. 红外通信波形观测与NEC协议物理层解析

红外遥控在嵌入式系统中仍是最广泛使用的低成本人机交互方式之一。尽管其数据速率低、距离短、易受干扰,但因其电路简单、功耗极低、协议成熟,在家电控制、工业遥控、教学实验等领域持续占据不可替代的地位。本节不讨论软件解码逻辑或中断处理流程,而是回归最底层——通过真实示波器波形,完整还原NEC协议在物理信号层面的构造原理、时序定义与电平映射关系。所有分析均基于实测数据,所有参数均来自NEC标准文档与VS1838B接收模块规格书,不依赖任何抽象库函数或中间层封装。

1.1 实验硬件配置与信号采集路径

实验平台采用STM32F103C8T6最小系统板,红外接收模块为VS1838B。该模块已集成红外信号放大、带通滤波(中心频率38 kHz)、解调与施密特触发整形功能,输出为标准TTL电平信号。其引脚定义如下:

引脚 功能 连接目标
VCC 电源输入 STM32开发板3.3V电源轨
GND 地线 开发板GND
OUT 解调后数字输出 GPIOA_Pin9(PA9)

PA9配置为浮空输入模式,无内部上下拉。外部电路采用4.7 kΩ上拉电阻至3.3 V,确保接收模块未收到有效红外信号时,OUT引脚呈高电平状态。此设计完全符合VS1838B数据手册推荐应用电路:当无红外载波时,内部光电三极管截止,OUT经上拉电阻输出高电平;当检测到38 kHz载波时,内部解调器翻转,OUT拉低。

示波器通道CH1接入PA9引脚,采样率设置为1 MSa/s,时基调整为2 ms/div,触发模式设为下降沿触发(因引导码起始为长低电平)。所用为PC端虚拟示波器(如PicoScope或Oliko),通过USB连接,软件实时显示并支持光标精确测量。

1.2 引导码(Leader Code)的实测波形与时间基准

按下遥控器任意按键瞬间,示波器捕获到的第一个显著特征是一段持续约9.1 ms的稳定低电平,紧随其后是一段约4.5 ms的高电平。光标测量值如下:

  • 起始下降沿时刻:t₀ = -9.53 ms
  • 引导码低电平结束时刻(上升沿):t₁ = -0.40 ms
  • Δt_low = t₁ - t₀ ≈ 9.13 ms
  • 随后高电平结束时刻(下降沿):t₂ = 4.10 ms
  • Δt_high = t₂ - t₁ ≈ 4.50 ms

该9.13 ms / 4.50 ms组合即为NEC协议规定的 引导码(Leader Code) 。其物理意义在于:为接收端提供足够长的同步窗口,使VS1838B内部锁相环(PLL)完成频率锁定,并通知MCU“一帧新数据即将开始”。此时间容差为±5%,即实际应用中允许8.5 ms ~ 9.5 ms的低电平与4.25 ms ~ 4.75 ms的高电平。

需特别注意电平极性反转现象:遥控器红外发射二极管在引导码阶段实际是 导通发光9 ms,随后关断4.5 ms ;而VS1838B作为接收端,其OUT引脚输出恰好相反——发光时输出低电平(0),关断时输出高电平(1)。这是由光电三极管工作原理决定的:光照增强 → 集电极电流增大 → 输出级晶体管饱和导通 → OUT对地短路 → 低电平。因此,所有后续分析均以 接收端OUT引脚的实际电平 为准,而非发射端的原始驱动信号。

1.3 数据位(Data Bit)的编码结构与时序差异

引导码结束后,立即进入32位数据帧传输。NEC协议不采用固定波特率的串行通信,而是通过 载波存在与否及持续时间 来编码逻辑值,其核心思想是避免长连0/1导致的时钟漂移问题。VS1838B已将38 kHz载波解调为纯净方波,因此我们直接观测到的是解调后的基带信号。

每个数据位由一个“高-低”电平对构成,其关键区别在于 低电平持续时间

逻辑值 高电平时间 低电平时间 总周期时间 物理含义(发射端)
0 0.56 ms 0.56 ms 1.12 ms 发送38 kHz载波0.56 ms + 关断0.56 ms
1 0.56 ms 1.68 ms 2.24 ms 发送38 kHz载波0.56 ms + 关断1.68 ms

实测波形验证了这一结构。在引导码之后的首个数据位区间,光标测量显示:
- 第一个高电平起始:t₃ = 4.10 ms
- 第一个高电平结束(下降沿):t₄ = 4.66 ms → Δt_high₁ = 0.56 ms
- 对应低电平结束(下一个高电平起始):t₅ = 5.22 ms → Δt_low₁ = 0.56 ms

该0.56 ms / 0.56 ms组合即为逻辑0。继续向后测量,可发现部分位的低电平明显延长至约1.68 ms(例如t₆ = 5.22 ms 至 t₇ = 6.90 ms),确认为逻辑1。二者高电平时间严格一致(0.56 ms),仅低电平时间呈3:1比例关系(1.68 ms / 0.56 ms = 3),这构成了NEC协议抗干扰的核心机制:接收端只需精确测量相邻下降沿之间的时间间隔,即可无歧义地区分0与1,无需维持全局时钟同步。

1.4 32位数据帧的完整结构与地址/命令校验机制

NEC标准帧长为32位,按顺序分为四段:用户地址(8位)、用户地址反码(8位)、命令码(8位)、命令码反码(8位)。实测波形清晰呈现了这一结构:

  1. 前8位(用户地址) :全部为逻辑0,对应8组0.56 ms高 + 0.56 ms低的等宽脉冲。
  2. 次8位(地址反码) :全部为逻辑1,对应8组0.56 ms高 + 1.68 ms低的长低脉冲。
  3. 再8位(命令码) :观测到序列 1010 0010 (即0xA2),对应高低脉冲交替组合。
  4. 末8位(命令反码) :观测到序列 0101 1101 (即0x5D),逐位与0xA2取反,完全吻合。

该设计提供了基础的错误检测能力。MCU在软件解码时,必须同时验证:
- 地址字节与反码字节是否互为按位取反;
- 命令字节与反码字节是否互为按位取反。

若任一校验失败,则丢弃该帧,防止误触发。值得注意的是,VS1838B本身不执行任何校验,它只负责将38 kHz载波解调为数字信号;所有协议解析均由MCU在GPIO中断或定时器捕获中完成。

1.5 VS1838B内部解调原理与载波滤除机制

一个常见误解是:“为何示波器看不到38 kHz载波?” 观察实测波形,所有高/低电平均为平稳直流电平,无任何高频振荡。这并非测量失误,而是VS1838B芯片的固有特性。

VS1838B内部集成了完整的红外接收链路:
- 前置放大器 :放大微弱的光电流信号;
- 带通滤波器(BPF) :中心频率38 kHz,带宽约±5 kHz,彻底抑制非38 kHz干扰(如日光、白炽灯闪烁);
- 解调器(Demodulator) :将38 kHz载波包络检出,输出其幅度变化;
- 施密特触发器(Schmitt Trigger) :对检出的包络进行整形,消除噪声毛刺,生成干净的方波。

因此,当遥控器发射一个逻辑0(38 kHz载波0.56 ms + 关断0.56 ms)时,VS1838B的BPF仅允许38 kHz成分通过,解调器将其识别为“有载波”,触发内部逻辑,使OUT引脚在载波期间保持低电平;关断期间无载波通过BPF,解调器输出静默,OUT被上拉电阻拉高。最终输出即为0.56 ms低 + 0.56 ms高的方波,38 kHz细节已被完全剥离。

这一设计极大简化了MCU端的处理负担:开发者无需编写复杂的载波计数器或FFT算法,仅需测量GPIO引脚电平跳变的时间间隔,即可完成全协议解析。这也是VS1838B成为行业事实标准的关键原因。

2. 时序精度要求与MCU捕获方案选型

物理层波形的精确解析,直接决定了软件解码的可靠性。NEC协议对时间容差有明确定义,MCU的捕获方案必须满足其严苛要求。

2.1 NEC协议时序容差规范

根据Philips NECEngineering Specification Rev. 1.0,各关键时序的允许偏差如下:

参数 标称值 允许范围 容差
引导码低电平 9 ms 8.5 ms ~ 9.5 ms ±5.6%
引导码高电平 4.5 ms 4.25 ms ~ 4.75 ms ±5.6%
逻辑0低电平 0.56 ms 0.5 ms ~ 0.62 ms ±10.7%
逻辑1低电平 1.68 ms 1.5 ms ~ 1.86 ms ±10.7%
位周期(0) 1.12 ms
位周期(1) 2.24 ms

可见,最严格的约束在引导码阶段(±5.6%),而数据位容差相对宽松(±10.7%)。这意味着MCU的时基精度必须优于±5%,否则无法可靠识别帧起始。

2.2 STM32通用定时器输入捕获方案

对于STM32F103系列,推荐使用TIM2或TIM3的输入捕获(Input Capture)功能。其优势在于:
- 硬件自动记录每次边沿发生的计数值,无软件中断延迟;
- 支持滤波器(ICx Filter),可配置4个连续采样周期才触发捕获,有效抑制GPIO引脚噪声;
- 捕获寄存器为16位,配合预分频器(PSC)可覆盖毫秒级时间测量。

典型配置如下(假设系统时钟72 MHz):

// 定时器基础配置
htim2.Instance = TIM2;
htim2.Init.Prescaler = 71;      // 72 MHz / (71+1) = 1 MHz → 1 μs/计数
htim2.Init.CounterMode = TIM_COUNTERMODE_UP;
htim2.Init.Period = 0xFFFF;     // 16位计数器,最大65535 μs (65.535 ms)
HAL_TIM_IC_Init(&htim2);

// 输入捕获通道1配置(对应PA9)
sConfigIC.ICPolarity = TIM_INPUTCHANNELPOLARITY_BOTHEDGE; // 捕获上升沿和下降沿
sConfigIC.ICSelection = TIM_ICSELECTION_DIRECTTI;
sConfigIC.ICPrescaler = TIM_ICPSC_DIV1;
sConfigIC.ICFilter = 0x07;      // 采样频率=定时器频率/8,即125 kHz,可滤除>62.5 kHz噪声
HAL_TIM_IC_ConfigChannel(&htim2, &sConfigIC, TIM_CHANNEL_1);

在中断服务函数中,读取 __HAL_TIM_GET_COUNTER(&htim2) __HAL_TIM_GET_COMPARE(&htim2, TIM_CHANNEL_1) 即可获得两次边沿间的精确时间差(单位:μs)。例如,若捕获到连续三个时间戳:t₀, t₁, t₂,则:
- 引导码低电平 = t₁ - t₀
- 引导码高电平 = t₂ - t₁

此方案误差主要来源于定时器时钟源精度(通常±1%)与GPIO输入滤波器引入的固定延迟(约1~2个系统时钟周期),完全满足NEC±5%容差要求。

2.3 GPIO中断+SysTick软件计时方案

若资源受限无法使用定时器捕获,可采用GPIO外部中断(EXTI)配合SysTick实现。但此方案风险较高:
- EXTI中断响应存在NVIC优先级延迟(通常数微秒);
- SysTick计数频率若低于1 MHz(如100 kHz),则0.56 ms分辨率仅为56个计数,量化误差达±0.89%(1/56),叠加中断延迟后总误差易超限。

仅建议用于对可靠性要求不高的演示场景,且必须关闭所有高优先级中断,并在中断服务函数中禁用SysTick中断以避免嵌套。

3. 接收端电平极性与发射端驱动的工程实践

理解电平极性反转不仅是理论要求,更是调试故障的钥匙。大量初学者在红外接收失败时,常陷入“为什么我收到的全是1?”或“为什么引导码识别不到?”的困惑,根源往往在此。

3.1 VS1838B输出电平与遥控器按键的对应关系

实测表明,VS1838B的OUT引脚行为严格遵循以下规则:
- 无红外信号 :OUT = 高电平(3.3 V)→ MCU读取GPIO为1;
- 接收到38 kHz载波 :OUT = 低电平(0 V)→ MCU读取GPIO为0;
- 载波中断 :OUT立即恢复高电平。

因此,NEC协议中“逻辑0”在接收端表现为“短低电平”,而“逻辑1”表现为“长低电平”。若开发者错误地将“高电平持续时间”作为判据(如认为高电平长就是1),则必然解码失败。

3.2 遥控器发射端的驱动真相

市售红外遥控器内部结构高度标准化:MCU(如HT6221)驱动一个红外LED,通过一个NPN三极管(如SS8050)进行电流放大。LED阳极接5 V,阴极经三极管集电极接地。当MCU输出高电平时,三极管导通,LED点亮;低电平时,三极管截止,LED熄灭。

因此,遥控器发射的“逻辑0”本质是:LED以38 kHz频率快速开关0.56 ms,人眼不可见,但VS1838B能检测到该载波;“逻辑1”则是LED以38 kHz开关0.56 ms后,保持熄灭1.68 ms。整个过程LED从未输出“直流高电平”,它永远在“开关”与“关断”间切换。

这一事实解释了为何不能用万用表直流档测量红外LED两端电压——你只会看到一个约2.5 V的平均值,毫无意义。唯有示波器才能捕捉其真实的开关行为。

3.3 硬件设计避坑指南

基于VS1838B的特性,在PCB布局与电路设计中必须规避以下陷阱:

  • 上拉电阻值选择 :推荐4.7 kΩ。阻值过小(如1 kΩ)会增大VS1838B灌电流负担,可能导致输出高电平不足(<2.4 V),被MCU误判为低;阻值过大(如10 kΩ)则上升沿缓慢,影响高速位(如逻辑1的1.68 ms)的精确捕获。
  • 电源去耦 :VS1838B对电源噪声敏感。必须在VCC引脚就近(<5 mm)放置0.1 μF陶瓷电容至GND,并联一个10 μF电解电容。未加去耦时,常见现象是接收距离骤减50%,或偶发误码。
  • 引脚布局 :OUT引脚走线应远离高速信号线(如USB、SPI)及大电流路径(如电机驱动),长度不超过3 cm。长线引入的分布电容会拖慢上升沿,导致逻辑1的1.68 ms低电平被误判为更长的无效脉冲。

4. 波形分析驱动的调试方法论

示波器不仅是观测工具,更是红外通信调试的终极诊断手段。当软件解码失败时,应按以下步骤逐层排查:

4.1 第一层:确认物理连接与供电

  • 测量VS1838B的VCC与GND间电压,必须为稳定3.3 V(±5%)。若为3.0 V,检查LDO负载能力或PCB走线压降。
  • 测量OUT引脚在无遥控时的静态电平,应为3.3 V。若为0 V,检查上拉电阻是否虚焊或阻值为0;若为1.8 V,检查VS1838B是否损坏或电源不足。
  • 对准遥控器,观察OUT是否有任何电平跳变。若完全无反应,更换遥控器电池或测试另一遥控器,排除发射端故障。

4.2 第二层:验证引导码完整性

  • 调整示波器时基至5 ms/div,触发模式为下降沿,捕获一次按键。
  • 确认是否存在9 ms左右的长低电平。若不存在,问题必在VS1838B前端(如滤波电容失效、BPF中心频率偏移)。
  • 若存在长低电平但后续无数据,检查引导码后4.5 ms高电平是否出现。若缺失,可能是VS1838B内部解调器锁定失败,尝试降低环境光强度或更换模块。

4.3 第三层:分析数据位时序一致性

  • 将时基缩至500 μs/div,聚焦引导码后的前几个数据位。
  • 测量每个位的高电平时间,应全部稳定在0.56 ms ±5%(0.53~0.59 ms)。若出现显著偏差(如0.4 ms或0.7 ms),说明定时器配置错误或时钟源不稳定。
  • 测量逻辑0与逻辑1的低电平时间比值,应严格为1:3。若偏离(如1:2.5),则需校准MCU时基,或检查示波器时间基准。

4.4 第四层:交叉验证地址/命令码

  • 记录实测波形中前16位(地址+反码)与后16位(命令+反码)的二进制序列。
  • 手动计算地址反码:对前8位逐位取反,应与第9~16位完全相同。若不同,问题在解码逻辑(如边沿捕获顺序错误、计数器溢出未处理)。
  • 同理验证命令码与反码。若仅地址校验失败而命令校验成功,大概率是引导码后第一个位捕获偏移了一个位周期。

我在实际项目中曾遇到一个案例:某批次VS1838B在低温(<5℃)下,引导码低电平缩短至8.2 ms,超出NEC容差,导致整机在北方冬季批量返厂。最终解决方案是在软件中动态放宽引导码容差至±8%,并增加温度传感器补偿。这印证了一个原则:协议规范是理想模型,真实硬件必须在极限条件下验证。

5. 从波形到代码:关键参数的工程化映射

将示波器读数转化为可运行的C代码,需建立精确的数学映射。以STM32 HAL库为例,核心是定义时间阈值常量:

// 基于1 MHz定时器时钟(1 μs/计数)的阈值定义
#define NEC_LEADER_LOW_MIN    8500U   // 8.5 ms → 8500 μs
#define NEC_LEADER_LOW_MAX    9500U   // 9.5 ms → 9500 μs
#define NEC_LEADER_HIGH_MIN   4250U   // 4.25 ms → 4250 μs
#define NEC_LEADER_HIGH_MAX   4750U   // 4.75 ms → 4750 μs
#define NEC_BIT0_LOW_MIN      500U    // 0.5 ms → 500 μs
#define NEC_BIT0_LOW_MAX      620U    // 0.62 ms → 620 μs
#define NEC_BIT1_LOW_MIN      1500U   // 1.5 ms → 1500 μs
#define NEC_BIT1_LOW_MAX      1860U   // 1.86 ms → 1860 μs

// 在定时器捕获中断中
uint32_t cap_val = HAL_TIM_ReadCapturedValue(&htim2, TIM_CHANNEL_1);
static uint32_t last_cap = 0;
uint32_t delta = cap_val - last_cap;
last_cap = cap_val;

// 判断引导码低电平
if ((delta >= NEC_LEADER_LOW_MIN) && (delta <= NEC_LEADER_LOW_MAX)) {
    state = NEC_STATE_WAIT_LEADER_HIGH;
}
// 判断数据位低电平
else if ((delta >= NEC_BIT0_LOW_MIN) && (delta <= NEC_BIT0_LOW_MAX)) {
    bit_value = 0;
}
else if ((delta >= NEC_BIT1_LOW_MIN) && (delta <= NEC_BIT1_LOW_MAX)) {
    bit_value = 1;
}

这些常量绝非凭空设定,而是直接源自示波器光标测量值(9.13 ms → 9130 μs)并向下取整至容差下限。实践中,建议在 NEC_BIT0_LOW_MIN NEC_BIT1_LOW_MIN 之间保留至少100 μs的安全间隙(620 μs vs 1500 μs),防止因温度漂移或时钟抖动导致误判。我踩过几次坑之后,现在所有红外项目都强制加入此间隙,并在量产测试中用温箱验证-40℃~85℃全温区稳定性。

真正的嵌入式工程师,从不把示波器束之高阁。它不是实验室的摆设,而是写在代码注释里的第一行调试依据。当你在凌晨三点面对一个不响的遥控器,最可靠的伙伴不是Stack Overflow,而是屏幕上那条跳动的、诚实的、不容置疑的绿色波形线。

Logo

openvela 操作系统专为 AIoT 领域量身定制,以轻量化、标准兼容、安全性和高度可扩展性为核心特点。openvela 以其卓越的技术优势,已成为众多物联网设备和 AI 硬件的技术首选,涵盖了智能手表、运动手环、智能音箱、耳机、智能家居设备以及机器人等多个领域。

更多推荐