当前位置:首页 > 公众号精选 > 嵌入式客栈
[导读]来源:公众号【鱼鹰谈单片机】作者:鱼鹰OspreyID  :emOsprey这篇笔记有如下内容:1、为什么需要计算各个线程的CPU使用率?2、该如何计算线程CPU使用率?3、FreeRTOS线程计算的弊端?如何打破FreeRTOS线程计算方式的时间限制?4、关键代码介绍。上次介绍...

来源:公众号【鱼鹰谈单片机】
作者:鱼鹰Osprey
ID   :emOsprey
这篇笔记有如下内容:
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):
void thread_stats_scheduler_hook(struct rt_thread *from, struct rt_thread *to){ static uint32_t schedule_last_time; uint32_t time; time = get_curr_time(); from->user_data = (time - schedule_last_time); schedule_last_time = time;}如何将这个函数注册到操作系统中被系统调用呢?通过这个函数即可:那么现在来分析这个钩子函数实现:一个静态变量,用于记录切换时的时间戳。每次任务开始切换时,更新这个时间戳,同时累积时间,这个时间保存在当前任务的user_data里面。难理解?看下图就清楚了。假设系统调度是从任务1切换到任务2,即from为任务1,to为任务2,此时获取的时间戳为 T1上一次的时间戳我们已经通过静态变量保留了,这里为T0,那么T1-T0就是from任务即任务1在本次运行的时间,只要下次运行任务1时继续不断的累积这个时间,那么就可以得到任务1的总运行时间。任务2同理。当然我们不可能一直累积下去,不然肯定会溢出,所以隔一段时间就需要清零,这个时间其实就是线程CPU计算的周期。    这里还有一个函数没有说,就是 get_curr_time(),在这里使用DWT,为了可以重新实现该函数,鱼鹰使用了弱属性 weak(关于这个看参考:《困惑多年,为什么 printf 可以重定向? 》)。
__weakuint32_t get_curr_time() { return DWT->CYCCNT; // don't use the function rt_tick_get()}这里可以看到有个注释,不要使用 rt_tick_get 函数,为啥?精度太低,有些任务本来执行了的,但是因为执行时间小于操作系统的时钟(比如1毫秒),那么就无法累积时间了,那么即使这个任务运行再多,时间累积也为 0,这肯定是我们不希望看到的。然后再说一个点,为了简化代码(钩子函数代码只有短短几行),鱼鹰这样的实现是有两个问题的。1、首次运行计算有误,因为静态变量应该在运行任务之前就初始化的(不应该初始化为 0),而钩子函数是在任务运行之后才调用的,所以从开机以来的时间被累加到第一个运行任务中了,这肯定是有问题的,不过后面随着系统的运行,静态变量被持续更新,就不会再出现这个问题了。2、为了减少修改,鱼鹰把线程的use_data当成一个变量使用了,实际上这个变量的功能应该是存储线程私有变量地址的,但是因为鱼鹰懒得修改太多代码,所以直接拿来用了。正因为如此,所以鱼鹰添加线程CPU计算时,只要修改很少的代码就可以了。
线程CPU计算
目前我们已经能够通过钩子函数获取各个线程的CPU执行时间,现在就看该如何计算了。为了计算各个线程的CPU使用率,我们需要确定计算周期,这里我们可以设置1秒计算一次。其次,我们需要确定在哪个任务执行计算。原理上来说,可以是系统中的任何一个任务,但是为了减少对系统的干扰,可以将计算工作放到优先级比较低的任务中进行,比如空闲任务。现在,看看函数是如何实现的:
// can call the function 1 s (max 60s when stm32f1xx because of dwt)void thread_cal_usage(thread_run_info_def *run_info){ static uint32_t total_time_last; uint32_t time, total_time; struct rt_list_node *node; struct rt_list_node *list; struct rt_thread *thread; uint32_t i; rt_enter_critical(); // 关闭系统调度,防止在计算过程中更新线程时间,影响计算 time = get_curr_time(); // 获取当前时间戳 total_time = time - total_time_last; // 计算运行总时间 total_time_last = time; // 更新时间 list =
本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