1. PS2键盘协议基础与单片机模拟场景
你可能在旧电脑上见过那个圆圆的紫色接口——那就是PS2键盘的专属插座。虽然现在USB键盘已成主流,但在嵌入式领域,PS2协议因其简单可靠的特性依然被广泛应用。我用STM32模拟PS2键盘时发现,只需要两个GPIO口就能实现完整键盘功能,这比USB协议简单太多了。
PS2协议本质上是一种双向同步串行通信协议,包含CLK(时钟)和DATA(数据)两根信号线。数据传输速率在10-20kHz之间,每个数据帧包含11位:1位起始位(总是0)、8位数据位(LSB先行)、1位奇校验位和1位停止位(总是1)。实际测试中发现,当CLK线从高电平变为低电平时,DATA线上的数据才有效。
2. 硬件连接与信号时序控制
2.1 接口电路设计
PS2接口的物理连接非常简单,只需要注意以下几点:
- 时钟线通常需要接单片机的输入捕获或外部中断引脚
- 数据线接普通GPIO即可
- 建议在两条线上都加上1kΩ上拉电阻
我曾在项目中直接省略上拉电阻,结果出现数据丢包现象。后来用示波器抓波形发现,当线路较长时信号上升沿变缓,加上上拉后问题立即解决。
2.2 关键时序参数
通过实测多款PS2键盘,总结出以下关键时序参数:
| 参数项 | 典型值 | 允许偏差 |
|---|---|---|
| 时钟周期 | 60μs | ±10μs |
| 数据建立时间 | 20μs | ≥5μs |
| 数据保持时间 | 40μs | ≥30μs |
| 帧间隔时间 | 50μs | ≥30μs |
在代码实现时,我习惯用定时器精确控制这些时序。比如用STM32的TIM2定时器产生20μs基准时基,所有延时都基于这个时基进行倍频或分频。
3. 单片机模拟键盘的核心代码实现
3.1 单比特发送函数
这是整个系统最底层的函数,直接操作GPIO实现单bit发送:
void PS2_SendBit(bool bit_val) { DATA_PIN = bit_val ? HIGH : LOW; // 准备数据 delay_us(20); // 保持数据稳定 CLK_PIN = LOW; // 拉低时钟线 delay_us(40); // 保持时钟低电平 CLK_PIN = HIGH; // 释放时钟线 delay_us(20); // 时钟高电平期间数据变化 }调试这个函数时有个坑:必须确保在CLK变高前DATA已经稳定。我有次把delay_us(20)放在CLK操作之后,导致PC端经常收到错误数据。
3.2 完整数据帧发送
基于单比特发送函数,我们可以构建完整的数据帧发送逻辑:
void PS2_SendByte(uint8_t data) { uint8_t parity = 1; // 奇校验计算 // 发送起始位 PS2_SendBit(0); // 发送8位数据 for(int i=0; i<8; i++) { bool bit = data & 0x01; PS2_SendBit(bit); parity ^= bit; // 计算奇校验 data >>= 1; } // 发送校验位和停止位 PS2_SendBit(parity); PS2_SendBit(1); // 帧间隔 delay_us(50); }实际应用中,PC端可能在忙无法立即接收数据。完善的实现应该增加主机抑制状态检测:
bool PS2_WaitHostReady() { int timeout = 5; // 尝试5次 while(timeout-- && !CLK_PIN) { delay_us(50); } return timeout > 0; }4. 键盘扫描码与特殊功能实现
4.1 第二套扫描码解析
现代PC主要使用第二套扫描码,每个按键都有独立的通码和断码。例如:
- 字母"A"的通码是0x1C,断码是0xF0+0x1C
- 组合键"Shift+A"会先发送0x12(Shift),再发0x1C
我在项目中建立了这样的扫描码映射表:
const uint8_t KEYMAP[] = { [0x1C] = 'A', [0x32] = 'B', // ...其他键值映射 [0x12] = KEY_SHIFT, [0x14] = KEY_CTRL };4.2 特殊功能处理
对于CapsLock、NumLock等带状态指示灯的按键,需要维护内部状态:
bool caps_lock = false; void HandleSpecialKey(uint8_t scancode) { switch(scancode) { case 0x58: // CapsLock caps_lock = !caps_lock; PS2_SetLEDs(0, caps_lock, 0); break; // 其他特殊键处理 } }5. 常见问题与调试技巧
5.1 数据丢包问题排查
遇到数据丢包时,建议按以下步骤排查:
- 用逻辑分析仪抓取CLK和DATA信号
- 检查时序是否符合规范
- 确认电源电压稳定(PS2设备对电压敏感)
- 检查线路阻抗是否匹配
5.2 抗干扰设计
在工业环境中,我通常会:
- 使用双绞线连接
- 在信号线上加100pF滤波电容
- 单片机端增加TVS二极管防护
有个项目在电机附近使用时出现随机误触发,后来发现是CLK线太长成了天线,缩短到10cm后问题消失。
6. 性能优化与扩展应用
6.1 中断驱动实现
对于资源紧张的单片机,可以用外部中断优化CLK检测:
void EXTI_IRQHandler() { static uint8_t bit_count = 0; static uint8_t shift_reg = 0; if(CLK_PIN == LOW) { bool bit = DATA_PIN; shift_reg = (shift_reg >> 1) | (bit << 7); if(++bit_count == 11) { ProcessScancode(shift_reg); bit_count = 0; } } }6.2 多设备扩展
通过模拟多个PS2设备,可以实现键盘+鼠标的复合功能。需要特别注意:
- 设备识别时序
- 冲突仲裁机制
- 电源负载能力
在某个工控面板项目中,我成功实现了ATmega328同时模拟键盘和触摸板,关键是要严格错开两者的通信时段。