0


学习使用meson+ninja构建C语言工程(含单元测试)

背景知识

  1. Meson

是一个高效、易用且跨平台的构建系统,主要用于描述和管理项目的构建过程。Meson 使用简洁的配置语言来定义项目的构建规则,并生成 Ninja 构建文件以实际执行构建任务。

  1. Ninja

是一个轻量级的构建工具,专注于快速并行构建。它通过构建任务的依赖关系图,实现高效的并行构建,从而加速项目的构建过程。

  1. CMocka

则是一个适用于 C 语言的单元测试框架,用于编写和执行单元测试,验证代码的正确性。

将 Meson、Ninja 和 CMocka 结合使用,可以创建一个高效的构建和测试环境,有助于开发人员在 C 语言项目中实现自动化构建和单元测试。

以下是使用 Meson、Ninja 和 CMocka 的一般工作流程:

  • 在 Meson 中配置项目:使用 Meson 的简洁配置语言描述项目的构建过程,包括编译选项、依赖项等。在 Meson 配置中,可以指定 CMocka 作为项目的单元测试框架。
  • 生成 Ninja 构建文件:运行 Meson 配置,生成 Ninja 构建系统所需的构建文件。
  • 编写单元测试:使用 CMocka 编写针对 C 代码的单元测试,包括设置测试用例、断言等。
  • 执行单元测试:通过 Ninja 构建系统执行单元测试,验证代码的正确性。通常可以在构建过程中集成单元测试的运行。
  • 查看测试报告:查看单元测试的结果和报告,检查测试用例的覆盖率和执行情况。

通过结合 Meson、Ninja 和 CMocka,开发人员可以实现高效的构建和测试流程,帮助确保代码的质量和稳定性。这种组合提供了一种简单且强大的方式来管理 C 语言项目的构建和单元测试。

具体来说:项目开发中一般将 Meson 和 Ninja 配合使用,Meson 负责构建项目依赖关系,Ninja 负责编译代码。meson和make一样,需要写描述文件告诉meson要构建什么,这个描述文件
就是meson.build,meson根据meson.build中的定义生成具体的构建定义文件build.ninja,
ninja根据build.ninja完成具体构建。

实验环境

  • 系统版本使用Ubuntu22.04,自带python3.10
  • 安装cmocka和pkg-config
  1. admin@pc:~$ sudo apt-get install libcmocka-dev pkg-config -y
  2. admin@pc:~/my_project$ dpkg -l libcmocka-dev | grep libc
  3. ii libcmocka-dev:amd64 1.1.5-2 amd64 development files for the CMocka unit testing framework
  4. admin@pc:~/my_project$
  5. admin@pc:~/my_project$ dpkg -l pkg-config | grep pkg-config
  6. ii pkg-config 0.29.2-1ubun
  • 安装meson和ninja
  1. admin@pc:~$ pip3 install meson ninja
  2. admin@pc:~$ pip3 freeze | grep meson
  3. meson==1.5.1
  4. admin@pc:~$ pip3 freeze | grep ninja
  5. ninja==1.11.1.1
  6. admin@pc:~$

项目构建

  • 项目的目录结构如下
  1. admin@pc:~$ tree my_project/
  2. my_project/
  3. ├── include
  4. └── mylib.h
  5. ├── meson.build
  6. ├── src
  7. ├── main.c
  8. ├── meson.build
  9. └── mylib.c
  10. └── test
  11. ├── meson.build
  12. └── test_mylib.c
  13. 3 directories, 7 files
  14. admin@pc:~$

源码分析

根目录

  • 包含meson.build文件和三个目录include/src/main
  • meson.build的源码如下:
  1. admin@pc:~/my_project$ cat meson.build
  2. project('my_project', 'c')
  3. # 指定头文件目录
  4. inc = include_directories('include')
  5. subdir('src')
  6. subdir('test')
  7. admin@pc:~/my_project$
  • project()指定项目名称和编程语言的类型
  • 创建一个变量inc,通过include_directories()命令用于引入头文件路径
  • subdir()命令将下级目录的meson.build文件包含进来,类似于CMake的add_subdirectory()

include目录

  • 放置头文件,声明函数
  1. admin@pc:~/my_project$ cat include/mylib.h
  2. #ifndef MYLIB_H
  3. #define MYLIB_H
  4. void greet(void);
  5. int add(int a, int b);
  6. #endif // MYLIB_H
  7. admin@pc:~/my_project$

