链接时CUDA多重定义错误

CUDA multiple definition error during linking

本文关键字:定义 错误 CUDA 链接      更新时间:2023-10-16

我开始在我的一个项目中使用CUDA和thrust,所以我对它的一些方面仍然是新的。我遇到了以下问题。

我必须分开。cu文件,我想使用一个共同的头文件,有一个结构体,他们都应该能够使用。它是这样的:

////////Global.h
#ifndef global_h
#define global_h
struct globalParam
{
    uint64_t spaceToUse;
    globalParams() : spaceToUse(1024*1024*1024) {}
};
globalParam glParam;
#endif

第一个.cu文件看起来像这样:

////////firstcufile.cu
#ifndef firstcufile_cu
#define firstcufile_cu
#include "Global.h"
template<typename T>
QVector<T> GPUcalculateNormSq(const QVector<T>& real, const QVector<T>& imag)
{
    QVector<T> result (real.size());
    uint64_t maxSpace = glParam.spaceToUse;
    //Some Code to use thrust and using tops maxSpace bytes.
    return result;
}
template QVector<float> GPUcalculateNormSq(const QVector<float>& real, const QVector<float>& imag);
template QVector<double> GPUcalculateNormSq(const QVector<double>& real, const QVector<double>& imag);
#endif

第二个.cu文件如下所示:

////////secondcufile.cu
#ifndef secondcufile_cu
#define secondcufile_cu
#include "Global.h"
template<typename T>
double getMean(const T& vec)
{
    uint64_t spaceNeededOnGPU = vec.size() * sizeof (T);
    uint64_t maxSpace = glParam.spaceToUse;
    //Some code to make sure tops maxSpace bytes on GPU
    double sum = thrust::reduce(std::begin(vec), std::end(vec));
    return sum / vec.size();
}
template double getMean(const QVector<float>& vec);
#endif

现在得到的错误是:

secondcufilecuda_d.o:(.bss+0x18): multiple definition of `glParam'
firstcufilecuda_d.o:(.bss+0x18): first defined here

上面的函数看起来很相似,但那是因为我试图使它们尽可能简单。可以将所有内容写入一个.cu文件中,但如果可能的话,我希望将其拆分。

链接器哪里做错了?我从一个Qt Creator项目内编译和链接。让我知道如果你需要我的行从.pro文件知道我如何使用nvcc编译器。

这是由于Global.h多次被包含,每次被包含它都提供:globalParam glParam;。这不是前向声明(即不仅仅是类型签名),而是相当于globalParam结构体的实际实例化。然后,这会导致两个变量都命名为glParam(每个对应于Global.h的单独#include),并且会给您带来多个定义错误。

快速修复:尝试使用extern,如果你想共享你的全局变量(然后链接器知道它只是一个引用,好吧,一个'外部'符号)。

更好的修复:考虑重构你的代码,通过引用或指针作为函数的实参传递全局形参。这样,您甚至不必在头文件中声明glParam变量,绕过整个问题,使您的代码更容易理解/推理。