3

我目前正在尝试使用 C 语言进行代码覆盖和测试的项目设置。我当前的堆栈是 CLion 用于 IDE,Clang 用于编译器,gcov 和 lcov 用于覆盖,Unity 用于测试框架,CMock 用于模拟/在测试期间存根。

我目前有以下包结构:

app/root
  | build
      | *.*
  |- cmake
     |- modules
        |- CodeCoverage.cmake
  |- coverage
      |- coverage.info
  |- external
      |- Unity
      |- CMock
      |- CMakeLists.txt
  |- src
      |- *.c
      |- *.h
      |- CMakeLists.txt
  |- tests
      |- *.c
      |- *.h
      |- CMakeLists.txt
  |- CMakeLists.txt

我更高级别的 CMakeLists.txt 看起来像:

cmake_minimum_required(VERSION 3.6)

project(my_c_app)

set(CMAKE_C_COMPILER "/usr/bin/clang")

set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} ${PROJECT_SOURCE_DIR}/cmake/modules)


set(CMAKE_VERBOSE_MAKEFILE ON)

add_subdirectory(external)
add_subdirectory(src)
add_subdirectory(tests)

我的应用级别 CMakeLists.txt 看起来像:

SET(CMAKE_CXX_FLAGS "-O0")
SET(CMAKE_C_FLAGS "-DLINUX -O0 -Wall -std=c99")

set(SOURCE_FILES
        util.c
        util.h)

add_executable(my_c_app ${SOURCE_FILES})

set(THREADS_PREFER_PTHREAD_FLAG ON)
find_package(Threads REQUIRED)
target_link_libraries(my_c_app Threads::Threads)

target_include_directories(my_c_app PUBLIC ${PROJECT_SOURCE_DIR}/include)

我的测试级别 CMakeLists.txt 看起来像:

enable_testing()

include(CodeCoverage)
include(CTest)

SET(CMAKE_CXX_FLAGS "-g -O0 -fprofile-arcs -ftest-coverage")
SET(CMAKE_C_FLAGS "-DLINUX -O0 -Wall -std=c99 -g -fprofile-arcs -ftest-coverage")

SETUP_TARGET_FOR_COVERAGE(coverage tests ${PROJECT_SOURCE_DIR}/coverage/coverage "'/usr/*';'tests/*';'external/*'")

add_executable(tests util_test.c)

target_link_libraries(tests Unity CMock)

add_test(tests util_test.c)

目前我的问题是我没有正确地做某事。尝试在 util.c 中测试函数时,我得到一个未定义的引用:

CMakeFiles/tests.dir/util_test.c.o: In function `test_my_method':
/home/patches/my_c_app/tests/util_test.c:6: undefined reference to `my_method'

我的 util_test.c 目前是:

#include <unity.h>
#include "../src/util.h"

void test_my_method(void) {
    uchar result = my_method();
    // assertion and other logic would go here
}

int main(void) {
    UNITY_BEGIN();
    RUN_TEST(test_my_method);
    return UNITY_END();
}

我是测试驱动的 c 开发和 CMake 的菜鸟,那么我应该设置测试以使它们依赖于 src 中的 c 文件的正确方法是什么?

如果我只是执行 TEST_ASSERT_EQUAL(1,1) 而不是调用 util.c 函数,我会看到:

1 Tests 0 Failures 0 Ignores
OK

Process finished with exit code 0

所以我觉得我 100% 陷入了某种链接器问题。

4

1 回答 1

3

util.c不是tests源代码的一部分,也不是在链接到tests. 因此,您没有为testsfor提供任何定义my_method,因此未定义参考。

main.c您的可执行文件的源文件中没有文件my_c_app。我猜你的主要功能是在util.c. 如果我是对的,请将其放入main.c文件中,并将您的应用级别 CMakeLists.txt 更改为:

SET(CMAKE_CXX_FLAGS "-O0")
SET(CMAKE_C_FLAGS "-DLINUX -O0 -Wall -std=c99")

set(SOURCE_FILES
        util.c
        util.h)

add_library(my_c_lib STATIC ${SOURCE_FILES})

set(THREADS_PREFER_PTHREAD_FLAG ON)
find_package(Threads REQUIRED)
target_link_libraries(my_c_app Threads::Threads)

target_include_directories(my_c_app PUBLIC ${PROJECT_SOURCE_DIR}/include)

add_executable(my_c_app main.c)
target_link_libraries(my_c_app my_c_lib)

现在,您的源代码已编译到my_c_lib您可以链接的静态库中。您的应用程序已链接到它,您也可以在测试级别 CMakeLists.txt 中将您的测试链接到它:

enable_testing()

include(CodeCoverage)
include(CTest)

SET(CMAKE_CXX_FLAGS "-g -O0 -fprofile-arcs -ftest-coverage")
SET(CMAKE_C_FLAGS "-DLINUX -O0 -Wall -std=c99 -g -fprofile-arcs -ftest-coverage")

SETUP_TARGET_FOR_COVERAGE(coverage tests ${PROJECT_SOURCE_DIR}/coverage/coverage "'/usr/*';'tests/*';'external/*'")

add_executable(tests util_test.c)

target_link_libraries(tests Unity CMock my_c_lib)
                                        ^^^^^^^^

add_test(tests util_test.c)

关于这一行:

add_test(tests util_test.c)

我在获取目标及其来源的文档中看不到签名。你想用这个来达到什么目的?

请注意,您可以使用C_STANDARD而不是指定 C 版本CMAKE_C_FLAGS

SET(CMAKE_C_STANDARD 99)

也可以声明编译定义target_compile_definitions以避免全局污染。

于 2016-10-06T07:42:43.277 回答