Как разрешить конфликт связывания GTest и LibTorch

Этот вопрос следует за моим предыдущим вопросом < / а>.

Я пишу программу на C ++ с OpenCV, Torch и NumCpp. Программа компилируется и пока работает нормально, но мне нужно написать модульные тесты.

Я следил за руководством Google построить GTest и GMock внутри моего проекта, но это не удается. Когда я не подключаю библиотеки Torch, это работает.

Ошибка при связывании GTest + Torch:

/usr/bin/ld: CMakeFiles/TryGTest_test.dir/test/boxTest.cpp.o: in function `testing::AssertionResult testing::internal::CmpHelperEQFailure<int, int>(char const*, char const*, int const&, int const&)':
/tmp/tmp.Z1zXnMtLsD/cmake-build-debug-ubuntu_2/googletest-src/googletest/include/gtest/gtest.h:1511: undefined reference to `testing::internal::EqFailure(char const*, char const*, std::string const&, std::string const&, bool)'
collect2: error: ld returned 1 exit status

Я полагаю, это исходит от того, что Libtorch определяет макросы с тем же именем, что и Gtest, а Google предлагает обходной путь, чтобы исправить этот тип проблемы, но мне нужно найти, какой макрос не работает.

Надеюсь, кто-то может помочь!

Вот мой CMakeLists.txt

cmake_minimum_required(VERSION 2.6)
project(TryGtest)

set(CMAKE_CXX_STANDARD 14) # C14 required to compile Torch
set(CMAKE_CXX_STANDARD_REQUIRED TRUE)

# Specifying we are using pthread for UNIX systems.
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${TORCH_CXX_FLAGS} -pthread -Wall")

find_package(OpenCV REQUIRED)
find_package(Torch REQUIRED)

if(NOT Torch_FOUND)
    message(FATAL_ERROR "Pytorch Not Found!")
endif(NOT Torch_FOUND)

message(STATUS "Pytorch status :")
message(STATUS "    libraries: ${TORCH_LIBRARIES}")
message(STATUS "    Torch Flags: ${TORCH_CXX_FLAGS}")

message(STATUS "OpenCV library status :")
message(STATUS "    version: ${OpenCV_VERSION}")
message(STATUS "    libraries: ${OpenCV_LIBS}")
message(STATUS "    include path: ${OpenCV_INCLUDE_DIRS}")

# -------- GOOGLE TEST ----------
# Download and unpack googletest at configure time
configure_file(CMakeLists.txt.in googletest-download/CMakeLists.txt)
execute_process(COMMAND ${CMAKE_COMMAND} -G "${CMAKE_GENERATOR}" .
        RESULT_VARIABLE result
        WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}/googletest-download )
if(result)
    message(FATAL_ERROR "CMake step for googletest failed: ${result}")
endif()
execute_process(COMMAND ${CMAKE_COMMAND} --build .
        RESULT_VARIABLE result
        WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}/googletest-download )
if(result)
    message(FATAL_ERROR "Build step for googletest failed: ${result}")
endif()

# Prevent overriding the parent project's compiler/linker
# settings on Windows
set(gtest_force_shared_crt ON CACHE BOOL "" FORCE)

# Add googletest directly to our build. This defines
# the gtest and gtest_main targets.
add_subdirectory(${CMAKE_CURRENT_BINARY_DIR}/googletest-src
        ${CMAKE_CURRENT_BINARY_DIR}/googletest-build
        EXCLUDE_FROM_ALL)

# The gtest/gtest_main targets carry header search path
# dependencies automatically when using CMake 2.8.11 or
# later. Otherwise we have to add them here ourselves.
if (CMAKE_VERSION VERSION_LESS 2.8.11)
    include_directories("${gtest_SOURCE_DIR}/include")
endif()
# -------------------------------------------------------------------------
enable_testing()
include_directories("${gtest_SOURCE_DIR}/include")


# Program executable
add_executable(TryGTest src/main.cpp src/box.cpp include/util.h)

# Test executable
add_executable(TryGTest_test test/main.cpp src/box.cpp test/boxTest.cpp include/util.h)

target_link_libraries(TryGTest PRIVATE pthread dl util ${TORCH_LIBRARIES} ${OpenCV_LIBS} )
target_link_libraries (TryGTest_test PRIVATE pthread dl util ${TORCH_LIBRARIES} ${OpenCV_LIBS} gtest gmock)

и CMakeLists.txt.in

