树莓派实时性优化
天下武功,唯快不破
优化目的
项目需要用 树莓派 4B 实现 毫秒级控制,并且误差控制在 5% 内(即 0.05ms = 50us)。 本身这种精度的控制应该考虑由 C 实现,但考虑到以下原因,最终还是选用了 Python 作为主语言。
用惯了 Python,好久好久没有写过 C 了,真的 不想写 C 啊啊啊啊 (/▽\)
控制过程中 Python 调用的 包多是封装了底层 C 库实现,实际性能并不差
整体程序框架中,需要 Python 实现的顶层控制非常简单
那么,为了实现 50 us 的控制精度,咱就需要针对树莓派上运行的 Raspberry Pi OS 及 Python 进程 进行优化。
优化内容
优化的主要思路参考了下面的两篇文章:
提升 Python 进程优先级
操作
1. 首先需要安装 wiringpi
包:
2. 其次需要在程序中加入以下内容:
原理
wiringpi
中的 piHiPri
函数通过调用 Linux 内核的 API,改变了当前进程的调度策略及优先级进一步提高了实时性。这和 Python 中标准库的 os.setpriority
不同,os.setpriority
并未改变调度策略,实际优化效果很有限。需要注意的是,此处设置的优先级是用于比较的相对值,严格地说并不是越大就越好。
绑定独立 CPU
操作
1. 首先需要更改内核启动参数:
!!!请注意:单引号中的空格为必须!!!
其中 3
可以更改为你想要指定的 CPU 编号(树莓派 4B 中 CPU 编号为 0-3)。
考虑到未来可能需要还原,此步将原文件添加 .bak 后缀进行了备份。
2. 接着安装后面要用到的 psutil
包
3. 最后在程序中加入以下内容:
原理
首先让系统不会主动调度程序到指定 CPU 上运行,也就是 保证指定 CPU 默认是完全空闲 的状态。其次让 程序主动要求在指定 CPU 上运行。这样程序就完成了 对指定 CPU 的独享。
打内核 RT 补丁
操作
1. 这里作者选择直接下载编译好的内核:
链接源自 Github,所以需要科学上网,你懂的 ( ̄_ ̄ )
2. 将下载的压缩包传至树莓派 /tmp 目录下:
基础操作,如不理解请善用搜索引擎
3. 依次运行以下命令:
4. 重启系统:
5. 检查内核版本:
若输出为 4.19.59-rt23-v7l+
,说明安装成功。
原理
Linux 是典型的 分时应用系统 ,对于实时性要求很高的应用,必须对内核本身动手术。而 RTLinux 则采取了一种比较聪明也比较折中的办法:他们实现一个最底层的精简的调度器,用于调度实时线程,原来的内核本身则成为实时调度器的一个优先级最低的任务。这样,当有实时任务时,普通内核已经建立于其上的普通进程被强制中断,实时线程被强制执行;只有当若有实时线程都让出 cpu 之后,普通内核才被运行,再由普通内核去调度执行普通的应用程序。
性能测试
测试方法
用
rt-tests
软件包中的cyclictest
工具测试系统实时性:用以下程序测试程序实时性:
测试数据
优化前
AVG: 82.94 us MAX: 6550.83 us MIN: 32.67 us
优化后
网上已有的树莓派 3B+ 上的具体性能测试数据
请注意,此处并非本次优化的数据。
测试结论
从以上数据可以看出,系统实时性提升主要体现在 极值 的降低上。
在实际程序中,均值与极值都有非常可观的降低。 优化完成后,手头 Python 运行的测试误差为 30 us 左右,考虑到项目可容许的偏差是 50 us,终于——咱不用去写 C 啦!o( ̄▽ ̄) ブ
参考源
最后更新于