05_FreeRTOS相对延时与绝对延时

硬件平台:

  • STM32F103

相关配置:

1. FreeRTOS延时函数

延时函数:

  • vTaskDelay:相对延时,从延时函数调用开始,任务进入阻塞状态,直到指定延时时间完毕。
  • vTaskDelayUntil:绝对延时,计算整个任务的执行时间,最后用指定延时时间减去任务当前的执行时间即为任务所需的延时。

![[Pasted image 20250108145450.png]]

1.1 相对延时 - vTaskDelay

函数说明:

  • 注意实现: INCLUDE_vTaskDelay设置为1
  • 参数1: 指定延时的时间,单位为滴答周期(STM32所一直的FreeRTOS滴答周期为1ms)
void vTaskDelay( const TickType_t xTicksToDelay );

1.2 绝对延时 - vTaskDelayUntil

函数说明:

  • 注意实现: INCLUDE_vTaskDelayUntil设置为1
  • 参数1: 当前滴答计数的值的指针
  • 参数2: 指定延时的时间,单位为滴答周期(STM32所一直的FreeRTOS滴答周期为1ms)
void vTaskDelayUntil( TickType_t *pxPreviousWakeTime,
                      const TickType_t xTimeIncrement );

获取滴答计数值函数:

  • 返回值: 返回当前滴答计数的值
  • 注意实现: 无法从中断中获取
volatile TickType_t xTaskGetTickCount( void );

2. 相对延时与绝对延时示例

示例流程:

  1. 系统启动函数:创建一个启动任务
  2. 启动任务执行函数
    1. 创建任务1
    2. 创建任务2
  3. 任务1:
    1. 串口打印Task1
    2. CPU延时200ms
    3. 相对延时1s
  4. 任务2:
    1. 获取当前计数周期的值
    2. 打印Task2
    3. CPU延时200ms
    4. 绝对延时1s

2.1 系统启动函数

函数说明:

  • 动态创建一个启动任务
  • 启动任务调度器
/* 启动任务相关配置 */
void startTask(void *args);                                 /* 启动任务执行函数 */
#define START_TASK_NAME "start"                             /* 启动任务的名称 */
#define START_TASK_STACK_DEPTH 128                          /* 启动任务栈大小 = 2 * 128 字节 */
#define START_TASK_PRIORITY 4                               /* 启动任务优先级 */
TaskHandle_t startTaskTCB;                                  /* 启动任务的控制块 */

void FreeRTOS_Start(void)
{

    /* 1. 创建启动任务 */
    xTaskCreate(
        startTask,
        START_TASK_NAME,
        START_TASK_STACK_DEPTH,
        NULL,
        START_TASK_PRIORITY,
        &startTaskTCB
    );

    /* 2. 启动任务调度器 */
    vTaskStartScheduler();
}

2.2 启动任务执行函数

函数说明:

  • 创建任务1与任务2
  • 回收起始任务
/* 2个子任务相关配置 */
/* --------------------Task1-------------------- */
void taskFunc1(void *args);                                 /* 任务1执行函数 */
#define TASK_NAME_1 "task1"                                 /* 任务1的名称 */
#define TASK1_PRIORITY 4                                    /* 任务1优先级 */
TaskHandle_t task1TCB;                                      /* 任务1的控制块 */
/* --------------------Task2-------------------- */
void taskFunc2(void *args);                                 /* 任务2执行函数 */
#define TASK_NAME_2 "task2"                                 /* 任务2的名称 */
#define TASK2_PRIORITY 3                                    /* 任务2优先级 */
TaskHandle_t task2TCB;                                      /* 任务2的控制块 */

