第六章: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。

相关推荐
国科安芯6 小时前
ASC4T245S分组双向控制架构深度解析:独立DIR/OE控制、QFN16封装与混合方向总线桥接
单片机·嵌入式硬件·物联网·fpga开发·架构·risc-v
独孤九剑打醒他7 小时前
双层Master-Worker软硬协同调度架构:从根源解决分布式数据一致性难题
后端·嵌入式硬件·硬件架构·硬件工程
andxe8 小时前
安科士AndXe 400G QSFP-DD LR8光模块芯片架构与品控体系解析
网络·光模块·光通信
去码头整点薯条988 小时前
网络实验报告10
网络
JNX_SEMI8 小时前
AT2401C 2.4GHz 全集成射频前端单芯片技术解析
前端·单片机·嵌入式硬件·物联网·硬件工程
坚定的共产主义生产设备永不宕机8 小时前
路由协议OSPF进阶一篇讲全
网络
碎碎念_49210 小时前
以太网技术、VLAN、STP详解
网络·stp·vlan
电子工程师成长日记-C5110 小时前
51单片机智能灯光控制系统
单片机·嵌入式硬件·51单片机
hbugs00110 小时前
【案例分享】全网首个华三数据中心流量可视化实验,基于EVE-NG V7平台
网络·网络协议·安全·devops·eve-ng
狂奔蜗牛(bradley)10 小时前
嵌入式软件编程思想之事件驱动+表驱动状态机+事件参数+优先级FIFO
单片机·mcu