使用 cmake 构建错误:找不到 -lpthreads

新手上路,请多包涵

我有一个在给定机器上顺利运行的 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-stubs0libc6-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 许可协议

阅读 3.4k
2 个回答

运行 cmake 时出现问题。虽然,在这种情况下,cmake 不是问题,但错误是无声的,并且 -lpthreads 相关的错误/警告是唯一写入 cmake 错误日志文件的内容,尽管这不会导致任何问题。我已经完成了 cmakelists.txt 的最小版本并开始逐行测试它,直到我发现哪个包导致它停止:最后我发现它是版本不匹配……

提示:搜索实际的错误信息

通常,您会查找 最后一条错误消息。 然而,在这种情况下,这种(通常有用的)策略会导致误入歧途。

您正在查看的是 CMakeCache.txtCMakeOutput.logCMakeError.log怎么会? 当配置阶段的某些宏或测试失败时,CMake “有用地”将这些文件转储到输出中。不幸的是,这些文件可能长达数千行,并且通常包含大量“ *** Error: xyz" 条目,用于各种配置检查。用于“ -lpthreads ”的那个恰好是最后一个日志中的一个…

解决方案:从 顶部 浏览日志,识别配置检查的部分,找到该点之前的 _最后一个配置检查_,CMake 识别失败并转储其日志。您也可以尝试搜索文本“ Configuring incomplete, errors occurred!

通常,您会在那里找到非常精确的实际错误消息,或者至少找到最后调用的宏或函数的名称/路径,这使您可以查明实际出了什么问题。

原文由 Sapiens 发布,翻译遵循 CC BY-SA 4.0 许可协议

我遇到了完全相同的问题,最小的 Cmakelists.txt 给了我相同的输出。要解决这个问题,只需 将 cmake 升级到 最新版本(在我的情况下为 3.15)

原文由 kaios 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题