OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 7244|回复: 12

学过ucos II的麻烦看一下,ucos II cpu使用率计算有几处不理解

[复制链接]

15

主题

181

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
389
金钱
389
注册时间
2014-11-8
在线时间
40 小时
发表于 2015-6-15 16:28:37 | 显示全部楼层 |阅读模式
5金钱

疑问一:while (OSStatRdy == OS_FALSE) {

        OSTimeDly(2u *

OS_TICKS_PER_SEC / 10u);  /* Wait until statistic task is ready 

        只要执行完OSStatInit();就会将OSStatRdy 变为OS_TRUE, 也就是说这句话在这之后只要一运行while

(OSStatRdy == OS_FALSE)就相当于while (0),不知道自己的理解对不对?

 

疑问二:OSIdleCtrMax/= 100uL;只要执行OS_TaskStat就会执行OSIdleCtrMax/= 100uL; OSIdleCtrMax0.1s     执行OSIdleCtrMax/=

100uL,不就是为了得到1ms0.1s/100)空闲任务计数的最大值吗?为什么要每次运行程序都要执行一遍

 

疑问三:OS_TaskStat走到(void)OSTaskSuspend(OS_PRIO_SELF);函数挂起自己什么时候被唤醒?OSIdleCtrRun100个滴答时钟的空闲值吗? 



下面是ucos II的OS_TaskStat函数的源码:

#if OS_TASK_STAT_EN > 0u

void  OS_TaskStat (void *p_arg)

{

#if OS_CRITICAL_METHOD == 3u                     /* Allocate storage for CPU status register           */

    OS_CPU_SR  cpu_sr = 0u;

#endif

 

 

 

    p_arg = p_arg;                               /* Prevent compiler warning for not using 'p_arg'     */

    while (OSStatRdy == OS_FALSE) {

        OSTimeDly(2u * OS_TICKS_PER_SEC / 10u);  /* Wait until statistic task is ready                 */

    }

    OSIdleCtrMax /= 100uL;//1ms的计数最大值

    if (OSIdleCtrMax == 0uL) {

        OSCPUUsage = 0u;

#if OS_TASK_SUSPEND_EN > 0u

        (void)OSTaskSuspend(OS_PRIO_SELF);

#else

        for (;;) {

            OSTimeDly(OS_TICKS_PER_SEC);//OS_TICKS_PER_SEC:1s的滴答数(1000ms/5ms = 200

        }

#endif

    }

    for (;;) {

        OS_ENTER_CRITICAL();

                                   //OSIdleCtrRun 为100个滴答时钟的空闲计数器值。(0.5s

        OSIdleCtrRun = OSIdleCtr;                /* 获取过去的一秒空闲计数器的值 */

        OSIdleCtr    = 0uL;                      /* 重置下一秒空闲计数器         */

        OS_EXIT_CRITICAL();

        OSCPUUsage   = (INT8U)(100uL - OSIdleCtrRun / OSIdleCtrMax);

        OSTaskStatHook();                        /* 调用用户自定义的钩子函数                */

#if (OS_TASK_STAT_STK_CHK_EN > 0u) && (OS_TASK_CREATE_EXT_EN > 0u)

        OS_TaskStatStkChk();                     /* 检查每个任务的堆栈                     */

#endif

        OSTimeDly(OS_TICKS_PER_SEC / 10u);       /* 在接下来的0.1s积累osidlectr      */

    }

}

#endif

 

 代码的中文备注是我自己写的,不一定对。 0.0

 

求各路大神指教。。。





Good good study , day day up...
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

70

主题

6756

帖子

0

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
12975
金钱
12975
注册时间
2012-11-26
在线时间
3786 小时
发表于 2015-6-15 17:03:21 | 显示全部楼层
学无止境
回复

使用道具 举报

15

主题

181

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
389
金钱
389
注册时间
2014-11-8
在线时间
40 小时
 楼主| 发表于 2015-6-15 17:18:13 | 显示全部楼层
回复【2楼】jermy_z:
---------------------------------
谢谢您,不过cpu使用率这个不难,知识很浅,只不过我是初学。。。
Good good study , day day up...
回复

使用道具 举报

15

主题

181

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
389
金钱
389
注册时间
2014-11-8
在线时间
40 小时
 楼主| 发表于 2015-6-16 13:07:10 | 显示全部楼层
自己顶,希望原子哥帮帮忙。。。
Good good study , day day up...
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165371
金钱
165371
注册时间
2010-12-1
在线时间
2110 小时
发表于 2015-6-16 23:04:14 | 显示全部楼层
回复【4楼】Just Beat It:
---------------------------------
我也没有研究,呵呵
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

15

主题

181

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
389
金钱
389
注册时间
2014-11-8
在线时间
40 小时
 楼主| 发表于 2015-6-17 00:06:45 | 显示全部楼层