void startTask(void *args)
{
    // 1. 代码进入临界区,防止代码中断
    vPortEnterCritical();      
    debug_printfln("执行启动任务...");         

    // 2. 创建3个任务
    xTaskCreate(                                        /* 任务1 */
        taskFunc1,
        TASK_NAME_1,
        START_TASK_STACK_DEPTH,
        NULL,
        TASK1_PRIORITY,
        &task1TCB
    );
    xTaskCreate(                                        /* 任务2 */
        taskFunc2,
        TASK_NAME_2,
        START_TASK_STACK_DEPTH,
        NULL,
        TASK2_PRIORITY,
        &task2TCB
    );

    
    // 3. 代码退出临界区
    debug_printfln("启动任务执行完毕...\n回收启动任务...");
    vPortExitCritical();

    // 4. 回收启动任务
    vTaskDelete(NULL);
}

2.3 任务1

函数说明

  1. 串口打印Task1
  2. CPU延时200ms
  3. 相对延时1s
void taskFunc1(void *args)                                  /* 任务1执行函数 */
{
    while (1)
    {
        debug_printfln("Task1...");
        Delay_ms(200);
        vTaskDelay(1000);
    }
}

2.4 任务2

函数说明:

  1. 获取当前计数周期的值
  2. 打印Task2
  3. CPU延时200ms
  4. 绝对延时1s
void taskFunc2(void *args)                                  /* 任务2执行函数 */
{
    TickType_t tickCount = xTaskGetTickCount();
    while (1)
    {
        debug_printfln("Task2...");
        Delay_ms(200);
        vTaskDelayUntil(&tickCount, 1000);
    }
    
}

执行效果:

  • Task2进入阻塞的时间大致为1000-200
  • Task1进入阻塞的时间为1000

![[Pasted image 20250108151452.png]]

文章声明:
本文章为作者学习FreeRTOS实时操作系统的笔记与巩固输出,欢迎同行交流与指教,文章借鉴来源小破站的某谷RTOS教程。

FreeRTOSDemo.h

#ifndef __FREERTOS_DEMO_H
#define __FREERTOS_DEMO_H

/* 任务相关 */
#include "FreeRTOS.h"
#include "task.h"

#include "Common_Debug.h"
#include "Common_Delay.h"
#include "Inf_Key.h"


/* 1. 启动操作系统 */
void FreeRTOS_Start(void);

#endif


FreeRTOSDemo.c

#include "FreeRTOS_Demo.h"

/* 启动任务相关配置 */
void startTask(void *args);                                 /* 启动任务执行函数 */
#define START_TASK_NAME "start"                             /* 启动任务的名称 */
#define START_TASK_STACK_DEPTH 128                          /* 启动任务栈大小 = 2 * 128 字节 */
#define START_TASK_PRIORITY 4                               /* 启动任务优先级 */
TaskHandle_t startTaskTCB;                                  /* 启动任务的控制块 */

/* 2个子任务相关配置 */
/* --------------------Task1-------------------- */
void taskFunc1(void *args);                                 /* 任务1执行函数 */
#define TASK_NAME_1 "task1"                                 /* 任务1的名称 */
#define TASK1_PRIORITY 4                                    /* 任务1优先级 */
TaskHandle_t task1TCB;                                      /* 任务1的控制块 */
/* --------------------Task2-------------------- */
void taskFunc2(void *args);                                 /* 任务2执行函数 */
#define TASK_NAME_2 "task2"                                 /* 任务2的名称 */
#define TASK2_PRIORITY 3                                    /* 任务2优先级 */
TaskHandle_t task2TCB;                                      /* 任务2的控制块 */


void FreeRTOS_Start(void)
{

    /* 1. 创建启动任务 */
    xTaskCreate(
        startTask,
        START_TASK_NAME,
        START_TASK_STACK_DEPTH,
        NULL,
        START_TASK_PRIORITY,
        &startTaskTCB
    );

    /* 2. 启动任务调度器 */
    vTaskStartScheduler();
}

