第六章:RTOS 任务 —— 任务逻辑与并发的 TDD 路径

这一章我们聊聊嵌入式开发的"高级阶段":RTOS(实时操作系统)

很多开发者认为 RTOS 的代码没法测,因为涉及任务调度、抢占和阻塞。但在 TDD 的世界里,我们有一个黄金法则:测试任务的"逻辑主体",而不是测试调度器本身。

6.1 核心思想:剥离 OS 依赖

如果你的代码里写满了 xQueueSendosDelay,它在 PC 上依然跑不起来。 策略: 将 RTOS 的 API 视为一种特殊的"底层硬件",通过 Mock 掉信号量、队列和延时函数,我们可以验证任务在各种同步场景下的行为。

6.2 接口抽象:OS_Interface.h

不要直接在业务代码里包含 FreeRTOS.h。我们定义一个中间层:

// OS_Interface.h

#ifndef OS_INTERFACE_H

#define OS_INTERFACE_H

#include <stdint.h>

#include <stdbool.h>

bool OS_QueueReceive(void* msg, uint32_t timeout);

void OS_SignalSemaphore(void);

void OS_Delay(uint32_t ms);

#endif

6.3 实战:测试一个"数据处理任务"

需求: 任务平时处于阻塞状态,等待队列消息。收到数据后,进行计算并根据结果判断是否触发紧急信号。

被测代码 (Data_Task.c):

void Data_Processing_Task_Loop(void) {

uint16_t sensor_data;

// 阻塞等待队列

if (OS_QueueReceive(&sensor_data, 100)) {

if (sensor_data > 900) {

OS_SignalSemaphore(); // 触发紧急信号

}

}

}

编写测试 (test_Data_Task.c): 我们要模拟队列接收成功和失败两种情况。

#include "unity.h"

#include "mock_OS_Interface.h"

#include "Data_Task.h"

// 测试场景 1:收到正常数据

void test_Task_ShouldDoNothing_WhenDataIsNormal(void) {

uint16_t fake_data = 500;

// 模拟队列收到数据 500

OS_QueueReceive_ExpectAnyArgsAndReturn(true);

OS_QueueReceive_ReturnArrayThruPtr_msg(&fake_data, 1);

// 执行一次任务循环

Data_Processing_Task_Loop();

// 结果:没有任何信号量被触发(隐含验证)

}

// 测试场景 2:收到异常高数据

void test_Task_ShouldSignalEmergency_WhenDataExceedsThreshold(void) {

uint16_t alarm_data = 950;

// 模拟队列收到 950

OS_QueueReceive_ExpectAnyArgsAndReturn(true);

OS_QueueReceive_ReturnArrayThruPtr_msg(&alarm_data, 1);

// 预期:信号量会被释放

OS_SignalSemaphore_Expect();

Data_Processing_Task_Loop();

}

6.4 进阶:如何测试"并发"和"竞态"?

在单机单元测试中,我们其实无法模拟真正的多线程抢占,但我们可以模拟"顺序交织"。

  • 测试重入性:连续调用同一个函数,利用 Mock 改变第二次调用的返回状态(比如模拟信号量已被占用)。

  • 测试超时逻辑

void test_Task_ShouldRetry_WhenQueueTimeout(void) {

// 第一次模拟超时

OS_QueueReceive_ExpectAnyArgsAndReturn(false);

// 第二次模拟成功

OS_QueueReceive_ExpectAnyArgsAndReturn(true);

Data_Processing_Task_Loop();

Data_Processing_Task_Loop();

}

6.5 本章核心:任务即函数

注意:

  1. 打破无限循环 :为了测试,不要把 while(1) 写死在任务逻辑函数里。可以写成 void Task_Step(void),由外部循环调用。这样测试用例才能控制执行步数。

  2. 测试粒度:不要试图测试 RTOS 的调度算法,我们要测的是"我的业务逻辑在拿到/拿不到资源时的反应"。

本章小结

现在,你已经掌握了如何将 RTOS 任务降维打击,转变为可控的单线程逻辑。这种方式能帮你规避 80% 的死锁和同步 Bug。

相关推荐
星夜夏空996 小时前
STM32单片机学习(20) —— 利用中断实现串口通信(填前面的坑)
stm32·单片机·学习
wengqidaifeng6 小时前
2026年电赛校赛备战MSPM0G3507+keil讲解(上)-----2025年电赛E题小车篇
单片机·嵌入式硬件·电赛
三易串口屏6 小时前
实验1 实时显示单片机的参数(整数、小数、中文 系统指令方式)
c语言·单片机·嵌入式硬件·mongodb·串口屏·三易串口屏
傅科摆 _ py6 小时前
企业 / 校园 合法远程访问工具详解
服务器·网络·数据库
晚烛7 小时前
CANN 数据流水线优化:从数据加载到模型输入的端到端加速
开发语言·网络·人工智能·python·深度学习
ylscode7 小时前
npm遭遇大规模供应链投毒:@antv生态被植入Shai-Hulud后门,全球开发者需紧急排查
网络·安全·web安全·安全威胁分析
彦为君7 小时前
JavaSE-11-网络编程(详细版)
java·前端·网络·ai·ai编程
头歌实践平台7 小时前
头歌静态路由与默认静态路由
运维·服务器·网络
学不懂飞行器7 小时前
【2024电赛H题硬核解析】自动行驶小车满分对策:多路灰度循迹与陀螺仪“交替盲走”融合算法(附源码)
stm32·单片机·嵌入式硬件·算法·电赛