src目录

  • 源码文件内容如下:
  1. admin@pc:~/my_project$ cat src/mylib.c
  2. #include <stdio.h>
  3. #include "mylib.h"
  4. void greet(void) {
  5. printf("Hello from mylib!\n");
  6. }
  7. int add(int a, int b) {
  8. return a + b;
  9. }
  10. admin@pc:~/my_project$
  11. admin@pc:~/my_project$ cat src/main.c
  12. #include "mylib.h"
  13. int main() {
  14. greet();
  15. return 0;
  16. }
  17. admin@pc:~/my_project$
  • meson.build文件如下
  1. admin@pc:~/my_project$ cat src/meson.build
  2. # 编译库
  3. libmylib = static_library('mylib', 'mylib.c', include_directories: inc)
  4. # 编译可执行文件并链接库
  5. executable('my_project', 'main.c', link_with: libmylib, include_directories: inc)
  6. admin@pc:~/my_project$
  • static_library()指定静态库文件的文件名和入口源文件
  • 'inc’在上一层meson.build已经定于
  • executable()创建可执行文件,'my_project’是最终可执行文件的名字,'main.c’是源文件,link_with指定lib库

test目录

  • 源码文件内容如下,故意构造了success的测试用例和fail测试用例:
  1. admin@pc:~/my_project$ cat test/test_mylib.c
  2. #include <stdarg.h>
  3. #include <stddef.h>
  4. #include <setjmp.h>
  5. #include <cmocka.h>
  6. #include "mylib.h"
  7. static void test_add_success(void **state) {
  8. assert_int_equal(add(2, 3), 5);
  9. assert_int_equal(add(-1, 1), 0);
  10. }
  11. static void test_add_fail(void **state) {
  12. assert_int_equal(add(2, 2), 5);
  13. }
  14. int main(void) {
  15. const struct CMUnitTest tests[] = {
  16. cmocka_unit_test(test_add_success),
  17. cmocka_unit_test(test_add_fail),
  18. };
  19. return cmocka_run_group_tests(tests, NULL, NULL);
  20. }
  21. admin@pc:~/my_project$
  • meson.build文件如下
  1. admin@pc:~/my_project$ cat test/meson.build
  2. # 依赖 CMocka
  3. cmocka_dep = dependency('cmocka', required: true)
  4. # 编译测试可执行文件
  5. test_executable = executable('test_mylib', 'test_mylib.c',
  6. dependencies: [cmocka_dep],
  7. include_directories: inc,
  8. link_with: libmylib)
  9. # 添加测试
  10. test('mylib tests', test_executable)
  11. admin@pc:~/my_project$

编译和执行

  • 构建
  1. admin@pc:~/my_project$ meson setup builddir
  • 编译
  1. admin@pc:~/my_project$ meson compile -C builddir/
  • 编出了两个可执行文件,都在builddir目录下
  1. admin@pc:~/my_project$ find ./ -name my_project
  2. ./builddir/src/my_project
  3. admin@pc:~/my_project$
  4. admin@pc:~/my_project$
  5. admin@pc:~/my_project$ find ./ -name test_mylib
  6. ./builddir/test/test_mylib
  7. admin@pc:~/my_project$
  • 分别执行
  1. admin@pc:~/my_project$ ./builddir/src/my_project
  2. Hello from mylib!
  3. admin@pc:~/my_project$
  4. admin@pc:~/my_project$
  5. admin@pc:~/my_project$ ./builddir/test/test_mylib
  6. [==========] Running 2 test(s).
  7. [ RUN ] test_add_success
  8. [ OK ] test_add_success
  9. [ RUN ] test_add_fail
  10. [ ERROR ] --- 0x4 != 0x5
  11. [ LINE ] --- ../test/test_mylib.c:13: error: Failure!
  12. [ FAILED ] test_add_fail
  13. [==========] 2 test(s) run.
  14. [ PASSED ] 1 test(s).
  15. [ FAILED ] 1 test(s), listed below:
  16. [ FAILED ] test_add_fail
  17. 1 FAILED TEST(S)
  18. admin@pc:~/my_project$

清理

  • 只要删除了’builddir’目录,就清清静静了
标签: c语言 单元测试

本文转载自: https://blog.csdn.net/ljyfree/article/details/141447336
版权归原作者 ljyfree 所有, 如有侵权,请联系我们删除。

“学习使用meson+ninja构建C语言工程(含单元测试)”的评论:

还没有评论