STM32与PS2手柄的无线交互:从硬件对接到按键解析
1. 认识PS2手柄与STM32的无线交互第一次接触PS2手柄和STM32的对接时我完全被这个经典游戏手柄的通信协议吸引了。你可能不知道这个2000年推出的手柄至今仍在嵌入式领域发光发热主要得益于它简单的通信协议和稳定的性能。我实测过市面上常见的几种手柄PS2手柄的性价比和易用性确实突出。PS2手柄通过SPI协议与主机通信而STM32恰好内置了硬件SPI控制器这种天然的匹配让对接变得简单。在实际项目中我常用PS2手柄来控制机器人、无人机或者作为调试输入设备。相比直接使用按键矩阵手柄带来的优势很明显摇杆提供模拟量输入、按键布局合理、还有震动反馈功能。这里有个有趣的现象虽然PS2手柄已经停产多年但淘宝上仍然能买到全新的兼容手柄价格只要30-50元。我买过不同批次的兼容手柄测试发现它们的通信协议完全一致这对开发者来说是个好消息。2. 硬件连接与引脚配置2.1 必备材料清单开始动手前你需要准备这些材料STM32开发板我用的是STM32F103C8T6最小系统板PS2手柄接收器拆机件约15元杜邦线若干3.3V稳压模块如果开发板没有3.3V输出特别提醒PS2手柄接收器的工作电压是3.3V绝对不能接5V我第一次实验时就烧坏了一个接收器。现在我的做法是先用万用表确认电压再连接。2.2 引脚连接示意图接收器有9个引脚但我们只需要连接6个接收器引脚 STM32引脚 DATA - PA6(MISO) CMD - PA7(MOSI) SCK - PA5(SCK) CS - PA4(SS) VCC - 3.3V GND - GND注意ATT引脚不需要连接这是索尼设计用来支持多设备共享总线的我们单设备应用可以直接忽略。连接时建议用不同颜色的杜邦线区分信号线后期调试会方便很多。3. 底层驱动实现3.1 SPI初始化配置在STM32CubeIDE中配置SPI1为主机模式hspi1.Instance SPI1; hspi1.Init.Mode SPI_MODE_MASTER; hspi1.Init.Direction SPI_DIRECTION_2LINES; hspi1.Init.DataSize SPI_DATASIZE_8BIT; hspi1.Init.CLKPolarity SPI_POLARITY_LOW; hspi1.Init.CLKPhase SPI_PHASE_1EDGE; hspi1.Init.NSS SPI_NSS_SOFT; hspi1.Init.BaudRatePrescaler SPI_BAUDRATEPRESCALER_256; hspi1.Init.FirstBit SPI_FIRSTBIT_MSB; hspi1.Init.TIMode SPI_TIMODE_DISABLE; hspi1.Init.CRCCalculation SPI_CRCCALCULATION_DISABLE;这里有个坑要注意PS2手柄的SPI时钟频率不能太高实测超过250kHz就容易出现通信失败。我建议初始设置为125kHz分频系数256稳定后再尝试提高。3.2 通信协议解析PS2手柄采用问答式通信主机先发送命令手柄返回状态数据。一个完整的通信周期如下拉低CS信号发送0x01命令字节发送0x42请求数据发送6个0x00空字节同时接收手柄返回的6字节数据拉高CS信号用STM32 HAL库实现的代码示例uint8_t txData[8] {0x01, 0x42, 0, 0, 0, 0, 0, 0}; uint8_t rxData[8] {0}; HAL_GPIO_WritePin(GPIOA, GPIO_PIN_4, GPIO_PIN_RESET); HAL_SPI_TransmitReceive(hspi1, txData, rxData, 8, 100); HAL_GPIO_WritePin(GPIOA, GPIO_PIN_4, GPIO_PIN_SET);4. 数据解析与处理4.1 按键状态解析接收到的第3字节到第6字节包含所有按键状态。我习惯用位域结构体来解析typedef struct { uint8_t select:1; uint8_t l3:1; uint8_t r3:1; uint8_t start:1; uint8_t up:1; uint8_t right:1; uint8_t down:1; uint8_t left:1; // 其他按键省略... } PS2_ButtonState;实际使用时要注意按键是低电平有效即按下时对应位为0。我通常会做一次取反操作PS2_ButtonState buttons *(PS2_ButtonState*)rxData[2]; buttons.all ^ 0xFF; // 全部取反4.2 摇杆数据处理摇杆数据在第5和第6字节范围是0x00-0xFF。但实际测试发现中立点不一定是0x80不同手柄有差异。我的做法是启动时先读取100次中立值求平均uint32_t sum_x 0, sum_y 0; for(int i0; i100; i){ PS2_ReadData(); sum_x rxData[4]; sum_y rxData[5]; } neutral_x sum_x / 100; neutral_y sum_y / 100;使用时减去中立值就得到相对位移int16_t x rxData[4] - neutral_x; int16_t y rxData[5] - neutral_y;5. 实际应用案例5.1 机器人遥控实现我用这套方案做了一个履带机器人手柄控制逻辑如下void HandlePS2Input(){ PS2_ReadData(); // 左摇杆控制前进/后退和转向 int16_t throttle (rxData[5] - neutral_y) / 2; int16_t steer (rxData[4] - neutral_x) / 2; // 右肩键加速 if(buttons.r1) { throttle * 2; steer * 2; } SetMotorSpeed(MOTOR_LEFT, throttle steer); SetMotorSpeed(MOTOR_RIGHT, throttle - steer); }5.2 菜单导航系统在另一个OLED显示项目中我用方向键导航菜单if(buttons.up !last_buttons.up) { menu_index (menu_index - 1 MENU_COUNT) % MENU_COUNT; } if(buttons.down !last_buttons.down) { menu_index (menu_index 1) % MENU_COUNT; } if(buttons.cross !last_buttons.cross) { ExecuteMenuItem(menu_index); }这里的关键是检测按键边缘变化避免长按导致的连续触发。我用了last_buttons保存上一帧状态来实现这个功能。6. 常见问题排查6.1 通信失败排查步骤当手柄无响应时我通常这样排查用逻辑分析仪抓取SPI波形确认CS、SCK信号正常检查3.3V电源是否稳定手柄工作时电流约20mA尝试降低SPI时钟频率更换手柄测试排除手柄故障有个特殊情况部分兼容手柄需要先按住SELECT键再上电才能进入SPI模式。这个坑我踩过好几次。6.2 数据抖动处理摇杆数据有时会出现1-2个LSB的抖动我的解决方案是软件滤波#define FILTER_SIZE 5 static uint8_t x_history[FILTER_SIZE]; static uint8_t y_history[FILTER_SIZE]; // 更新历史数据 memmove(x_history, x_history1, FILTER_SIZE-1); x_history[FILTER_SIZE-1] rxData[4]; // 对y轴做同样处理... // 中值滤波 qsort(x_history, FILTER_SIZE, sizeof(uint8_t), compare); uint8_t filtered_x x_history[FILTER_SIZE/2];这种简单的滤波算法就能消除大部分抖动而且不会引入明显延迟。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2624007.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!