FreeRTOS学习笔记(11)——CPU使用率统计

   日期:2021-01-06     浏览:232    评论:0    
核心提示:一、基本概念CPU 使用率其实就是系统运行的程序占用的 CPU 资源,表示机器在某段时间程序运行的情况,如果这段时间中,程序一直在占用 CPU 的使用权,那么可以人为 CPU 的利用率是 100%。CPU 的利用率越高,说明机器在这个时间上运行了很多程序,反之较少。利用率的高低与 CPU 强弱有直接关系,就像一段一模一样的程序,如果使用运算速度很慢的 CPU,它可能要运行 1000ms,而使用很运算速度很快的 CPU 可能只需要 10ms,那么在 1000ms 这段时间中,前者的 CPU 利用率就是 1

一、基本概念

CPU 使用率其实就是系统运行的程序占用的 CPU 资源,表示机器在某段时间程序运行的情况,如果这段时间中,程序一直在占用 CPU 的使用权,那么可以人为 CPU 的利用率是 100%。CPU 的利用率越高,说明机器在这个时间上运行了很多程序,反之较少。利用率的高低与 CPU 强弱有直接关系,就像一段一模一样的程序,如果使用运算速度很慢的 CPU,它可能要运行 1000ms,而使用很运算速度很快的 CPU 可能只需要 10ms,那么在 1000ms 这段时间中,前者的 CPU 利用率就是 100%,而后者的 CPU 利用率只有 1%,因为 1000ms 内前者都在使用 CPU 做运算,而后者只使用 10ms 的时间做运算,剩下的时间 CPU 可以做其他事情。

FreeRTOS 是多任务操作系统,对 CPU 都是分时使用的:比如 A 任务占用 10ms,然后 B 任务占用 30ms,然后空闲 60ms,再又是 A 任务占 10ms,B 任务占 30ms,空闲 60ms;

二、CPU利用率统计

在调试的时候很有必要得到当前系统的 CPU 利用率相关信息,但是在产品发布的时候,就可以把 CPU 利用率统计这个功能去掉,因为使用任何功能的时候,都是需要消耗系统资源的,FreeRTOS 是使用一个外部的变量进行统计时间的,并且消耗一个高精度的定时器,其用于定时的精度是系统时钟节拍的 10-20 倍,比如当前系统时钟节拍是 1000HZ,那么定时器的计数节拍就要是 10000-20000HZ。而且 FreeRTOS 进行 CPU 利用率统计的时候,也有一定缺陷,因为它没有对进行 CPU 利用率统计时间的变量做溢出保护,我们使用的是 32 位变量来系统运行的时间计数值,而按 20000HZ 的中断频率计算,每进入一中断就是 50us,变量加一,最大支持计数时间:2^32 * 50us / 3600s = 59.6 分钟,运行时间超过了 59.6 分钟后统计的结果将不准确,除此之外整个系统一直响应定时器 50us 一次的中断会比较影响系统的性能。

三、配置选项

FreeRTOSConfig.h 配置与系统运行时间和任务状态收集有关的配置选项,并且实现
portCONFIGURE_TIMER_FOR_RUN_TIME_STATS()portGET_RUN_TIME_COUNTER_VALUE() 这两个宏定义


//启用运行时间统计功能 
#define configGENERATE_RUN_TIME_STATS 1 
//启用可视化跟踪调试 
#define configUSE_TRACE_FACILITY 1 

#define configUSE_STATS_FORMATTING_FUNCTIONS 1

extern volatile uint32_t CPU_RunTime; 

#define portCONFIGURE_TIMER_FOR_RUN_TIME_STATS() (CPU_RunTime = 0ul) 
#define portGET_RUN_TIME_COUNTER_VALUE() CPU_RunTime

然后需要实现一个中断频率为 20000HZ 定时器,用于系统运行时间统计,其实很简单,只需将 CPU_RunTime 变量自加即可,这个变量是用于记录系统运行时间的,中断服务函数见下


volatile uint32_t CPU_RunTime = 0UL;

void BASIC_TIM_IRQHandler (void)
{ 
    if(TIM_GetITStatus( BASIC_TIM, TIM_IT_Update) != RESET ) 
    { 
        CPU_RunTime++; 
        TIM_ClearITPendingBit(BASIC_TIM , TIM_FLAG_Update);
    }
}

然后我们就可以在任务中调用 vTaskGetRunTimeStats()vTaskList() 函数获得任务的相关信息与 CPU 使用率的相关信息,然后打印出来即可

memset(CPU_RunInfo,0,400); //信息缓冲区清零

vTaskList((char *)&CPU_RunInfo); //获取任务运行时间信息 

printf("---------------------------------------------\r\n");
printf("任务名 任务状态 优先级 剩余栈 任务序号\r\n");
printf("%s", CPU_RunInfo);
printf("---------------------------------------------\r\n");
 
memset(CPU_RunInfo,0,400); //信息缓冲区清零

vTaskGetRunTimeStats((char *)&CPU_RunInfo); 
 
printf("任务名 运行计数 使用率\r\n");
printf("%s", CPU_RunInfo);
printf("---------------------------------------------\r\n\n");

四、示例


#include "FreeRTOS.h"
#include "task.h"

