树莓派实时性优化
天下武功,唯快不破
优化目的
项目需要用 树莓派 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
并未改变调度策略,实际优化效果很有限。需要注意的是,此处设置的优先级是用于比较的相对值,严格地说并不是越大就越好。
The priority parameter works relative to others – so you can make one program priority 1 and another priority 2 and it will have the same effect as setting one to 10 and the other to 90 (as long as no other programs are running with elevated priorities)
priority 参数 相对 于其他参数有效 - 因此您可以将一个程序的优先级设置为 1,将另一个程序的优先级设置为 2,其效果与将一个设置为 10,另一个设置为 90 相同(只要没有其他程序以更高的优先级运行)。
绑定独立 CPU
操作
1. 首先需要更改内核启动参数:
!!!请注意:单引号中的空格为必须!!!
其中 3
可以更改为你想要指定的 CPU 编号(树莓派 4B 中 CPU 编号为 0-3)。
考虑到未来可能需要还原,此步将原文件添加 .bak 后缀进行了备份。
2. 接着安装后面要用到的 psutil
包
3. 最后在程序中加入以下内容:
此处的 3
为上文提到的 CPU 编号
原理
首先让系统不会主动调度程序到指定 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
优化后
AVG: 82.94 us MAX: 6550.83 us MIN: 32.67 us
网上已有的树莓派 3B+ 上的具体性能测试数据
请注意,此处并非本次优化的数据。
测试结论
从以上数据可以看出,系统实时性提升主要体现在 极值 的降低上。
在实际程序中,均值与极值都有非常可观的降低。 优化完成后,手头 Python 运行的测试误差为 30 us 左右,考虑到项目可容许的偏差是 50 us,终于——咱不用去写 C 啦!o( ̄▽ ̄) ブ
参考源
最后更新于