我是OpenACC的新手,我正在从头开始写一个新程序(我很清楚以前处理类似问题时哪些循环的计算成本会很高(。我得到了一个";未定义的引用";来自nvlink。从我的研究中,我发现这是因为我创建的类没有生成设备代码。然而,我不明白为什么会发生这种情况,以及如何解决它
下面我从我的代码中发送一个MWE。
包括/vec1.h
#ifndef VEC1_H
#define VEC1_H
class Vec1{
public:
double data[1];
#pragma acc routine seq
Vec1();
#pragma acc routine seq
Vec1(double x);
#pragma acc routine seq
Vec1 operator* (double x);
};
#endif
src/vec1.cpp
#include "vec1.h"
Vec1::Vec1(){
data[0] = .0;
}
Vec1::Vec1(double x){
data[0] = x;
}
Vec1 Vec1::operator*(double c){
Vec1 r = Vec1(0.);
r.data[0] = c*data[0];
return r;
}
vec1_test_gpu.cpp
#include "vec1.h"
#define NUM_VECTORS 1000000
int main(){
Vec1 vec1_array[NUM_VECTORS];
for(int iv=0; iv<NUM_VECTORS; ++iv){
vec1_array[iv] = Vec1(iv);
}
#pragma acc data copyin(vec1_array)
#pragma acc parallel loop
for(int iv=0; iv<NUM_VECTORS; ++iv){
vec1_array[iv] = vec1_array[iv]*2;
}
return 0;
}
我以以下方式编译它们
$ nvc++ src/vec1.cpp -c -I./include -O3 -march=native -ta=nvidia:cuda11.2 -fPIC
$ nvc++ -shared -o libvec1.so vec1.o
$ nvc++ vec1_test_gpu.cpp -I./include -O3 -march=native -ta=nvidia:cuda11.2 -L./ -lvec1
错误消息出现在最后一个命令之后,并读取nvlink error : Undefined reference to '_ZN4Vec1mlEd' in '/tmp/nvc++jOtCBiT_m38d.o'
这里的问题是您试图调用一个设备例程,"Vec1::运算符*";,它包含在主程序内核的共享对象中。nvc++的OpenACC实现使用CUDA来针对NVIDIA设备。由于CUDA没有用于设备代码的动态链接器,至少目前还没有,因此不支持这样做。
你需要静态地链接它,或者移动";并行环路";到共享对象中。
注意-ta";标志已被弃用。请考虑使用"-acc-gpu=cuda1.2〃;相反