如何对嵌入式软件进行单元测试

ceedling就是一款嵌入式软件测试框架。ceedling是一个用ruby语言编写的C语言自动化测试框架,它集成了Cmock、Unity和Cexception等多个开源项目。在整个ceedling框架中,使用unity进行代码测试,使用CMock生成模拟函数,使用CException进行异常处理,ceedling本身可以理解为对这三个框架的整合,提供自动化配置、操作等,简化开发。

一 ceedling安装步骤

ceedling使用ruby开发脚本,使用gcc工具链编译,因此需要安装ruby和gcc。

(1)首先安装ruby。ruby安装包见《rubyinstaller-devkit-3.0.2-1-x64.zip》

(2)安装 gcc。拷贝文件夹《mingw64》拷贝到目录C:\Program Files\下。之后把路径C:\Program Files\mingw64\bin添加到系统环境变量中。

验证:gcc --version

(3)安装ceedling,执行命令gem install ceedling,执行此命令需要连接外网。

安装完毕验证:ceedling version

二 ceedling配置

ceedling是通过project.yml来进行配置。project.yml解释如下:

# 项目元数据

:project:

:use_exceptions: FALSE # 是否启用 C++ 异常(C 项目一般 FALSE)

:use_test_preprocessor: TRUE # 是否对测试文件也跑一遍预处理器

:use_auxiliary_dependencies: TRUE

:build_root: build # 所有中间文件、目标文件、测试可执行文件都会放到 build/

:test_file_prefix: test_ # 测试源文件必须 test_ 开头(可改)

# 路径配置

:paths:

