无法使用 CMake 和 CUDA 构建共享库

Fail to build shared library using CMake and CUDA

我正在尝试使用 CMake 构建包含 CUDA 代码的共享库。我正在使用包 findCUDA。 我在 linking 阶段遇到问题:

Linking CXX shared library shlibcuda.so
/usr/bin/c++  -fPIC -std=c++0x -fopenmp -O3 -DNDEBUG   -shared -Wl,-soname,shlibcuda.so -o shlibcuda.so CMakeFiles/shlibcuda.dir/src/cuda/./shlibcuda_generated_calibrate.cu.o CMakeFiles/shlibcuda.dir/src/cuda/./shlibcuda_generated_cleaning.cu.o CMakeFiles/shlibcuda.dir/./shlibcuda_intermediate_link.o  -L/usr/local/cuda-6.5/lib64/libcudart.so -Wl,-rpath,/mylibs/lib:/usr/local/cuda-6.5/lib64 
/usr/bin/ld: CMakeFiles/shlibcuda.dir/./shlibcuda_intermediate_link.o: relocation R_X86_64_32S against `__nv_module_id' can not be used when making a shared object; recompile with -fPIC
CMakeFiles/shlibcuda.dir/./shlibscuda_intermediate_link.o: error adding symbols: Bad value

this question 及其答案我发现问题可能是要 linked 的目标文件之一没有使用 -fPIC 选项编译。我将 -Xcompiler -fPIC 添加到 CUDA_NVCC_FLAGS。

事实上,正如您在下面的行中看到的,当构建过程到达所谓的中间link文件的构建时,没有-fPIC 传递给编译器:

[100%] Building NVCC intermediate link file CMakeFiles/shlibcuda.dir/./shlibcuda_intermediate_link.o
/usr/local/cuda-6.5/bin/nvcc -m64 -ccbin "/usr/bin/cc" -dlink CMakeFiles/shlibcuda.dir/src/cuda/./shlibcuda_generated_calibrate.cu.o CMakeFiles/shlibcuda.dir/src/cuda/./shlibcuda_generated_cleaning.cu.o  -o CMakeFiles/shlibcuda.dir/./shlibcuda_intermediate_link.o

我的 NVCC 标志如下:

#CUDA include directories
find_package(CUDA REQUIRED)
set(CUDA_NVCC_FLAGS ${CUDA_NVCC_FLAGS}; -Xcompiler -fPIC; -O3; -gencode arch=compute_32,code=sm_32; -ccbin /usr/bin/g++ -std=c++11)

我做错了什么?如果问题是缺少-fPIC,编译中间link文件时如何传递那个选项?

我正在使用 CUDA 6.5 并且我正在传递 -ccbin /usr/bin/g++ -std=c++11 选项,因为我需要在主机代码中使用一些 c++11。

我的cmake: 2.8.12.2.

这是一个由 this patch 解决的 cmake 问题(从 cmake 3.2.0 开始包含)。编译中间 link 文件时,-fPIC 标志也会被传递。

但是出现了另一个问题,因为在我的配置中我必须显式地传递主机编译器:

[100%] Building NVCC intermediate link file CMakeFiles/shlibcuda.dir/./shlibcuda_intermediate_link.o
    /usr/local/cuda-6.5/bin/nvcc -Xcompiler -fPIC -O3 -gencode arch=compute_32,code=sm_32 -ccbin /usr/bin/g++ -std=c++11 -m64 -ccbin "/usr/bin/gcc-4.8" -dlink CMakeFiles/shlibcuda.dir/src/cuda/./shlibcuda_generated_calibrate.cu.o CMakeFiles/shlibcuda.dir/src/cuda/./shlibcuda_generated_cleaning.cu.o  -o CMakeFiles/shlibcuda.dir/./shlibcuda_intermediate_link.o
nvcc fatal   : redefinition of argument 'compiler-bindir'

因为 -ccbin /usr/bin/g++ -std=c++11-ccbin "/usr/bin/gcc-4.8" 都存在。

这是一个已知的cmake open issue n. 0013674。讨论仍在进行中,但应用了该线程中附带的补丁(它只是删除了 CUDA_NVCC_FLAGS 中对“-ccbin”的检查)问题已解决。

issue in the new issue tracker 的讨论中,一个建议是使用 CUDA_HOST_COMPILER 变量而不是 -ccbin