为了练习使用 CUDA 编码,我做了一个小测试场景,其中包含三个文件:
memory.c
持有纯C
代码memory_kernels.h
CUDA 内核的声明和启动内核的函数memory_kernels.cu
内核的定义
程序应该做的是在主机上创建一个整数数组,将其复制到设备并查询元素。内核将打印出一些细节。
但是,我收到错误:
Error in memory_kernels.cu at line 43 with error code "unspecified launch failure"
三个文件的源代码如下:
/**
* memory.c
*
* Test copying large arrays to device
* and printing from kernel
*/
/* Include standard libraries */
#include <stdlib.h>
#include <stdio.h>
/* Include local header files */
#include "memory_kernels.h"
int main() {
/* Size of array */
int i, N = 1024;
/* Array */
int *intArr = (int *) malloc( N * sizeof(int) );
/* Fill array */
for( i = 0; i < N; i++ ) {
intArr[i] = i;
}
/* Run CUDA code */
cuda_mem( &intArr );
/* Clean up device */
cudaDeviceReset();
/* Everything done */
exit(EXIT_SUCCESS);
}
/**
* memory_kernels.h
*
* Declarations for CUDA kernels
*/
/* Determine compiler */
#ifdef __cplusplus
#define EXTCFUNC extern "C"
#else
#define EXTCFUNC extern
#endif
#ifndef KERNELS_H
#define KERNELS_H
/* Standard libraries (only needed for debugging) */
#include <stdio.h>
/* Include CUDA header files */
#include <cuda.h>
#include <cuda_runtime.h>
#define CUDA_CALL(x) do { if((x) != cudaSuccess) { \
printf("Error in %s at line %d with error code \"%s\"\n",__FILE__,__LINE__,cudaGetErrorString(x)); \
exit(x);}} while(0)
/* Device globals */
__device__ int *d_intArr;
/* Device kernels */
__global__ void mem();
/* Host access functions */
EXTCFUNC void cuda_mem( int **intArr );
#endif
/**
* memory_kernels.cu
*
* CUDA kernel implementations
*/
/* Include header file */
#include "memory_kernels.h"
__global__ void mem() {
int i = threadIdx.x;
int a = d_intArr[i];
printf("i = %d a = %d\n",i,a);
}
/* Determine compiler */
#ifdef __cplusplus
#define EXTCFUNC extern "C"
#else
#define EXTCFUNC extern
#endif
/**
* cuda_mem()
*
* Test copying large array to device
* and printing from kernel
*/
EXTCFUNC void cuda_mem( int **intArr ) {
/* Local variables */
int N = 1024;
/* Initialise device variables */
CUDA_CALL( cudaMalloc( (void **) &d_intArr, sizeof(int) * N ) );
/* Copy to device initial values */
CUDA_CALL( cudaMemcpy( d_intArr, *intArr, sizeof(int) * N, cudaMemcpyHostToDevice ) );
/* Run kernel */
mem <<< 1,N >>> ();
CUDA_CALL( cudaPeekAtLastError() );
CUDA_CALL( cudaDeviceSynchronize() );
/* Free local scoped dynamically allocated memory */
CUDA_CALL( cudaFree( d_intArr ) );
}
使用以下命令完成编译:
nvcc -c -o memory.o memory.c -arch=sm_20
nvcc -c -o memory_kernels.o memory_kernels.cu -arch=sm_20
nvcc -o memory memory.o memory_kernels.o -arch=sm_20
并在带有 CUDA 4.0 的 NVIDIA Tesla M2050 上运行。计算能力 2.0 需要printf()
在内核中使用。
在四处寻找解决方案后,错误代码表明我在从全局内存读取时内核中存在分段错误。但是,我启动的线程数与数组的大小相同。
经过试验,我感觉错误是在复制intArr
到设备时引起的。也许我把我的指针都搞混了?
我了解文件结构是否有点奇怪,但它都是更大程序的一部分,但我已将错误减少到这个较小的情况。