我有一个在给定机器上顺利运行的 c++ 项目,现在我正试图在另一台具有相同操作系统(Xubuntu 14.04)的机器上编译它。
我已经安装了所有依赖项,并且正在使用 cmake 构建项目,尽管它因以下错误而停止:
确定 pthread_create 函数是否存在于 pthreads 中失败,输出如下: … /usr/bin/ld: cannot find -lpthreads
包含编译器标志的 cmakelists.txt 行如下:
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -O3 -lpthread -DNDEBUG -DEIGEN_MPL2_ONLY")
set(CMAKE_C_FLAGS_DEBUG "-g -O0 -Wall -lpthread -DEIGEN_MPL2_ONLY")
set(CMAKE_CXX_FLAGS "${CMAKE_C_FLAGS} -O3 -lpthread -I/usr/include/freetype2 -DNDEBUG -DEIGEN_MPL2_ONLY")
set(CMAKE_CXX_FLAGS_DEBUG "-g -O0 -Wall -lpthread -I/usr/include/freetype2 -DEIGEN_MPL2_ONLY")
我已经做了一些研究,并且已经尝试了以下方法:
- 使用 -pthread/-threads/-thread/-lpthreads 而不是 -lpthread,这并不能解决问题,并且在没有找到以下包的情况下停止构建: find_package (Threads)
- 在上面的 cmakelists 行中更改了 -lpthread 的顺序,这给出了相同的错误
- 使用了不同版本的 gcc/g++:尝试了 4.4、4.6 和 4.8,没有任何变化
- 在 /usr/lib/ 中创建了指向 libpthread.so 的符号链接,没有任何更改
我会很感激一些帮助,因为我已经不知道下一步该尝试什么了。
编辑 1
图书馆应该是:
$ find /lib -name "*pthread*"
/lib/x86_64-linux-gnu/libpthread-2.19.so
/lib/x86_64-linux-gnu/libpthread.so.0
还找到了 pthread_create:
$ nm /lib/x86_64-linux-gnu/libpthread.so.0 | grep "pthread_create"
0000000000008430 t __pthread_create_2_1
00000000000081430 T pthread_create@@GLIBC_2.2.5
我还验证了 libpthread-stubs0 和 libc6-dev 都存在。
编辑 2
这是 FindThreads.cmake 文件内容的一部分,位于 /usr/share/cmake-2.8/Modules/:
if(CMAKE_HAVE_SPROC_H AND NOT CMAKE_THREAD_PREFER_PTHREAD)
# We have sproc
set(CMAKE_USE_SPROC_INIT 1)
else()
# Do we have pthreads?
CHECK_INCLUDE_FILES("pthread.h" CMAKE_HAVE_PTHREAD_H)
if(CMAKE_HAVE_PTHREAD_H)
#
# We have pthread.h
# Let's check for the library now.
#
set(CMAKE_HAVE_THREADS_LIBRARY)
if(NOT THREADS_HAVE_PTHREAD_ARG)
# Check if pthread functions are in normal C library
CHECK_SYMBOL_EXISTS(pthread_create pthread.h CMAKE_HAVE_LIBC_CREATE)
if(CMAKE_HAVE_LIBC_CREATE)
set(CMAKE_THREAD_LIBS_INIT "")
set(CMAKE_HAVE_THREADS_LIBRARY 1)
set(Threads_FOUND TRUE)
endif()
if(NOT CMAKE_HAVE_THREADS_LIBRARY)
# Do we have -lpthreads
CHECK_LIBRARY_EXISTS(pthreads pthread_create "" CMAKE_HAVE_PTHREADS_CREATE)
if(CMAKE_HAVE_PTHREADS_CREATE)
set(CMAKE_THREAD_LIBS_INIT "-lpthreads")
set(CMAKE_HAVE_THREADS_LIBRARY 1)
set(Threads_FOUND TRUE)
endif()
# Ok, how about -lpthread
CHECK_LIBRARY_EXISTS(pthread pthread_create "" CMAKE_HAVE_PTHREAD_CREATE)
if(CMAKE_HAVE_PTHREAD_CREATE)
set(CMAKE_THREAD_LIBS_INIT "-lpthread")
set(CMAKE_HAVE_THREADS_LIBRARY 1)
set(Threads_FOUND TRUE)
endif()
if(CMAKE_SYSTEM MATCHES "SunOS.*")
# On sun also check for -lthread
CHECK_LIBRARY_EXISTS(thread thr_create "" CMAKE_HAVE_THR_CREATE)
if(CMAKE_HAVE_THR_CREATE)
set(CMAKE_THREAD_LIBS_INIT "-lthread")
set(CMAKE_HAVE_THREADS_LIBRARY 1)
set(Threads_FOUND TRUE)
endif()
endif()
endif()
endif()
if(NOT CMAKE_HAVE_THREADS_LIBRARY)
# If we did not found -lpthread, -lpthread, or -lthread, look for -pthread
if("THREADS_HAVE_PTHREAD_ARG" MATCHES "^THREADS_HAVE_PTHREAD_ARG")
message(STATUS "Check if compiler accepts -pthread")
try_run(THREADS_PTHREAD_ARG THREADS_HAVE_PTHREAD_ARG
${CMAKE_BINARY_DIR}
${CMAKE_ROOT}/Modules/CheckForPthreads.c
CMAKE_FLAGS -DLINK_LIBRARIES:STRING=-pthread
COMPILE_OUTPUT_VARIABLE OUTPUT)
if(THREADS_HAVE_PTHREAD_ARG)
if(THREADS_PTHREAD_ARG STREQUAL "2")
set(Threads_FOUND TRUE)
message(STATUS "Check if compiler accepts -pthread - yes")
else()
message(STATUS "Check if compiler accepts -pthread - no")
file(APPEND
${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log
"Determining if compiler accepts -pthread returned ${THREADS_PTHREAD_ARG} instead of 2. The compiler had the following output:\n${OUTPUT}\n\n")
endif()
else()
message(STATUS "Check if compiler accepts -pthread - no")
file(APPEND
${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log
"Determining if compiler accepts -pthread failed with the following output:\n${OUTPUT}\n\n")
endif()
endif()
if(THREADS_HAVE_PTHREAD_ARG)
set(Threads_FOUND TRUE)
set(CMAKE_THREAD_LIBS_INIT "-pthread")
endif()
endif()
endif()
endif()
编辑 3
使用了一个最小的 Cmakelists.txt,如下所示:
cmake_minimum_required (VERSION 2.4)
find_package(Threads)
这产生了以下输出:
-- Looking for include file pthread.h
-- Looking for include file pthread.h - found
-- Looking for pthread_create
-- Looking for pthread_create - not found.
-- Looking for pthread_create in pthreads
-- Looking for pthread_create in pthreads - not found
-- Looking for pthread_create in pthread
-- Looking for pthread_create in pthread - found
-- Found Threads: TRUE
原文由 Sapiens 发布,翻译遵循 CC BY-SA 4.0 许可协议
运行 cmake 时出现问题。虽然,在这种情况下,cmake 不是问题,但错误是无声的,并且 -lpthreads 相关的错误/警告是唯一写入 cmake 错误日志文件的内容,尽管这不会导致任何问题。我已经完成了 cmakelists.txt 的最小版本并开始逐行测试它,直到我发现哪个包导致它停止:最后我发现它是版本不匹配……
提示:搜索实际的错误信息
通常,您会查找 最后一条错误消息。 然而,在这种情况下,这种(通常有用的)策略会导致误入歧途。
您正在查看的是
CMakeCache.txt
、CMakeOutput.log
或CMakeError.log
。 怎么会? 当配置阶段的某些宏或测试失败时,CMake “有用地”将这些文件转储到输出中。不幸的是,这些文件可能长达数千行,并且通常包含大量“*** Error: xyz"
条目,用于各种配置检查。用于“-lpthreads
”的那个恰好是最后一个日志中的一个…解决方案:从 顶部 浏览日志,识别配置检查的部分,找到该点之前的 _最后一个配置检查_,CMake 识别失败并转储其日志。您也可以尝试搜索文本“
Configuring incomplete, errors occurred!
”通常,您会在那里找到非常精确的实际错误消息,或者至少找到最后调用的宏或函数的名称/路径,这使您可以查明实际出了什么问题。