C测试 - Unity和CMake的未定义引用

时间:2016-10-06 04:30:17

标签: c unit-testing cmake ctest

我目前正在尝试使用C代码覆盖和测试进行项目设置。我的当前堆栈是IDE的CLion,编译器的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%坚持某种链接器问题。

1 个答案:

答案 0 :(得分:2)

util.c不属于tests来源,也不会在链接到tests的库中编译。因此,您不能为tests 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)

我无法在documentation中看到签名及其来源的签名。你想用这个来实现什么?

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

SET(CMAKE_C_STANDARD 99)

编译定义也可以与target_compile_definitions一起声明,以避免全球污染。