#include "bsp_led.h"
#include "bsp_usart.h"
#include "bsp_TiMbase.h"
#include "string.h"



 
static TaskHandle_t AppTaskCreate_Handle = NULL;

static TaskHandle_t LED1_Task_Handle = NULL;
static TaskHandle_t LED2_Task_Handle = NULL;
static TaskHandle_t CPU_Task_Handle = NULL;


static void AppTaskCreate(void);

static void LED1_Task(void* pvParameters);
static void LED2_Task(void* pvParameters);
static void CPU_Task(void* pvParameters);
static void BSP_Init(void);


int main(void)
{ 	
    BaseType_t xReturn = pdPASS;

    
    BSP_Init();
    
    xReturn = xTaskCreate((TaskFunction_t )AppTaskCreate,  
                          (const char*    )"AppTaskCreate",
                          (uint16_t       )512,  
                          (void*          )NULL,
                          (UBaseType_t    )1, 
                          (TaskHandle_t*  )&AppTaskCreate_Handle); 
               
    if(pdPASS == xReturn)
    { 
        vTaskStartScheduler();   
    }
    else
    { 
        return -1;  
    }
  
    while(1);       
}


static void AppTaskCreate(void)
{ 
    BaseType_t xReturn = pdPASS;
  
    taskENTER_CRITICAL();           //进入临界区
  
    
    xReturn = xTaskCreate((TaskFunction_t )LED1_Task, 
                          (const char*    )"LED1_Task",
                          (uint16_t       )512,   
                          (void*          )NULL,	
                          (UBaseType_t    )2,	    
                          (TaskHandle_t*  )&LED1_Task_Handle);
    if(pdPASS == xReturn)
    { 
        printf("创建LED1_Task任务成功!\r\n");
    }
  
    
    xReturn = xTaskCreate((TaskFunction_t )LED2_Task, 
                          (const char*    )"LED2_Task",
                          (uint16_t       )512,   
                          (void*          )NULL,	
                          (UBaseType_t    )3,	    
                          (TaskHandle_t*  )&LED2_Task_Handle);
    if(pdPASS == xReturn)
    { 
        printf("创建LED2_Task任务成功!\r\n");
    }

    
    xReturn = xTaskCreate((TaskFunction_t )CPU_Task, 
                          (const char*    )"CPU_Task",
                          (uint16_t       )512,   
                          (void*          )NULL,	
                          (UBaseType_t    )4,	    
                          (TaskHandle_t*  )&CPU_Task_Handle);
    if(pdPASS == xReturn)
    { 
        printf("创建CPU_Task任务成功!\r\n");
    }
  
    vTaskDelete(AppTaskCreate_Handle); //删除AppTaskCreate任务
  
    taskEXIT_CRITICAL();            //退出临界区
}


static void LED1_Task(void* parameter)
{ 	
    while (1)
    { 
        LED1_ON;
        vTaskDelay(500);   
        printf("LED1_Task Running,LED1_ON\r\n");
        LED1_OFF;     
        vTaskDelay(500);   		 		
        printf("LED1_Task Running,LED1_OFF\r\n");
    }
}

static void LED2_Task(void* parameter)
{ 	
    while (1)
    { 
        LED2_ON;
        vTaskDelay(300);   
        printf("LED2_Task Running,LED2_ON\r\n");
    
        LED2_OFF;     
        vTaskDelay(300);   		 		
        printf("LED2_Task Running,LED2_OFF\r\n");
    }
}

static void CPU_Task(void* parameter)
{ 	
    uint8_t CPU_RunInfo[400];		//保存任务运行时间信息
  
    while (1)
    { 
        memset(CPU_RunInfo,0,400);				//信息缓冲区清零
    
        vTaskList((char *)&CPU_RunInfo);  //获取任务运行时间信息
    
        printf("---------------------------------------------\r\n");
        printf("任务名 任务状态 优先级 剩余栈 任务序号\r\n");
        printf("%s", CPU_RunInfo);
        printf("---------------------------------------------\r\n");
    
        memset(CPU_RunInfo,0,400);				//信息缓冲区清零
    
        vTaskGetRunTimeStats((char *)&CPU_RunInfo);
    
        printf("任务名 运行计数 利用率\r\n");
        printf("%s", CPU_RunInfo);
        printf("---------------------------------------------\r\n\n");
        vTaskDelay(1000);   		
    }
}


static void BSP_Init(void)
{ 
	
	NVIC_PriorityGroupConfig( NVIC_PriorityGroup_4 );
	
	
	LED_GPIO_Config();

	
	USART_Config();
  
    
	BASIC_TIM_Init();
}

• 由 Leung 写于 2021 年 1 月 5 日

• 参考:野火FreeRTOS视频与PDF教程

 
打赏
 本文转载自:网络 
所有权利归属于原作者,如文章来源标示错误或侵犯了您的权利请联系微信13520258486
更多>最近资讯中心
更多>最新资讯中心
0相关评论

推荐图文
推荐资讯中心
点击排行
最新信息
新手指南
采购商服务
供应商服务
交易安全
关注我们
手机网站:
新浪微博:
微信关注:

13520258486

周一至周五 9:00-18:00
(其他时间联系在线客服)

24小时在线客服