cmake_minimum_required(VERSION 2.8.2)

project(googletest-download NONE)

include(ExternalProject)
ExternalProject_Add(googletest
        GIT_REPOSITORY    https://github.com/google/googletest.git
        GIT_TAG           release-1.10.0
        SOURCE_DIR        "${CMAKE_CURRENT_BINARY_DIR}/googletest-src"
        BINARY_DIR        "${CMAKE_CURRENT_BINARY_DIR}/googletest-build"
        CONFIGURE_COMMAND ""
        BUILD_COMMAND     ""
        INSTALL_COMMAND   ""
        TEST_COMMAND      ""
        )

person totok    schedule 02.07.2020    source источник
comment
Есть только три макроса, которые вы можете отменить с помощью этого временного решения: FAIL, SUCCEED и TEST. Я сомневаюсь, что что-то из этого поможет. Как вызвать компоновщик? Можете показать полную команду?   -  person n. 1.8e9-where's-my-share m.    schedule 02.07.2020
comment
@ n.'pronouns'm. Я отредактировал сообщение своими CMakeLists   -  person totok    schedule 02.07.2020
comment
@ n.'pronouns'm. Вы уверены, что мы можем только переопределить эти слова? Я добавил -DGTEST_DONT_DEFINE_ASSERT_EQ=1 к параметрам компиляции, и мне пришлось переименовать ASSERT_EQ в моем тестовом коде в GTEST_ASSERT_EQ, что доказывает, что это может сработать.   -  person totok    schedule 02.07.2020
comment
CMakeLists слишком далек от действия. Я не могу сказать, что он делает, не создав ваш проект. Вот почему я попросил команду вызова компоновщика, а не сценарий конфигурации, который ее создает.   -  person n. 1.8e9-where's-my-share m.    schedule 02.07.2020
comment
Как я могу предоставить вам то, что вам нужно?   -  person totok    schedule 02.07.2020
comment
Видимо все АССЕРТ_ ?? макросы также могут быть неопределенными, хотя это не задокументировано.   -  person n. 1.8e9-where's-my-share m.    schedule 02.07.2020
comment
Попробуйте make VERBOSE=1, это печатает команды по мере их вызова.   -  person n. 1.8e9-where's-my-share m.    schedule 02.07.2020
comment
Вот подробный вывод.   -  person totok    schedule 02.07.2020
comment
Это странно. Команда выглядит нормально. Предполагается, что рассматриваемая функция находится в libgtest.a, который построен из gtest-all.cc.o, который построен из gtest-all.cc, который включает gtest.cc, который содержит рассматриваемую функцию. Попробуйте использовать nm -C libgtest.a и nm -C gtest-all.cc.o и grep для EqFailure.   -  person n. 1.8e9-where's-my-share m.    schedule 02.07.2020
comment
Позвольте нам продолжить это обсуждение в чате.   -  person totok    schedule 02.07.2020


Ответы (1)


PyTorch использует флаг компиляции -D_GLIBCXX_USE_CXX11_ABI=0 (что должно быть уголовным преступлением в IMNSHO 2020). Вам нужно скомпилировать весь код, включая gtest, с этим флагом.

Этот флаг находится в TORCH_CXX_FLAGS, однако gtest использует свой собственный CMakeLists.txt со своим собственным набором флагов. Вы должны добавить его вручную. Самый простой способ - это, вероятно, add_compile_definitions(_GLIBCXX_USE_CXX11_ABI=0) где-нибудь в верхней части файла CMakeLists.txt.

person n. 1.8e9-where's-my-share m.    schedule 02.07.2020
comment
Ты. Находятся. Гений. Большое спасибо за помощь! Можешь просто отредактировать ответ? -D не должен быть в команде add_compile_definitions, он добавляется автоматически. Спасибо за время, которое вы уделили мне! - person totok; 02.07.2020
comment
Не могли бы вы объяснить мне или прислать мне документацию о том, почему нельзя использовать этот флаг ABI? - person totok; 02.07.2020
comment
Флаг понижает стандартную библиотеку ABI до версии pre-gcc5.1, которая является древней и делает невозможной полную поддержку C ++ 11 (это было причиной изменения ABI в первую очередь). Это изменение обратно несовместимо, поэтому невозможно связать вместе библиотеки, использующие старый и новый ABI. Посмотрите dual abi для получения дополнительной информации. - person n. 1.8e9-where's-my-share m.; 02.07.2020