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

相关推荐
睡不醒男孩03082317 小时前
CLup 6.x 版本中针对StarRocks 存算一体集群的完整操作手册
java·服务器·网络·clup
dog25018 小时前
网络长尾延时的重尾本质
开发语言·网络·php
戴西软件18 小时前
戴西 DLM 许可授权管理系统:破解无网络环境下工业软件授权难题,助力制造企业降本增效
网络·人工智能·python·深度学习·程序人生·算法·制造
6v6-博客18 小时前
u盘出现图片、文件,文件夹全部隐藏怎么办
网络
西城微科方案开发18 小时前
多品类电子秤一体化PCBA整体方案
单片机·嵌入式硬件·电子秤
hj28625118 小时前
Linux 网络服务综合笔记(概念 + 命令 + 实操案例)2
linux·运维·网络
火花页.18 小时前
【正点原子ZYNQ领航者7020】PS端GPIO中断→按键控制LED实验
单片机·嵌入式硬件
ye1501277745518 小时前
4.2V升6V1A同步升压WT3213
单片机·嵌入式硬件·其他·硬件工程
A_humble_scholar19 小时前
Linux(七)调度器:从硬件矛盾到进程切换的底层逻辑
linux·服务器·网络
嵌入式小站19 小时前
STM32 可移植教程 02:按键状态机,消抖、长按、释放一行也不用多写(实战篇)
chrome·vscode·stm32·单片机·嵌入式硬件