原文地址:http://www.isnowfy.com/introduction-to-cuda/
CUDA的全称是Compute Unified Device Architecture, 是NVIDIA® 公司的并行计算架构,主要是要来利用GPU的计算能力,来提高计算性能。进一步的说是在GPU上提供标准C编程语言,为在支持CUDA的NVIDIA GPU上进行并行计算而提供了统一的软硬件解决方案。
为什么要用GPU来计算呢,CPU不如GPU吗,这就要从CPU和GPU的架构说起了。
图片中绿色的是ALU(运算器),可以看到,CPU只有4个ALU,而GPU中的ALU多很多,并且GPU中把更多的晶体管用于数据处理方面。之所以这样是因为CPU和GPU的功能不一样,CPU要考虑上下文切换等所以需要复杂的控制单元和缓存来提高执行效率,而GPU不需要那样复杂的控制逻辑,所以GPU可以通过增加并行处理单元和存储控制单元来提供处理能力和存储器带宽。因而同期的GPU的计算能力会比CPU高很多个档次。为了充分利用GPU的运算能力,就有了现在的CUDA编程。
但是利用GPU运算有几个缺点。
所以如果你的计算内容不能高度并行化的话那么就不要去考虑用GPU来加快你的计算了。
--------------简介完毕,下面进行配置开发环节----------------
那么我们来搞一下CUDA开发环境的搭建。首先我是在windows下搞的,本来想去linux下搞,但是CUDA对gcc的版本有要求,不能太高,比较囧。
基本配置完成,之后我们在vs中选择新建项目,会在visual c++中看到cuda一项,选CUDAWinApp来新建一个CUDA的程序。建好之后,我们会看到一个sample.cu,代码如下
1 /******************************************************************** 2 3 4 * sample.cu 5 6 7 * This is a example of the CUDA program. 8 9 10 *********************************************************************/ 11 12 13 #include <stdio.h> 14 15 16 #include <stdlib.h> 17 18 19 #include <cuda_runtime.h> 20 21 22 //#include <device_launch_parameters.h> 23 24 25 /************************************************************************/ 26 27 28 /* Init CUDA */ 29 30 31 /************************************************************************/ 32 33 34 bool InitCUDA(void){ 35 36 37 int count = 0; 38 39 40 int i = 0; 41 42 43 cudaGetDeviceCount(&count); 44 45 46 if(count == 0) { 47 48 49 fprintf(stderr, "There is no device.\n"); 50 51 52 return false; 53 54 55 } 56 57 58 for(i = 0; i < count; i++) { 59 60 61 cudaDeviceProp prop; 62 63 64 if(cudaGetDeviceProperties(&prop, i) == cudaSuccess) { 65 66 67 if(prop.major >= 1) { 68 69 70 break; 71 72 73 } 74 75 76 } 77 78 79 } 80 81 82 if(i == count) { 83 84 85 fprintf(stderr, "There is no device supporting CUDA.\n"); 86 87 88 return false; 89 90 91 } 92 93 94 cudaSetDevice(i); 95 96 97 printf("CUDA initialized.\n"); 98 99 100 return true; 101 102 103 } 104 105 106 /************************************************************************/ 107 108 109 /* Example */ 110 111 112 /************************************************************************/ 113 114 115 __global__ static void HelloCUDA(char* result, int num){ 116 117 118 int i = threadIdx.x; 119 120 121 char p_HelloCUDA[] = "Hello CUDA!"; 122 123 124 if(i < num) result[i] = p_HelloCUDA[i]; 125 126 127 } 128 129 130 /************************************************************************/ 131 132 133 /* HelloCUDA */ 134 135 136 /************************************************************************/ 137 138 139 int main(int argc, char* argv[]){ 140 141 142 if(!InitCUDA()) { 143 144 145 return 0; 146 147 148 } 149 150 151 char *device_result = 0; 152 153 154 char host_result[12] ={0}; 155 156 157 cudaMalloc((void**) &device_result, sizeof(char) * 11); 158 159 160 HelloCUDA<<<1, 11, 0>>>(device_result, 11); 161 162 163 cudaMemcpy(host_result, device_result, sizeof(char) * 11, cudaMemcpyDeviceToHost); 164 165 166 printf("%s\n", host_result); 167 168 169 cudaFree(device_result); 170 171 172 return 0; 173 174 175 }
ctrl+F5运行,我们就能看到Hello CUDA!字样了。
让我们简单看一下这个程序,在CUDA编程中用__global__来表示这是个可以有CPU调用的运行在GPU上的一个函数。由于GPU访问数据都是在显存中获取的,所以我们需要调用cudaMalloc在显存中开辟空间,用法类似于在内存中开辟空间的malloc。之后调用函数HelloCUDA<<<1, 11, 0>>>(device_result, 11);尖括号中有三项,分别表示block的数量,thread的数量,和共享内存的大小。用threadIdx.x我们可以获取thread的编号,来达到并行的效果。最后我们需要调用cudaMemcpy把数据从显存中取回到内存中,并释放显存中的数据,这样cpu就可以获得最后的结果了。
前面提到了block还有thread,我们据此继续讲解一下CUDA的编程架构。
用__global__修饰的函数被称作一个kernel函数或者是叫做一个kernel,一个kernel会有一个grid(目前是,据说以后可能会是多个),一个grid包含多个block,一个block包含多个thread,thread就类似于平常cpu运算中的线程thread。这样的排布是因为我们需要不同粒度的并行,处于同一个block的thread可以通过共享内存来交换数据,而不同block的thread不能直接共享数据。为了能更方便的运算一二三维的数组,grid中的block或者是block中的thread都可以按照不同的维度来排布,比如如果block中的thread按照二维排布的话,我们可以通过threadIdx.x和threadIdx.y来获得当前thread的位置,这样的结构使得运算二维或三维数组更加自然,而且可以避免除法和取余操作,加快了速度。
那基本介绍就这样了,如果还想深入了解的话还是需要去看一下相关的书籍和文档才可以。
参考资料: 《深入浅出CUDA》 《CUDA 编程指南4.0》 《GPU高性能运算之CUDA》