回复【5楼】正点原子:
---------------------------------
好吧。。。
Good good study , day day up...
回复

使用道具 举报

0

主题

9

帖子

0

精华

新手上路

积分
36
金钱
36
注册时间
2015-4-13
在线时间
0 小时
发表于 2015-6-18 11:23:53 | 显示全部楼层
第一个是只调用一次。第二个也是只计算一次,以后就一直在for循环里了,只有当max>0不满足时才再次计算。第三个如果挂起那应该是需要其它任务来让它就绪,要不然自己是不可能让自己再次就绪的。这是我的理解,希望能对楼主有帮助。
回复

使用道具 举报

0

主题

9

帖子

0

精华

新手上路

积分
36
金钱
36
注册时间
2015-4-13
在线时间
0 小时
发表于 2015-6-18 12:15:51 | 显示全部楼层
不好意思啊,这里我说的有误,刚刚没仔细看你的程序,你的这个max是在主for循环外面就计算好了,所以后续的就不会再计算,除非是把这个任务删除后重新加入进来才会再次计算,我的这个版本是在外做自此计算,主for循环里做检测,当max>0不满足就再计算,一般是不会出现max不满足条件的。
回复

使用道具 举报

15

主题

181

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
389
金钱
389
注册时间
2014-11-8
在线时间
40 小时
 楼主| 发表于 2015-6-19 13:01:19 | 显示全部楼层
回复【8楼】asciiy:
---------------------------------
十分感谢您的回答。但我还有些疑问想向您请教:
①#define OS_TASK_SUSPEND_EN        1u 根据下面蓝色的代码编译器会编译(void)OSTaskSuspend(OS_PRIO_SELF);那就是说函数OS_TaskStat自己挂起了自己,那么他什么时候被唤醒?
#if OS_TASK_SUSPEND_EN > 0u
        (void)OSTaskSuspend(OS_PRIO_SELF);
#else
        for (;;) {
            OSTimeDly(OS_TICKS_PER_SEC);//OS_TICKS_PER_SEC:1s的滴答数(1000ms/5ms = 200)
        }
#endif

②也就是说 OSIdleCtrMax/= 100uL;只会执行一遍,所以说 OSIdleCtrMax经过 OSIdleCtrMax/= 100uL;这个函数计算的值就是1ms(0.1s/100)所得空闲任务计数的最大值。
这是计算CPU使用率的语句OSCPUUsage   = (INT8U)(100uL - OSIdleCtrRun / OSIdleCtrMax);   分母是1ms所得空闲任务计数的最大值,那么分子就应该是0.1s(1ms*100)空闲任务所记的值,但是没发现任务是0.1s被唤醒的啊?
Good good study , day day up...
回复

使用道具 举报

0

主题

3

帖子

0

精华

新手上路

积分
25
金钱
25
注册时间
2015-1-17
在线时间
0 小时
发表于 2015-6-20 22:45:59 | 显示全部楼层
帮顶。。。。
回复

使用道具 举报

0

主题

9

帖子

0

精华

新手上路

积分
36
金钱
36
注册时间
2015-4-13
在线时间
0 小时
发表于 2015-6-22 23:31:08 | 显示全部楼层
首先说明下,你的这个版本和我的不太一样,我的这个没有挂起的宏定义,而且这里如果进入到这个if里面,那么也是说明这个任务初始化是有问题的,因为max == 0,为啥这里会加这个检测我也不是很清楚。假如进入到这里面了,如果你不改变宏定义那么是肯定需要使用取消挂起函数进行激活才能使用的,你可以用板子测试一下,在使用挂起的情况下是不会执行到这个任务的。
第二个如果任务没有被挂起,那么就会一直处于for循环内,这里就有一个OSTimeDly函数,在时间片到你设定的时间时就会再次把这个任务置于就绪列表中,于是如果没有更高优先级的任务时就会调用这个任务了,这里注意这里的是指0.1ms唤醒,并不是调用。
回复

使用道具 举报

0

主题

9

帖子

0

精华

新手上路

积分
36
金钱
36
注册时间
2015-4-13
在线时间
0 小时
发表于 2015-6-22 23:34:36 | 显示全部楼层
至于为啥会调用,你直接去看void SysTick_Handler(void)
{
    OSIntEnter(); 
    OSTimeTick(); 
    OSIntExit(); 
}里面的OSTimeTick(); 你就明白了,它会遍历任务控制块链表的tick,并每次进行减1操作,到任务控制块的tick为零时,这个任务就会被置于就绪列表中,这样就等待优先级到自己的时候就调用了。
回复

使用道具 举报

22

主题

2251

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
4471
金钱
4471
注册时间
2013-4-22
在线时间
335 小时
发表于 2015-6-23 09:17:31 | 显示全部楼层
我觉得这就是定时而已,做时钟和系统时钟用,没有其他的
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-2-23 17:10

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表