当前位置: 首页 > news >正文

第九章:OTA 与 Flash 驱动 —— 如何用TDD验证固件升级逻辑的鲁棒性

这一章我们将切入一个非常关键且具有挑战性的场景:OTA(Over-the-Air)固件更新

OTA 逻辑最怕的是什么?是中途断电、Flash 写入失败、校验和不匹配。如果你在真机上测试这些异常,可能需要反复烧录、断开电源,甚至不小心把片子变“砖”。在 TDD 的世界里,我们可以优雅地模拟这些灾难。

9.1 OTA 测试的痛点

  1. 硬件寿命:频繁擦写内部 Flash 也是有寿命限制的。

  2. 场景模拟:你很难手动模拟“在写入第 1024 个字节时突然断电”。

  3. 复杂校验:CRC32、SHA256 的验证在 PC 上跑比在 MCU 上快得多。

9.2 接口抽象:Storage_Interface.h

为了让 OTA 逻辑可测,我们必须隔离 STM32 的HAL_FLASH_Program

// Storage_Interface.h
#ifndef STORAGE_INTERFACE_H
#define STORAGE_INTERFACE_H

#include <stdint.h>
#include <stdbool.h>

typedef enum { FLASH_OK, FLASH_ERROR, FLASH_BUSY } FlashStatus_t;

FlashStatus_t HW_Flash_ErasePage(uint32_t page_addr);
FlashStatus_t HW_Flash_Write(uint32_t addr, uint8_t* data, uint32_t len);
void HW_System_Reset(void);

#endif

9.3 实战:测试“断点续传”逻辑

我们要实现一个 OTA 接收器:它按包接收固件,记录已写入的偏移量,如果写入失败要能重试。

编写测试 (test_Ota_Service.c):我们要模拟一个非常极端的情况:第一次写入成功,第二次写入失败。

#include "unity.h"
#include "mock_Storage_Interface.h"
#include "Ota_Service.h"

void test_Ota_Should_HandleFlashWriteFailure(void) {
uint8_t dummy_data[128] = {0xA5};

// 1. 模拟第一包数据写入:返回成功
HW_Flash_Write_ExpectAndReturn(0x08010000, dummy_data, 128, FLASH_OK);
bool result = Ota_ProcessPacket(0, dummy_data, 128);
TEST_ASSERT_TRUE(result);

// 2. 模拟第二包数据写入:硬件突然报错(比如电压不稳或页损坏)
HW_Flash_Write_ExpectAndReturn(0x08010080, dummy_data, 128, FLASH_ERROR);

result = Ota_ProcessPacket(128, dummy_data, 128);

// 验证:逻辑层应该识别出失败,并返回 false,以便触发重传机制
TEST_ASSERT_FALSE(result);
}

9.4 进阶技巧:模拟“脏数据”与校验失败

在 OTA 中,校验和(Checksum)是最后一道防线。

void test_Ota_Should_RejectFirmware_WhenChecksumMismatch(void) {
// 模拟 Flash 读取出的数据
// 我们不需要真的读 Flash,直接 Mock 掉读取函数返回“坏数据”
HW_Flash_Read_StubWithCallback(my_Fake_Flash_Read_Bad_Data);

// 执行校验逻辑
bool is_valid = Ota_VerifyChecksum();

// 验证:校验不通过,且绝不调用系统重启去运行坏固件
TEST_ASSERT_FALSE(is_valid);
// 确保没有误触发系统重启(这也是一种验证)
HW_System_Reset_Expect(); // 如果你预期它报错后重启到 Bootloader
}

9.5 本章核心:防御式编程:

  1. 状态机保护:OTA 过程中如果收到无关的串口命令,状态机是否会崩溃?

  2. 原子操作模拟:模拟擦除了一半突然断电,Bootloader 能否识别出固件不完整?

  3. 边界检查:Mock 掉 Flash 大小接口,验证如果固件包超过 Flash 容量,代码是否会越界写入。

本章小结

这一章证明了 TDD 不仅能测“好用”的情况,更能测“坏掉”的情况。对于 OTA 这种高风险模块,这种“离线异常注入”是保证产品不批量变砖的唯一手段。

http://www.jsqmd.com/news/926468/

相关文章:

  • 拆解USB PD协议层消息:从Source到Sink,一次完整的充电握手都说了啥?
  • 2026年稻城亚丁四姑娘山旅游品牌TOP5客观盘点 - 优质品牌商家
  • 告别跑断腿!用UltraVNC MSI包+域组策略,半小时搞定全公司远程协助部署
  • 保姆级教程:用迅为RK3568开发板从零烧写实时系统固件(附常见问题排查)
  • 华为RH2288HV3服务器BIOS与iBMC固件升级专用HPM包(含操作指引)
  • CRMEB多商户商城v2.3.2源码包:支持人人分销开通、批量秒杀配置、商品定时上下架及同城配送全流程
  • 告别手动抓包!用CPAL脚本的log函数,实现CANoe自动化测试日志的智能管理
  • MATLAB雨流计数脚本:从结温波动数据直接算IGBT疲劳损伤值
  • 2026年6月湖北武汉工伤维权律所怎么选?这份专业指南助你避坑 - 2026年企业资讯
  • 避坑指南:用WebViewForWindow在Unity播WebRTC,绿屏和硬件加速怎么关?
  • 告别拍脑袋估算!用RUSLE模型5步搞定土壤侵蚀强度计算(附数据获取渠道)
  • 别再只用NTP了!手把手教你用LinuxPTP(ptp4l)实现微秒级时间同步
  • 从网格划分到端口设置:一份给ADS新手的Momentum RF仿真避坑指南(含Via阵列、电感Q值处理)
  • 从RISC-V的ecall指令到用户态printf:一次完整的xv6系统调用“扩胸运动”
  • 手把手教你为Ubuntu 22.04编译安装蓝牙驱动(解决5.15/5.17/5.18内核蓝牙失灵)
  • 基于C++实现(控制台)文件压缩
  • 轻量强大的文件收纳管理工具
  • 保姆级教程:用UE5的Niagara系统,从零手搓一个会动的火焰特效(附材质球避坑点)
  • 不只是环境搭建:用OSG+OSGEARTH 3.1+VS2022快速验证你的三维地理可视化开发环境
  • 2026年Q2青海管道疏通品牌评测:本土适配性深度对比 - 优质品牌商家
  • 成都墙绘单价全维度解析:3d墙绘/四川墙体彩绘公司/四川墙绘公司/地面墙绘/从品类到场景的成本逻辑 - 优质品牌商家
  • 保姆级教程:用davfs2在Ubuntu 22.04上挂载WebDAV网盘(含常见错误排查)
  • 韩文长文本理解失效?Gemini 2.0韩语支持断层分析,3类政务/法律文档误译率高达41.6%,附绕过方案
  • 肺结节CT影像YOLOv5-ready数据集:220+训练图+28测试图+一键可视化脚本
  • 基于C++实现(控制台)学生选课系统
  • 丙午年四月十五那时月
  • 2026年q2西宁管道疏通核心技术与主流企业解析:西宁工地泥浆池清淤/西宁市政管道清淤/优选推荐 - 优质品牌商家
  • 小米高通手机QCN校准参数快速写入工具(9008模式直刷)
  • UE5 GAS实战:别再直接扣血了!用Meta Attributes和Set by Caller重构你的RPG伤害系统
  • 从机器翻译到智驾:规则派的黄昏与数据革命的终局(五)