void startTask(void *args)
{
    // 1. 代码进入临界区,防止代码中断
    vPortEnterCritical();      
    debug_printfln("执行启动任务...");         

    // 2. 创建3个任务
    xTaskCreate(                                        /* 任务1 */
        taskFunc1,
        TASK_NAME_1,
        START_TASK_STACK_DEPTH,
        NULL,
        TASK1_PRIORITY,
        &task1TCB
    );
    xTaskCreate(                                        /* 任务2 */
        taskFunc2,
        TASK_NAME_2,
        START_TASK_STACK_DEPTH,
        NULL,
        TASK2_PRIORITY,
        &task2TCB
    );

    
    // 3. 代码退出临界区
    debug_printfln("启动任务执行完毕...\n回收启动任务...");
    vPortExitCritical();

    // 4. 回收启动任务
    vTaskDelete(NULL);
}

void taskFunc1(void *args)                                  /* 任务1执行函数 */
{
    while (1)
    {
        debug_printfln("Task1...");
        Delay_ms(200);
        vTaskDelay(1000);
    }
}

void taskFunc2(void *args)                                  /* 任务2执行函数 */
{
    TickType_t tickCount = xTaskGetTickCount();
    while (1)
    {
        debug_printfln("Task2...");
        Delay_ms(200);
        vTaskDelayUntil(&tickCount, 1000);
    }
    
}


extern void xPortSysTickHandler(void);
void        SysTick_Handler(void)
{
    if(xTaskGetSchedulerState() != taskSCHEDULER_NOT_STARTED)
    {
        xPortSysTickHandler();
    }
}

