在GTX Titan卡中重叠内核执行和数据传输的最佳策略是什么?

时间:2013-07-10 07:29:28

标签: cuda

当我尝试重叠数据传输和内核执行时,无论我使用什么流,卡似乎都按顺序执行所有内存传输

所以,如果我发出以下内容:

  • 流1: MemcpyA_HtoD_1; Kernel_1; MemcpyA_DtoH_1
  • stream 2: MemcpyA_HtoD_2; Kernel_2; MemcpyA_DtoH_2

MemcpyA_HtoD_2将等到MemcpyA_DtoH_1完成。因此没有实现重叠。无论我使用什么配置的流,Memcpy操作总是按顺序发布。因此,实现重叠的唯一方法包括缓冲输出或延迟输出传输直到下一次迭代。

我使用CUDA 5.5,Windows 7 x64和GTX Titan。所有cpu内存都被固定,data_transfers使用异步版本完成。

请参阅以下屏幕,其中包含以下行为:

发布,host_to_device - >内核 - > device_to_host(正常行为)并且无法重叠。

non overlapping

发出host_to_device - >内核(避免内核之后的device_to_host)重叠...因为无论我尝试什么流配置,所有内存副本都按顺序执行。

overlapping

更新

如果有人有兴趣重现这个问题,我编写了一个合成程序,显示出这种不良行为。它是使用CUDA 5.5的完整VS2010解决方案

VS2010 Streams Not Working link

有人可以在linux上执行此操作以测试重叠吗?

#include "cuda_runtime.h"
#include "device_launch_parameters.h"


#include <stdio.h>
#define N 1024*1024

__global__ void someKernel(int *d_in, int *d_out) {
    for (int i = threadIdx.x; i < threadIdx.x + 1024; i++) {
        d_out[i] = d_in[i];
    }
}

int main () {
    int *h_bufferIn[100];
    int *h_bufferOut[100];
    int *d_bufferIn[100];
    int *d_bufferOut[100];

    //allocate some memory
    for (int i = 0; i < 100; i++) {
        cudaMallocHost(&h_bufferIn[i],N*sizeof(int));
        cudaMallocHost(&h_bufferOut[i],N*sizeof(int));
        cudaMalloc(&d_bufferIn[i], N*sizeof(int));
        cudaMalloc(&d_bufferOut[i], N*sizeof(int));
    }

    //create cuda streams
    cudaStream_t st[2];
    cudaStreamCreate(&st[0]);
    cudaStreamCreate(&st[1]);

    //trying to overlap computation and memcpys
    for (int i = 0; i < 100; i+=2) {
        cudaMemcpyAsync(d_bufferIn[i], h_bufferIn[i], N*sizeof(int), cudaMemcpyHostToDevice, st[i%2]);
        someKernel<<<1,256, 0, st[i%2]>>>(d_bufferIn[i], d_bufferOut[i]);
        cudaMemcpyAsync(h_bufferOut[i], d_bufferOut[i], N*sizeof(int), cudaMemcpyDeviceToHost, st[i%2]);
        cudaStreamQuery(0);

        cudaMemcpyAsync(d_bufferIn[i+1], h_bufferIn[i+1], N*sizeof(int), cudaMemcpyHostToDevice, st[(i+1)%2]);
        someKernel<<<1,256, 0, st[(i+1)%2]>>>(d_bufferIn[i+1], d_bufferOut[i+1]);
        cudaMemcpyAsync(h_bufferOut[i+1], d_bufferOut[i+1], N*sizeof(int), cudaMemcpyDeviceToHost, st[(i+1)%2]);
        cudaStreamQuery(0);
    }
    cudaDeviceSynchronize();
}

1 个答案:

答案 0 :(得分:1)

  

TL; DR:问题是由Nsight Monitor中的WDDM TDR延迟选项引起的!设置为false时,会出现问题。相反,如果你设置了   TDR延迟值到一个非常高的数字,&#34;启用&#34;选项   是的,问题就消失了。

请阅读下面的其他(较旧)步骤,直到我找到上述解决方案,以及其他一些可能的原因。

我最近才能部分解决这个问题!我认为它特定于windows和aero。请尝试这些步骤并发布结果以帮助其他人!我在GTX 650和GT 640上尝试过它。

  

在您执行任何操作之前,请考虑using both onboard gpu(as display) and the discrete gpu (for computations),因为nvidia驱动程序已经验证了Windows的问题!当你使用板载gpu时,说驱动程序没有满载,所以很多bug都被规避了。此外,在工作时保持系统响应能力!

  1. 确保您的并发问题与旧驱动程序(包括BIOS)等其他问题无关。
  2. 转到计算机&gt;属性
  3. 选择左侧的高级系统设置
  4. 转到“高级”标签
  5. 关于效果点击设置
  6. 在“视觉效果”选项卡中,选择&#34;调整以获得最佳性能&#34;子弹。
  7. 这将禁用空气动力学和几乎所有视觉效果。如果此配置有效,您可以尝试逐个启用视觉效果框,直到找到导致问题的精确框!

      

    或者,您可以:

    1. 右键单击桌面,选择个性化
    2. 从基本主题中选择一个没有空气动力学的主题。
    3. 这也可以如上所述,但启用了更多可视化选项。对于我的两个设备,此设置也有效,所以我保留了它。

        

      当您尝试这些解决方案时,请回到此处并发布您的发现!

      对我来说, 解决了大多数情况下的问题(我做过平铺的dgemm) ,但请注意我仍然无法运行&#34 ; simpleStreams&#34;正确并实现并发......

        

      更新 <问题通过新的Windows安装完全解决 !!前面的步骤改进了某些情况下的行为,但只有全新安装解决了所有问题!

      我会尝试找一种解决这个问题的不那么激进的方法,也许只需恢复注册表即可。