PX4飞控实战:为纳雷NRA12激光雷达手搓一个串口驱动(附完整源码)
PX4飞控实战为纳雷NRA12激光雷达手搓一个串口驱动附完整源码去年夏天我在调试一台农业植保无人机时遇到了一个棘手的问题——现有的激光雷达在强光环境下表现不稳定。经过多次测试对比最终选定了纳雷NRA12这款抗干扰能力强的激光雷达。但当我准备将其接入PX4飞控时发现官方固件并不支持这款设备。于是一场从零开始的驱动开发之旅就此展开。1. 硬件准备与协议分析在开始编码之前我们需要对硬件和通信协议有充分了解。NRA12激光雷达通过串口与飞控通信采用115200波特率8位数据位无校验位1位停止位。其数据协议采用二进制帧格式每帧包含12个字节具体结构如下字节位置含义说明0-1帧头固定为0xAA 0xAA2-3命令字0x0C 0x07表示距离数据4数据索引递增序号5保留位固定为0x006-7距离数据高位在前单位厘米8-11帧尾固定为0x55 0x55实际测试中发现雷达在强光直射下仍能保持稳定输出这得益于其特殊的光学滤波设计。接线时需要注意使用Pixhawk的TELEM/SERIAL4口UART4确保供电电压在4.5-5.5V范围内避免数据线与电源线平行走线过长2. 驱动框架搭建PX4的驱动开发有其特定的框架要求。我们需要创建一个完整的驱动模块主要包括以下文件nra12/ ├── CMakeLists.txt ├── Kconfig ├── NRA12.cpp ├── NRA12.hpp ├── module.yaml ├── nra12_main.cpp ├── nra12_parser.cpp └── nra12_parser.h关键点在于继承ScheduledWorkItem类并实现必要接口。在NRA12.hpp中我们定义了核心类结构class NRA12 : public px4::ScheduledWorkItem { public: NRA12(const char *port, uint8_t rotation distance_sensor_s::ROTATION_DOWNWARD_FACING); virtual ~NRA12(); int init(); void print_info(); private: int collect(); void Run() override; void start(); void stop(); PX4Rangefinder _px4_rangefinder; NRA12_PARSE_STATE _parse_state{NRA12_PARSE_STATE::STATE0_UNSYNC}; char _linebuf[24]{}; char _port[20]{}; int _fd{-1}; // ...其他成员变量 };特别需要注意module.yaml的配置它定义了模块的启动命令和参数module_name: nanoradar nra12 lidar serial_config: - command: nra12 start -d ${SERIAL_DEV} port_config_param: name: SENS_NRA12_CFG group: Sensors3. 数据解析实现数据解析是驱动最核心的部分。NRA12采用状态机模式解析数据流我在nra12_parser.h中定义了14种解析状态enum class NRA12_PARSE_STATE { STATE0_UNSYNC 0, STATE1_GOT_START1, STATE2_GOT_START2, // ...中间状态省略 STATE12_GOT_END2 };实际解析函数需要处理各种异常情况。以下是关键代码片段int nra12_parse(char c, char *parserbuf, unsigned *parserbuf_index, NRA12_PARSE_STATE *state, float *dist) { switch (*state) { case STATE12_GOT_END2: if (c 0xaa) *state STATE1_GOT_START1; else *state STATE0_UNSYNC; break; case STATE0_UNSYNC: if (c 0xaa) *state STATE1_GOT_START1; break; // ...其他状态处理 case STATE11_GOT_END1: if(c 0x55) { *state STATE12_GOT_END2; unsigned int t1 parserbuf[2]; unsigned int t2 parserbuf[1]; t2 8; t2 t1; if (t2 0xFFFFu) *dist ((float)t2)/100; } break; } return -1; }在实测中发现解析器需要特别处理以下边界情况数据帧不完整时的恢复机制连续收到多个0xAA时的同步问题电磁干扰导致的异常数据4. 系统集成与测试完成驱动开发后需要将其集成到PX4构建系统中。主要修改点包括在src/drivers/distance_sensor/CMakeLists.txt中添加add_subdirectory(nra12)修改src/drivers/distance_sensor/Kconfigselect DRIVERS_DISTANCE_SENSOR_NRA12在src/drivers/drv_sensor.h中定义设备类型#define DRV_DIST_DEVTYPE_NRA12 0xC2在板级配置如boards/px4/fmu-v5/default.px4board中启用驱动CONFIG_COMMON_DISTANCE_SENSOR_NRA12y编译并烧录固件后通过以下步骤测试# 启动驱动 nra12 start -d /dev/ttyS3 # 查看状态 nra12 status # 通过uORB查看数据 listener distance_sensor测试时发现NRA12在30米范围内的测距误差小于2%但在强反光表面如玻璃温室会出现短暂数据异常。解决方法是在驱动中增加数据滤波_px4_rangefinder.set_min_distance(0.4f); // 原厂建议0.3m实际提高到0.4m减少误报 _px4_rangefinder.set_max_distance(30.0f); _px4_rangefinder.set_fov(math::radians(1.15f));5. 参数配置与优化为了让驱动更灵活我们通过PX4参数系统提供配置选项。主要参数包括参数名描述默认值单位SENS_NRA12_CFG串口设备选择禁用-SENS_NRA12_ROT传感器安装方向向下-SENS_NRA12_MIN最小有效距离0.4米SENS_NRA12_MAX最大有效距离30.0米在QGroundControl中配置参数的步骤进入参数界面搜索NRA12设置SENS_NRA12_CFG为对应的串口如TELEM/SERIAL4根据实际安装调整旋转参数实际飞行测试中发现在高速移动时偶尔会出现数据丢失。通过调整采样策略解决了这个问题// 原采样间隔7ms改为动态调整 void NRA12::Run() { if (collect() -EAGAIN) { ScheduleClear(); ScheduleOnInterval(7_ms, 87 * 9); // 重试时增加间隔 return; } }6. 性能调优与问题排查驱动开发完成后我使用PX4自带的性能分析工具进行了优化# 查看驱动性能统计 perf # 输出示例 nra12: read: events, elapsed time, max time, max time %% nra12: com_err: events常见问题及解决方法无数据输出检查接线是否正确确认串口配置波特率、数据位等使用示波器检查信号质量数据不稳定增加电源滤波电容调整min_distance参数检查是否有电磁干扰源驱动无法启动确认CONFIG_COMMON_DISTANCE_SENSOR_NRA12y检查module.yaml配置查看系统日志dmesg一个特别棘手的问题是在高温环境下50℃出现的通信中断。最终发现是串口线材质量问题更换为屏蔽线后问题解决。7. 完整源码实现所有开发完成的源代码已托管至Git仓库包含以下关键文件NRA12.cpp核心部分实现了设备初始化和数据采集int NRA12::init() { // 串口配置 unsigned speed B115200; termios uart_config{}; if ((termios_state cfsetispeed(uart_config, speed)) 0) { PX4_ERR(CFG: %d ISPD, termios_state); return -1; } // 8N1配置 uart_config.c_cflag ~CSIZE; uart_config.c_cflag | CS8; uart_config.c_cflag ~PARENB; uart_config.c_cflag ~CSTOPB; // 应用配置 if ((termios_state tcsetattr(_fd, TCSANOW, uart_config)) 0) { PX4_ERR(baud %d ATTR, termios_state); return -1; } // 启动工作队列 start(); return PX4_OK; }nra12_main.cpp实现了模块的shell接口extern C __EXPORT int nra12_main(int argc, char *argv[]) { const char *device_path NRA12_DEFAULT_PORT; uint8_t rotation distance_sensor_s::ROTATION_DOWNWARD_FACING; // 参数解析 int ch; while ((ch px4_getopt(argc, argv, R:d:, myoptind, myoptarg)) ! EOF) { switch (ch) { case R: rotation (uint8_t)atoi(myoptarg); break; case d: device_path myoptarg; break; } } // 命令分发 if (!strcmp(argv[myoptind], start)) { return nra12::start(device_path, rotation); } else if (!strcmp(argv[myoptind], stop)) { return nra12::stop(); } // ...其他命令 }整个开发过程中最耗时的部分是协议解析的稳定性测试。前后共尝试了三种不同的状态机实现方案最终选择了现在这个在性能和可靠性之间取得平衡的版本。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2466134.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!