<think>好的,用户想了解FreeRTOS中的相对延时绝对延时的概念及实现机制。首先,我需要回忆一下FreeRTOS的相关知识,确保自己理解正确。 根据引用内容,相对延时是使用vTaskDelay函数,基于当前时间加上延时时间。而绝对延时使用vTaskDelayUntil,基于上一次唤醒时间点,确保固定间隔。用户可能想知道它们的具体区别和内部实现。 首先,我应该明确两者的定义和区别。相对延时是从调用函数的时间点开始算,容易受到任务执行时间和中断的影响,导致间隔不稳定。绝对延时则是根据上一次的结束时间来计算,保持固定周期,适合周期性任务。 接下来是实现机制部分。FreeRTOS内部使用系统节拍计数器xTickCount来跟踪时间。相对延时计算的是当前节拍数加上延时时间,而绝对延时则维护了一个变量记录上一次的唤醒时间,每次加上周期时间,确保即使任务执行时间有波动,周期仍然稳定。 用户可能的应用场景是实时任务调度,比如需要精确控制频率的数据采集或通信协议处理。这里需要举例说明两者的适用情况,帮助用户理解何时使用哪种延时方式。 另外,需要注意任务执行时间不能超过延时周期,否则绝对延时也无法保证稳定性。这一点在引用3中提到过,如果A_A代码执行50ms,绝对延时会调整剩余延时时间为50ms,保持总间隔100ms。而相对延时则可能因为执行时间导致间隔拉长。 最后,总结两者的对比,强调相对延时的灵活性和绝对延时的精确性。可能需要提醒用户根据具体需求选择合适的方法,比如周期性任务优先使用绝对延时,非周期性或不确定执行时间的任务使用相对延时。 检查是否覆盖了用户的问题:概念、实现机制、区别、应用场景。是否有遗漏?可能需要补充代码示例,但用户的问题中没有要求代码,所以可能不需要。不过之前的示例中有代码块,但根据系统指令,用户不可见部分可能有相关要求。需要按照用户的指令,回答结构清晰,分点说明,引用正确。 确认引用的使用,用户提供了三个引用,需要正确标注。例如,相对延时的定义引用[^1],绝对延时的解决长耗时问题引用[^2],执行时间影响的例子引用[^3]。 现在组织内容,先介绍两种延时,再分别说明实现机制,对比区别,最后应用场景。确保每个部分都有引用支持,并且自然标注。 可能的问题:如何保证绝对延时的准确性?这涉及到系统节拍的精度和任务调度机制。需要提到FreeRTOS的任务调度器如何管理延时列表,以及如何计算唤醒时间。 还要注意用户可能混淆两种延时的使用场景,需要明确举例说明,比如数据采样每100ms用绝对延时,而等待资源可用用相对延时。 总结时,强调选择延时方法的依据,帮助用户做出正确决策。</think>### FreeRTOS相对延时绝对延时的实现机制分析 #### 一、核心概念对比 1. **相对延时 (vTaskDelay)** - 以**调用时刻为起点**计算延时时间,例如调用$vTaskDelay(100)$表示从当前时间开始等待100个系统节拍[^1]。 - 实际间隔时间 = 任务执行时间 + 指定延时时间 - 特点:延时起点动态变化,适合非周期性场景 2. **绝对延时 (vTaskDelayUntil)** - 以**预定时间点为起点**计算延时时间,例如调用$vTaskDelayUntil(\&prev\_wake\_time, 100)$表示在上次唤醒时间基础上增加100个节拍[^3]。 - 实际间隔时间 = Max(指定周期时间, 任务执行时间) - 特点:维护固定周期,适合严格周期性任务 #### 二、实现机制解析 1. **相对延时实现** ```c void vTaskDelay(const TickType_t xTicksToDelay) { prvAddCurrentTaskToDelayedList(xTicksToDelay, pdFALSE); } ``` - 基于当前系统节拍计数器$xTickCount$,计算目标时间点$xTimeToWake = xTickCount + xTicksToDelay$ - 将任务移出就绪队列,加入**延时等待列表**(按时间排序的链表) - 系统节拍中断触发时,检查并唤醒到期任务 2. **绝对延时实现** ```c void vTaskDelayUntil(TickType_t *const pxPreviousWakeTime, const TickType_t xTimeIncrement) { TickType_t xTimeToWake = *pxPreviousWakeTime + xTimeIncrement; TickType_t xAlreadyDelayed = xTaskGetTickCount() - *pxPreviousWakeTime; if(xAlreadyDelayed >= xTimeIncrement) { // 处理超时补偿 xTimeToWake += xAlreadyDelayed / xTimeIncrement * xTimeIncrement; } prvDelayUntil(&xTimeToWake); // 专用延时函数 *pxPreviousWakeTime = xTimeToWake; } ``` - 维护持续更新的$pxPreviousWakeTime$变量记录上次唤醒时间 - 采用**时间补偿机制**处理任务执行超时情况[^3] - 任务唤醒后立即更新下次唤醒时间点 #### 三、关键差异对比表 | 特性 | 相对延时(vTaskDelay) | 绝对延时(vTaskDelayUntil) | |---------------------|-------------------------|--------------------------| | **时间基准** | 当前时刻 | 预定时间点 | | **周期稳定性** | 受任务执行时间影响 | 保持严格周期[^3] | | **适用场景** | 非周期性等待 | 周期性任务(如ADC采样) | | **时间补偿** | 无 | 自动补偿超时 | | **API参数** | 单参数(延时量) | 双参数(基准时间指针,周期)| #### 四、应用场景示例 1. **相对延时适用场景** 等待外部事件触发(如按键检测): ```c while(1) { if(xSemaphoreTake(btn_signal, 100/portTICK_PERIOD_MS) == pdTRUE) { process_button(); // 事件触发处理 } vTaskDelay(10); // 非精确间隔的轮询 } ``` 2. **绝对延时适用场景** 精确的100Hz数据采集系统: ```c TickType_t xLastWakeTime = xTaskGetTickCount(); while(1) { adc_sample(); // 耗时5ms的采样操作 vTaskDelayUntil(&xLastWakeTime, 10); // 严格保证10ms间隔 } ``` 即使采样过程耗时波动,系统仍能维持$100ms \pm 1$个节拍的周期精度[^2] #### 五、实现原理示意图 ``` 相对延时时序: |--任务执行--|****100ms延时****|--任务执行--|****100ms延时****| 绝对延时时序: |--任务执行--|****95ms补偿延时****|--任务执行--|****100ms固定间隔****| ``` (假设任务执行时间为5ms,系统节拍为1ms)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值