小型AUV控制系统开发笔记[一]基于STM32的消息中继器

从2021-2-9开始到2021-2-25该项目基本完成,写一篇笔记算是记录也是纪念把。
第二句话,要感谢一下@iroek,他DE了程序里的所有BUG。如果说为什么要第二句话,因为如果可以的话他喜欢低调。

先来一张全家福
在这里插入图片描述
在这里插入图片描述

虽然小型AUV的控制算法是由运算能力较强的RK3399芯片完成的,但是在收集传感器数据、编写传感器驱动方面,在运行Linux内核的芯片上写起来不是很方便,于是便有了标题中的基于STM32的消息中继器。

消息中继器主要完成的任务就像他的名字一样,完成消息中继。收集各个外设(传感器)的数据,然后以一定的封包格式通过一种特定的通信方式发送给RK3399。

目前该程序已经基本完成,后续添加设备只需要在现有程序框架下添加相应的代码。

硬件

  1. STM32F103C8T6核心板
  2. P30高度计 官方网站
  3. B30深度计 官方网站

软件

  1. STM32CubeIDE
  2. 串口调试助手

软件基本架构
软件采用HAL库为基础,通过Freertos进行任务调度,实现多个任务的调度。
目前主要有三个任务

  1. rosserial时间同步任务
  2. B30深度计任务
  3. P30高度计任务

下面主要根据上述任务对整个中继器的程序进行介绍

rosserial时间同步任务

该任务是最重要的一个任务,用来与RK3399上的rosserial节点进行时间同步,其依赖于rosserial库,进而其他任务才能将数据发送到rosmaster。在RK3399的ROS环境下可以通过较为简单的方式获取STM32上传感器所publish的数据。
下面主要说明一下该任务在源码中路径以及如何移植。
在这里插入图片描述
除了上述三个文件还有main.c里面涉及任务创建的内容以及ros_lib文件夹中的消息库。

可以在安装有ROS环境的机器(STM32未来要连接的机器)上运行下述命令来生成需要安装的库

rosrun rosserial_arduino make_libraries.py /home
cd home

目录下的ros_lib文件夹就为上图中的ros_lib文件夹。

如果要移植rosserial,说到底需要需要实现如下4个函数

  • int init()
    • init函数主要用来执行硬件的初始化操作
  • uint32_t time()
    • 获取时间戳单位为ms
  • int read()
    • 读取一个字节
  • int write(uint8_t* data, uint16_t length)
    • 写入一定长度的数据

由于使用HAL库,time直接可以由HAL_GetTick()函数获取,read和write和使用的通信方式有关,在使用串口的情况下可以调用串口发送和接收相关的函数。
消息中继器采用的是,开启串口的接受DMA中断,在中断内将每一个字节复制进入环形缓冲区。read函数直接从环形缓冲区内读数据。具体代码可以在STM32ROSSerialDevice.h内找到。

深度计任务

深度计任务主要是通过硬件IIC读取深度计的深度数据,通过rosserial 将其数据发布到相应话题下。
深度计提供了一个官方例程是使用软件IIC的。深度计官方例程

在这里插入图片描述
除了图片上标注的三个文件还有main.c中和其相关的任务代码。

移植深度计库主要是使用HAL库函数

//阻塞
HAL_StatusTypeDef HAL_I2C_Master_Transmit(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint8_t *pData, uint16_t Size, uint32_t Timeout);
HAL_StatusTypeDef HAL_I2C_Master_Receive(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint8_t *pData, uint16_t Size, uint32_t Timeout);
//非阻塞
HAL_StatusTypeDef HAL_I2C_Master_Transmit_IT(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint8_t *pData, uint16_t Size);
HAL_StatusTypeDef HAL_I2C_Master_Receive_IT(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint8_t *pData, uint16_t Size);

在使用DMA中断接收数据要慎用HAL_I2C_Master_Receive,该函数内部会关闭所有中断,会导致DMA丢失数据。

PS1:在使用HAL_I2C_Master_Transmit_IT如果手动模拟错误拉低SDA线,会导致单片机宕机。
PS2:高度计库目前没有做硬件平台无关的处理,目前硬件和库还处于耦合状态,未来时间充足,会优化代码。

高度计

高度计任务通过配置P30设备到自动工作状态下,解析数据,将距离和置信度发布到相应话题下。
官方提供一个开源的测试程序ping-viewer 项目地址
消息中继器使用和其相同的数据解析库 ping-cpp 项目地址

在这里插入图片描述
移植P30需要实现

  1. uint8_t init()
    init函数主要用来执行硬件的初始化操作
  2. uint16_t readByte()
    读取一个字节返回
  3. size_t write(uint8_t* data, uint16_t length)
    写入一定数量的数据
  4. uint32_t get_tick()
    获取时间戳,单位为ms

与移植rosserial相似,同样采用环形缓冲区储存数据,利用串口DMA中断接收数据。
PS:目前已知ping-parser存在可能数组越界BUG,该BUG会导致单片机宕机。
在数据部分丢失情况下,恰好包头完整,但后续PAYLOAD_LENGTH丢失,后续数据补位导致,PAYLOAD_LENGTH超过缓冲区大小,在接收数据时并没有做溢出检查,会导致数组越界。

项目Github地址

对于整个AUV控制系统相关的程序,都会在 https://github.com/UWVG开源。
该项目地址在 https://github.com/UWVG/AUV-Message-Relayer

项目里的commit都是原汁原味的,并没有最后统一改成一个,因为那些也是调试BUG的美好记忆,你说对吧。

免费下载,论文、设计请注明引用! 微小型AUV具有体积小,灵活性高、隐蔽性好等特点,可以工作于其它大型水下机器人无法进入的区域。民用上可以应用于海洋矿产勘探、海底地形探测,沉船打捞,水下考古,海洋生物探测等;军事上可以用来反水雷,作为自航水雷的载体、监察海战时水下敌情等。 首先,本文对所设计的微小型AUV的结构、推进器分布进行介绍,并对其进行受力分析和建立运动方程。结合运动方程设计了被控对象模型未知的AUV自动定深、自动定航控制器;同时研究了传统的PID控制、模糊控制、自适应控制等算法,并最终设计了应用于该微小型AUV的模糊参数自适应PID控制算法。 其次,对该具有多传感器的微小型AUV控制系统进行了研究设计。针对分布式控制系统总体机构及其通信总线进行了设计;分别详细设计了分布式系统的各个子系统;着重研究、设计了理论、算法及软件实现方案;计了基于CAN总线的分布式微小型AUV控制系统,提高了系统的稳定性和模块化程度,在结构上优化了系统的复杂性。最终形成了由软硬件系统组成分布式控制系统。 再其次,根据SINS、DVL和深度计这三个传感器的姿态角、角速度,线速度、加速度,深度等导航信息进行了AUV的航位推算研究与实现;并使用综合水池实验室的X-Y航车系统,反复试验,对航位推算进行了标定,修正了安装误差角和刻度因子。提高了航位推算精度。结合航位推算和AUV制导控制设计了有海流影响的AUV自动巡航控制器。抗海流自动巡航控制器除抗海流功能外可以补偿SINS与艏向安装误差带来的控制性能缺陷。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值