web-dev-qa-db-ja.com

ライブラリはリンクされていますが、参照は未定義です

UbuntuでopenCLプログラムをコンパイルして、以前は1度動作したNVIDIAカードを使用しようとしていますが、

#include <CL/cl.h>
#include <iostream>
#include <vector>

using namespace std;

int main() {
  cl_platform_id platform;
  cl_device_id device;
  cl_context context;
  cl_command_queue command_queue;
  cl_int error;

  if(clGetPlatformIDs(1, &platform, NULL) != CL_SUCCESS) {
    cout << "platform error" << endl;
  }

  if(clGetDeviceIDs(platform, CL_DEVICE_TYPE_GPU, 1, &device, NULL) != CL_SUCCESS) {
    cout << "device error" << endl;
  }

  context = clCreateContext(NULL, 1, &device, NULL, NULL, &error);
  if(error != CL_SUCCESS) {
    cout << "context error" << endl;
  }

  command_queue = clCreateCommandQueue(context, device, 0, &error);
  if(error != CL_SUCCESS) {
    cout << "command queue error" << endl;
  }

  return 0;
}

私はそれをそのようにコンパイルします、

g++ -I/usr/local/cuda/include -L/usr/lib/nvidia-current -lOpenCL opencl.cpp

そして、私はこの結果を得る

/tmp/ccAdS9ig.o: In function `main':
opencl.cpp:(.text+0x1a): undefined reference to `clGetPlatformIDs'
opencl.cpp:(.text+0x3d): undefined reference to `clGetDeviceIDs'
opencl.cpp:(.text+0x65): undefined reference to `clCreateContext'
opencl.cpp:(.text+0x85): undefined reference to `clCreateCommandQueue'
collect2: ld returned 1 exit status

だが nm -D /usr/lib/nvidia-current/libOpenCL.soは、libOpenCL.soに少なくともclGetPlatformIDが含まれていることを教えてくれます

0000000000002400 T clGetKernelWorkGroupInfo
0000000000002140 T clGetMemObjectInfo
0000000000002e80 T clGetPlatformIDs
0000000000002de0 T clGetPlatformInfo
0000000000002310 T clGetProgramBuildInfo
00000000000022f0 T clGetProgramInfo
00000000000021f0 T clGetSamplerInfo

私は何かが欠けていますか。

34
user615457

gcc manページから:

   -llibrary
   -l library
       Search the library named library when linking.  (The second alternative with the library as a separate argument is only for POSIX compliance and is not recommended.)

       It makes a difference where in the command you write this option; the linker searches and processes libraries and object files in the order they are specified.  Thus, foo.o
       -lz bar.o searches library z after file foo.o but before bar.o.  If bar.o refers to functions in z, those functions may not be loaded.

       The linker searches a standard list of directories for the library, which is actually a file named liblibrary.a.  The linker then uses this file as if it had been specified
       precisely by name.

したがって、コンパイルコマンドのファイル引数の後に-lOpenCLを指定してみてください。

また、共有ライブラリファイルであるlibOpenCL.soでシンボルを検索します。コマンドを使用して、プログラムをlibOpenCL.a形式の静的ライブラリに再度リンクします。

27
Geoffroy

リンクしているときは、ライブラリとソースファイルの順序が異なります。たとえば、あなたの場合、

g ++ -I/usr/local/cuda/include -L/usr/lib/nvidia-current -lOpenCL opencl.cpp

openCLライブラリで定義された関数は、ルックアップを要求する前に何もないため、ロードされない場合があります。ただし、使用する場合は、

g++ opencl.cpp -I/usr/local/cuda/include -L/usr/lib/nvidia-current -lOpenCL  

関数のリクエストはOpenCLライブラリで検出され、ロードされます。

32
user995502

または、ヘッダーとライブラリパスをグローバル変数に追加できます。

export CPLUS_INCLUDE_PATH=$CPLUS_INCLUDE_PATH:/usr/local/cuda/include
export LIBRARY_PATH=$LIBRARY_PATH:/usr/lib/nvidia-current

設定することもできます

export PATH=$PATH:/usr/local/cuda/bin

今すぐ実行できるはずです

g++ opencl.cpp
0
Jesse