单片机怎么实现真正的多线程
单片机怎么实现真正的多线程?
不考虑多核情况时,CPU在一个时间点只能做一件事,因为切换的速度快所以看起来好像是同时执行多个线程而已。
实际上就是用定时器来做时基,以时间片的方式分别执行来实现的,只不过实现起来细节比较复杂,核心思想就是你猜想的那样。最近很多小伙伴找我,说想要一些单片机的资料,然后我根据自己从业十年经验,熬夜搞了几个通宵,精心整理了一份「单片机资料从专业入门到高级教程+工具包」,点个关注,全部无偿共享给大家!!!
评论区回复“888”,关注我之后私信回复“666”,即可拿走。
如果你仅仅想知道具体思路,我可以告诉你。就是划分一块内存区域做线程的上下文切换空间,另外以一个定时器做定时时基。例如设为1ms,每隔1ms检测是否有其它线程要工作,如果有,保存当前线程的CPU寄存器以及工作状态到当前线程的上下文空间,从要运行的线程上下文空间取出寄存器值填充到CPU寄存器中,这样就完成了线程的切换,CPU就接着另一个线程的工作继续做下去了。
打个比方,我们把一个人关在小房间里,给他布置了写作业和洗衣服的两个任务。
线程CPU使用率该如何计算?
这篇笔记有如下内容:
1、为什么需要计算各个线程的CPU使用率?
2、该如何计算线程CPU使用率?
3、FreeRTOS线程计算的弊端?如何打破 FreeRTOS 线程计算方式的时间限制?
4、关键代码介绍。
上次介绍了如何计算整个系统的CPU使用率:
《单片机里面的CPU使用率是什么鬼? 》
《实操RT-Thread系统CPU利用率功能添加 》
但是却没有介绍该如何计算每个线程(任务)的CPU使用率。
为什么要计算线程CPU使用率
首先要问的是,为什么要计算线程的CPU使用率,有啥用?
我们知道系统的CPU使用率关注的是整个系统的使用情况,使用率越低,表示越能更及时的响应外部情况,整个系统的性能也会越好。
但这是从系统整体考量的,并不能反映单个线程的执行情况。
比如虽然整体的CPU使用率是30%,但是有一个线程占据了25%的使用率,一个线程使用率是5%,那么你肯定会想,为啥这个线程需要占用这么高的CPU使用率,是不是代码写的有问题,是不是代码可以优化一下?
当系统运行时,如果你能实时观察各个线程的CPU使用率,那么你就能知道平时这个线程的CPU使用情况是怎样的,为什么后来又高那么多,那么你就可以由此分析出这个线程可能出现了问题,也就可以针对性的进行检查了。
这点对于合作开发的项目更是明显,很多时候因为有些线程的代码不是自己写的,所以根本不知道代码执行情况,一旦系统出现问题,那么可能就是互相甩锅了。
而当计算了线程的CPU使用率,一旦发现某个线程执行异常,那么就能交给负责的人去查看了。
所以说,使用操作系统的项目是非常有必要计算各个线程(任务)的CPU使用率的。
就好比你的电脑,风扇嗡嗡响(CPU高负荷运行),如果只有一个系统CPU使用率,发现高达90%,但是你却根本不知道为什么这么高,所以只能重启。
而一旦有了进程CPU使用率,查看一下哪个进程CPU使用率高,把对应的进程关闭就行了,根本不需要重启电脑。
如何计算线程CPU使用率?
那么现在就来看看该如何计算各个线程的CPU使用率。
从前面的笔记,我们其实也可以猜测该如何计算,无非就是获取每个线程的执行时间 罢了。
比如,1秒时间内,空闲任务执行700毫秒,任务1执行200毫秒,任务2执行100毫秒,那么各个任务的CPU使用率分别是 70%、20%、10%。
以前计算系统的CPU使用率的时候,采用了软件方法计算空闲任务的运行时间,这必然是不够准确的,所以最好的方式是采用硬件计时。
因为鱼鹰采用STM32F103进行测试,所以使用DWT外设进行精确计时,不过麻烦的是,在KEIL 软件仿真 情况下,DWT外设是无法工作的,所以如果要测试的话,必须使用硬件仿真的方式,不过如果真要KEIL软件仿真的话,也不是没有办法,就是使用硬件定时器,这个按下不表。
毕竟,DWT外设的功能在这里说白了也就是个定时器而已。
既然要获取线程的执行时间,关键一点就是,我们要知道操作系统什么时候会切换到某一个线程运行,什么时候又会从这个线程切出,到另一个线程执行呢?
这个关键还是在系统内置的钩子函数 。上次的笔记鱼鹰介绍过空闲钩子函数,今天介绍另一个钩子,任务切换钩子函数。
这个钩子函数的特点就是,每当系统需要切换到下一个任务时,就会先执行 这个函数。这个函数一般有两个参数,当前任务 和即将切换的任务 。
只要设置任务切换的钩子函数,并且有时间戳,那么计算一个任务的执行时间也就不那么困难了。
比如,操作系统在时刻12345 ms 切换到空闲任务执行,突然一个任务就绪,开始准备执行,所以在时刻12445切换到那个就绪任务执行,那么空闲任务的执行时间我们也就可以准确计算出来了。
12445 – 12345 = 100 ms
也就是说,这一次空闲任务执行了 100 毫秒。
如果我们要计算单位时间(比如1秒内)空闲任务的执行时间,我们只要在每次运行到空闲任务时累计 时间即可。
比如1秒内,空闲任务执行了 5 次,分别是 10、200、100、200、50,累计时间为
10 + 200 + 100 + 200 + 50 = 560毫秒
由此,可计算空闲任务的CPU使用率为 56%,从而可计算出系统的CPU使用率是44%。
是的,通过线程的CPU使用率方法,我们其实也可以计算整个系统的CPU使用率。而且这种计算方式比前面所说的计算方法更准确,更科学。
前面采用时间戳进行计算,但是时间戳是会溢出的,那个时候,你的时间计算还是准确的吗?
FreeRTOS线程计算限制?
现在鱼鹰就来说说第三个问题,FreeRTOS线程计算的弊端?如何打破 FreeRTOS 线程计算方式的时间限制?
从网上查找FreeRTOS任务CPU计算相关的资料,可以得到以下信息:
1、需要开一个定时器,这个定时器中断频率是操作系统时钟的十几倍(为了保证计算精度)。
2、一个64 位的变量在定时器自加更新,一旦变量溢出,时间计算就会出现问题。
(相关细节可查看安富莱教程)
第一个问题会导致系统性能下降(中断频率太高,一般是微秒级别的),而第二个问题导致在一段时间内(小时级别)线程CPU使用率计算准确,超出时间后,计算会有问题,所以教程中不建议在正式版本加入此功能。
第一个问题其实很好解决,就是使用硬件定时器,不再由CPU去更新时间,这样不会占用CPU时间,第二个问题其实也非常好解决,就是通过《延时功能进化论 》的方式解决溢出问题,这里不再展开说其中的奥妙。
任务切换钩子函数的实现
总之,鱼鹰接下来的实现方式解决了以上两个痛点,即使无限执行下去,也不会影响到计算精度问题,唯一对系统产生的一点影响,只有在任务切换时消耗的一点计算时间(微秒级别)。
那么先上任务切换钩子函数 关键实现代码(RT-Thread):
如何将这个函数注册到操作系统中被系统调用呢?
通过这个函数即可:
那么现在来分析这个钩子函数实现:
一个静态变量,用于记录切换时的时间戳。
每次任务开始切换时,更新这个时间戳,同时累积时间,这个时间保存在当前任务的user_data里面。
难理解?看下图就清楚了。
假设系统调度是从任务1切换到任务2,即from为任务1 ,to为任务2 ,此时获取的时间戳为 T1。
上一次的时间戳我们已经通过静态变量保留了,这里为T0,那么T1-T0就是from任务即任务1 在本次运行的时间,只要下次运行任务1时继续不断的累积这个时间,那么就可以得到任务1的总运行时间。
任务2同理。
当然我们不可能一直累积下去,不然肯定会溢出,所以隔一段时间就需要清零,这个时间其实就是线程CPU计算的周期 。
这里还有一个函数没有说,就是 get_curr_time(),在这里使用DWT,为了可以重新实现该函数,鱼鹰使用了弱属性 weak(关于这个看参考:《》)。
这里可以看到有个注释,不要使用 rt_tick_get 函数,为啥?
精度太低,有些任务本来执行了的,但是因为执行时间小于 操作系统的时钟(比如1毫秒),那么就无法累积时间了,那么即使这个任务运行再多,时间累积也为 0,这肯定是我们不希望看到的。
然后再说一个点,为了简化代码(钩子函数代码只有短短几行),鱼鹰这样的实现是有两个问题的。
1、首次运行计算有误,因为静态变量应该在运行任务之前就初始化的(不应该初始化为 0),而钩子函数是在任务运行之后才调用的,所以从开机以来的时间被累加到第一个运行任务中了,这肯定是有问题的,不过后面随着系统的运行,静态变量被持续更新,就不会再出现这个问题了。
2、为了减少修改,鱼鹰把线程的use_data当成一个变量使用了,实际上这个变量的功能应该是存储线程私有变量地址的,但是因为鱼鹰懒得修改太多代码,所以直接拿来用了。正因为如此,所以鱼鹰添加线程CPU计算时,只要修改很少的代码就可以了。
线程CPU的计算
目前我们已经能够通过钩子函数获取各个线程的CPU执行时间,现在就看该如何计算了。
为了计算各个线程的CPU使用率,我们需要确定计算周期,这里我们可以设置1秒计算一次。
其次,我们需要确定在哪个任务执行计算。
原理上来说,可以是系统中的任何一个任务,但是为了减少对系统的干扰,可以将计算工作放到优先级比较低的任务中进行,比如空闲任务。
现在,看看函数是如何实现的:
注释已经很详尽了,所以不多做讨论。主要说以下几点:
1、为什么需要关闭调度器,可以使用关中断吗?
关调度器是为了防止在获取各个线程执行时间时,因为系统调度而导致执行时间被更新,从而导致计算有误,所以需要关闭调度器。
那么为什么不使用关中断的方式呢?没有必要。一旦关中断,那么中断就无法响应了,所以在可以关调度器的情况下满足要求,就不应该关中断。
2、为什么分两步计算,为什么不将最终的计算放在第一个循环中执行呢?
节省时间,为了尽量减少关调度器的时间,能省一点是一点。毕竟只要能获取到关键信息,啥时候计算都一样。
3、因为线程CPU计算周期是自动计算的,所以,计算周期其实就是该函数的调用周期,即2秒调用一次,那么线程CPU计算周期就是2秒,但是需要注意的是,调用周期必须小于定时器的溢出时间,即当你使用 DWT 时,调用周期应该在 60 秒以下(72 M 系统时钟),否则计算是有问题的。
现在我们已经算是完成了线程CPU计算问题,但为了使用方便,我们需要把它打印出来,或者把这些信息字符串化:
这里将线程名、线程执行时间、线程使用率 都打印出来了,但是需要注意的是,这里的time 时间单位是定时器的单位,而不是微秒、毫秒,比如如果使用 DWT,那么单位就是 1/72 微秒,即如果 time 值为 1000,那么换算到微秒,应该是 1000/72 秒,当然了,你也可以在打印的同时就把时间换算一下,这个自由发挥就好。
最后,鱼鹰将代码提交到了 RT-Thread 官方工程里面,这是鱼鹰第一次使用Git 提交开源项目(操作不熟练,还是上网搜的教程),也不知道最后合并了没有。
如果对完整代码感兴趣的,也可以在后台回复关键字领取。
喜欢的话,记得关注鱼鹰哦!
相关问答
51 单片机 如何实现多 线程 多任务?有哪些程序可以在“后台”自己运行?1、51单片机无法实现多线程任务,只能实现单线程。这是由51单片机的硬件决定的,否则多线程芯片就没有必要设计出来了。2、51单片机的硬件资源,如:定时器、计...
单片机 技术是一门即将被淘汰的技术吗?感谢邀请答题!目前来说,这十几年内是不可能淘汰的,即使面临着人工AI、大数据和云计算等技术的多方面竞争,单片机也可以说是立于不败之地!毕竟需求放在那里。...
如何在 单片机 装Linux系统?按道理来说有这几年经验,应该已经熟悉了几款单片机,像stm系列或者ti的,这样应该对soc的体系结构比较了解,一般单片机是在Windows下开发,熟悉一些总线,熟悉...
中断服务程序入口地址的形成?要形成中断服务程序入口地址,您需要按照以下步骤操作:1.首先,确定您要中断的服务程序的入口点。这通常是一个函数或方法,用于启动服务程序并处理请求。2.在...
JAVA学会可以干什么?谢谢邀请!作为一名IT从业者,同时也出版过Java编程书籍,所以我来回答一下这个问题。首先,Java语言是当前最为流行的全场景编程语言之一,在Web开发、大数据...比...
Java具体是做什么的?Java的起源Java源自Sun公司的一个叫Green的项目,其初始目的是为家用消费电子产品开发一个分布式代码系统,让人们可以通过这个技术,把E-mail发送给电冰箱、电...
Linux内核开发与Linux驱动开发有什么关系?我做过驱动开发,说说我的看法。本质上说Linux内核开发和Linux驱动开发是不一样的,或者说驱动开发是内核开发的一部分,因为驱动属于内核。目前国内驱动开发和内...
浪潮服务器有自己的CPU吗?服务器同样也是电脑,也同样是有CPU(中央处理器)的,只是服务器电脑不同于家用pc电脑的性能和工作范围,其主要用于科学运算,存储和大量的数据交换,所以服务...
19考研党考研英语书籍有哪些,有对比的吗?考研复习书籍,每年最热门的书籍就那么几本,这里我将详细地介绍一下我的考研经验,也会介绍相关的考研复习书籍!我是3年前考的研了,当时考了380多分,成功跨考...与...
大一学生,软件工程专业,目前学了c语言,数据结构,我想接下来暑假期间,自己该自学一些什么课程?软工专业偏应用多些,尽管与计算机科学与技术教材一样,侧重点是有所不同的。理论紧密结合实际是学科的目标,题主也应紧跟发展潮流。暑假期间学习纯理论的知识...