:test:

  • test/** # 递归搜索 test 目录下所有测试文件

:source:

  • src/** # 被测源码

:include:

  • inc # 公共头文件

  • mocks # 手动桩或额外包含路径

  • third_party/cmsis/include # 第三方库

# 工具链定义

:tools:

:test_compiler:

:executable: gcc # 可执行文件名(Windows 用 gcc.exe)

:arguments:

  • -std=c11 # C 语言标准

  • -Wall

  • -Wextra

  • -Werror # 把告警当错误

  • -Wno-unused-parameter # 针对 Unity 生成代码的告警

  • -I"${':include'}" # 展开 :paths:include 里的路径

  • -I"${':source'}"

  • -DUNITY_INCLUDE_CONFIG_H # 告诉 Unity 使用 Ceedling 生成的配置

# 全局宏定义

:defines:

:common: &common_defines # 定义锚点,后面可以引用

  • UNIT_TEST

  • DEBUG=1

:test:

  • *common_defines # 引用锚点

  • TEST # 仅在测试编译时生效

:release:

  • *common_defines

  • NDEBUG # 发布版本去掉 assert

# CMock 配置

:cmock:

:mock_prefix: mock_ # 生成的桩文件名前缀

:when_no_prototypes: :warn # 如果头文件没有函数原型就告警

:enforce_strict_ordering: TRUE # 是否严格要求函数调用顺序

:includes:

  • stddef.h # 每个桩文件都强制包含的头

  • stdint.h

:treat_inlines: :include # 也 mock 声明为 static inline 的函数

:treat_externs: :exclude # 不自动 mock extern 变量

:plugins:

  • :ignore # 支持 ignore_args

  • :callback # 支持 callback 插件

  • :expect_any_args # 支持 ExpectAnyArgs

# Unity 配置

:unity:

:defines:

  • UNITY_INCLUDE_DOUBLE # 让 Unity 支持 TEST_ASSERT_EQUAL_DOUBLE

# 插件(gcov / bullseye / xml 报告等)

:plugins:

:enabled:

  • gcov # 覆盖率报告

  • xml_tests_report # 生成 JUnit XML 给 CI

  • stdout_pretty_tests_report # 彩色命令行输出

:gcov:

:reports:

  • HtmlDetailed # 生成 html

:gcovr:

:html_medium_threshold: 75 # 覆盖率 75% 以上标绿

:html_high_threshold: 90

# 环境变量

:environment:

  • :path:

  • C:\tools\mingw64\bin # Windows 下额外 PATH

三 如何使用ceedling编写测试代码

  1. 在某一目录(英文目录)下,执行如下命令:

ceedling new my_project

目录结构自动生成如下所示:

my_project

├─ src/ # 放被测源码

├─ test/ # 放测试文件

├─ project.yml # 配置文件

把你要测试的源文件(例i2c_dw_core.c)拷贝到src/目录下,并且i2c_dw_core.c的头文件导入更换为:

#include "i2c_dw_core.h"

#include "i2c_dw_core_cmock.h"

  1. 新建文件i2c_dw_core.h,内容如下所示:

#ifndef I2C_DW_CORE_H

#define I2C_DW_CORE_H

#include "base.h"

如下函数声明是待测函数的声明

int i2c_dw_init_master(struct dw_i2c_dev *dev);

void i2c_dw_xfer_msg(struct dw_i2c_dev *dev);

#endif

(3)新建文件i2c_dw_core_cmock.h内容如下所示:

#ifndef I2C_DW_CORE_CMOCK

#define I2C_DW_CORE_CMOCK

如下函数声明是需要打桩的函数的声明

void __i2c_dw_disable_nowait(struct dw_i2c_dev *dev);

void rt_completion_done(struct rt_completion *completion);

#endif

(4)在src/目录下新建文件base.h(所有待测源文件的数据类型定义可以共用base.h),内容如下:

#ifndef BASE_H

#define BASE_H

开始添加待测文件用到的所有数据类型的定义

typedef unsigned int UINT32;

#endif

(5)生成文件test_i2c_dw_core.c,并在此文件中编写测试代码

执行命令ceedling module:create[i2c_dw_core]

在test/目录下,打开文件test_i2c_dw_core.c进行测试代码编写。

首先在#include "i2c_dw_core.h"语句后添加如下一条语句: #include "mock_i2c_dw_core_cmock.h"

开始编写测试代码。。。

四 ceedling常用打桩宏及其用法

一般用前四个

(1)Func_Ignore()

彻底忽略型

Func要求:无返回值,如有返回值则不能使用此宏。有无入参均可。

源文件调用多少次都放行。

(2)Func_IgnoreAndReturn(ret)

忽略入参,但固定返回ret

Func要求:有返回值。有无入参均可。

源文件调用多少次都放行。

(3)Func_Expect(arg1, arg2, ...)

Func要求:无返回值,有入参。

这个宏作用是告诉 CMock "下一次调用 Func 时,实参必须完全等于括号里的值,否则测试立即失败"。参数按位置逐一比较。

此宏只能匹配一次调用(调用后就被"消耗"掉)。

如果函数还有返回值,需用 Func_ExpectAndReturn(arg1, arg2, ..., ret)

(4)Func_ExpectAndReturn(a,b,ret)

Func要求:有返回值,有入参。这个宏作用是期望下一次调用的参数必须严格匹配括号里的值,并且让调用返回ret。参数个数、顺序、值都必须完全一致

此宏只能匹配一次调用(调用后就被"消耗"掉)。

(5)Func_ExpectAnyArgs()

Func要求:无返回值,有入参。

这个宏作用是告诉 CMock"下一次调用 Func 时,不管传什么参数,都算通过"。

只能匹配一次调用(调用后就被"消耗"掉)。

(6)Func_ExpectAnyArgsAndReturn(ret)

Func要求:有返回值,有入参。

这个宏作用是告诉CMock:"下一次调用 Func 时,参数完全无所谓,但必须返回ret"。

此宏只能匹配一次调用(调用后就被"消耗"掉)。

五 执行单元测试用例

以管理员身份打开windows终端,然后进入工程目录my_project

执行如下命令:

ceedling clean

ceedling test:test_i2c_dw_core.c #仅执行测试代码文件test_i2c_dw_core.c

ceedling gcov:test_i2c_dw_core.c #生成测试报告,测试报告在目录my_project\build\artifacts\gcov\gcovr下。

相关推荐
前端工作日常1 天前
单元测试与E2E测试中使用浏览器的原因及区别
前端·单元测试
猿小蔡-Cool2 天前
Robolectric如何启动一个Activity
android·单元测试·rebolectric
猿小蔡2 天前
Robolectric拿到当前的Activity
android·单元测试
程序员二黑2 天前
性能测试全流程解析:从需求分析到报告输出
单元测试·测试·ab测试
猿小蔡2 天前
Robolectric如何启动一个Activity
android·单元测试
测试19983 天前
单元测试到底是什么?该怎么做?
自动化测试·软件测试·python·测试工具·职场和发展·单元测试·测试用例
程序员二黑4 天前
2025年五大顶级自动化测试工具深度解析:测开工程师的终极选择指南
单元测试·测试·ab测试
Poppy .^0^4 天前
Maven 项目单元测试实战指南:从环境搭建到问题排查全解析
单元测试·log4j·maven
魏波.6 天前
如何对springboot mapper 编写单元测试
单元测试