▶ 使用函数 cudaMallocPitch() 和配套的函数 cudaMemcpy2D() 来使用二维数组。C 中二维数组内存分配是转化为一维数组,连贯紧凑,每次访问数组中的元素都必须从数组首元素开始遍历;而 cuda 中这样分配的二维数组内存保证了数组每一行首元素的地址值都按照 256 或 512 的倍数对齐,提高访问效率,但使得每行末尾元素与下一行首元素地址可能不连贯,使用指针寻址时要注意考虑尾部。

 // cuda_rumtime_api.h
extern __host__ cudaError_t CUDARTAPI cudaMallocPitch(void **devPtr, size_t *pitch, size_t widthByte, size_t height); extern __host__ cudaError_t CUDARTAPI cudaMemcpy2D(void *dst, size_t dpitch, const void *src, size_t spitch, size_t width, size_t height, enum cudaMemcpyKind kind);

● cudaMAllocPitch() 传入存储器指针 **devPtr,偏移值的指针 *pitch,数组行字节数 widthByte,数组行数 height。函数返回后指针指向分配的内存(每行地址对齐到 AlignByte 字节,为 256B 或 512B),偏移值指针指向的值为该行实际字节数(= sizeof(datatype) * width + alignByte - 1) / alignByte)。

● cudaMemcpy2D() 传入目标存储器的指针 *dst,目标存储器行字节数 dpitch,源存储器指针 *src,源存储器行字节数 spitch,数组行字节数 width,数组行数 height,拷贝方向 kind。这里要求存储器行字节数不小于数组行字节数,多出来的部分就是每行尾部空白部分。

● 整个测试代码。

 #include <stdio.h>
#include <malloc.h>
#include <cuda_runtime_api.h>
#include "device_launch_parameters.h" __global__ void myKernel(float* devPtr, int height, int width, int pitch)
{
int row, col;
float *rowHead; for (row = ; row < height; row++)
{
rowHead = (float*)((char*)devPtr + row * pitch); for (col = ; col < width; col++)
{
printf("\t%f", rowHead[col]);// 逐个打印并自增 1
rowHead[col]++;
}
printf("\n");
}
} int main()
{
size_t width = ;
size_t height = ;
float *h_data, *d_data;
size_t pitch; h_data = (float *)malloc(sizeof(float)*width*height);
for (int i = ; i < width*height; i++)
h_data[i] = (float)i; printf("\n\tAlloc memory.");
cudaMallocPitch((void **)&d_data, &pitch, sizeof(float)*width, height);
printf("\n\tPitch = %d B\n", pitch); printf("\n\tCopy to Device.\n");
cudaMemcpy2D(d_data, pitch, h_data, sizeof(float)*width, sizeof(float)*width, height, cudaMemcpyHostToDevice); myKernel << <, >> > (d_data, height, width, pitch);
cudaDeviceSynchronize(); printf("\n\tCopy back to Host.\n");
cudaMemcpy2D(h_data, sizeof(float)*width, d_data, pitch, sizeof(float)*width, height, cudaMemcpyDeviceToHost); for (int i = ; i < width*height; i++)
{
printf("\t%f", h_data[i]);
if ((i + ) % width == )
printf("\n");
} free(h_data);
cudaFree(d_data); getchar();
return ;
}

● 输出结果:

        Alloc memory.
Pitch = B Copy to Device.
0.000000 1.000000 2.000000 3.000000 4.000000 5.000000
6.000000 7.000000 8.000000 9.000000 10.000000 11.000000
12.000000 13.000000 14.000000 15.000000 16.000000 17.000000
18.000000 19.000000 20.000000 21.000000 22.000000 23.000000
24.000000 25.000000 26.000000 27.000000 28.000000 29.000000 Copy back to Host.
1.000000 2.000000 3.000000 4.000000 5.000000 6.000000
7.000000 8.000000 9.000000 10.000000 11.000000 12.000000
13.000000 14.000000 15.000000 16.000000 17.000000 18.000000
19.000000 20.000000 21.000000 22.000000 23.000000 24.000000
25.000000 26.000000 27.000000 28.000000 29.000000 30.000000

▶ 使用函数 cudaMalloc3D() 和配套的函数 cudaMemcpy3D() 来使用三维数组。因为涉及的参数较多,需要定义一些用来传参的结构,形式上和二维数组的使用有较大差距,不好看。

● 涉及的相关代码

 // driver_types.h
struct cudaArray; // cuda 数组
typedef struct cudaArray * cudaArray_t;// cuda 指针 struct __device_builtin__ cudaPitchedPtr
{
void *ptr; // 实际数组指针(用完后要用 cudaFree() 释放掉)
size_t pitch; // 数组行字节数
size_t xsize; // 数组列数
size_t ysize; // 数组行数
}; struct __device_builtin__ cudaExtent
{
size_t width; // 数组行字节数
size_t height; // 数组行数
size_t depth; // 数组层数
}; struct __device_builtin__ cudaPos
{
size_t x;
size_t y;
size_t z;
}; struct __device_builtin__ cudaMemcpy3DParms
{
cudaArray_t srcArray; // 原数组指针
struct cudaPos srcPos; // 原数组偏移
struct cudaPitchedPtr srcPtr; // ?Pitched source memory address cudaArray_t dstArray; // 目标数组指针
struct cudaPos dstPos; // 目标数组偏移
struct cudaPitchedPtr dstPtr; // ?Pitched destination memory address struct cudaExtent extent; // 数组实际尺寸(去掉对齐用的空白部分)
enum cudaMemcpyKind kind; // 拷贝类型
}; // driver_functions.h
static __inline__ __host__ struct cudaPitchedPtr make_cudaPitchedPtr(void *d, size_t p, size_t xsz, size_t ysz)
{ // 简单生成 cudaPitchedPtr 结构的方法
struct cudaPitchedPtr s; s.ptr = d;
s.pitch = p;
s.xsize = xsz;
s.ysize = ysz; return s;
} static __inline__ __host__ struct cudaPos make_cudaPos(size_t x, size_t y, size_t z)
{ // 简单的生成 cudaPos 结构的方法
struct cudaPos p; p.x = x;
p.y = y;
p.z = z; return p;
} static __inline__ __host__ struct cudaExtent make_cudaExtent(size_t w, size_t h, size_t d)
{ // 简单的生成 cudaExtent 结构的方法
struct cudaExtent e; e.width = w;
e.height = h;
e.depth = d; return e;
} // cuda_runtime_api.h
extern __host__ cudaError_t CUDARTAPI cudaMalloc3D(struct cudaPitchedPtr* pitchedDevPtr, struct cudaExtent extent); extern __host__ cudaError_t CUDARTAPI cudaMemcpy3D(const struct cudaMemcpy3DParms *p);

● 完整的测试程序

 #include <stdio.h>
#include <malloc.h>
#include <cuda_runtime_api.h>
#include "device_launch_parameters.h"
#include <driver_functions.h> __global__ void myKernel(cudaPitchedPtr devPitchedPtr, cudaExtent extent)
{
float * devPtr = (float *)devPitchedPtr.ptr;
float *sliceHead, *rowHead;
// 可以定义为 char * 作面、行迁移的时候直接加减字节数,取行内元素的时候再换回 float * for (int z = ; z < extent.depth; z++)
{
sliceHead = (float *)((char *)devPtr + z * devPitchedPtr.pitch * extent.height);
for (int y = ; y < extent.height; y++)
{
rowHead = (float*)((char *)sliceHead + y * devPitchedPtr.pitch);
for (int x = ; x < extent.width / sizeof(float); x++)// extent 存储的是行有效字节数,要除以元素大小
{
printf("\t%f",rowHead[x]);// 逐个打印并自增 1
rowHead[x]++;
}
printf("\n");
}
printf("\n");
}
} int main()
{
size_t width = ;
size_t height = ;
size_t depth = ;
float *h_data; cudaPitchedPtr d_data;
cudaExtent extent;
cudaMemcpy3DParms cpyParm; h_data = (float *)malloc(sizeof(float) * width * height * depth);
for (int i = ; i < width * height * depth; i++)
h_data[i] = (float)i; printf("\n\tAlloc memory.");
extent = make_cudaExtent(sizeof(float) * width, height, depth);
cudaMalloc3D(&d_data, extent); printf("\n\tCopy to Device.\n");
cpyParm = {};
cpyParm.srcPtr = make_cudaPitchedPtr((void*)h_data, sizeof(float) * width, width, height);
cpyParm.dstPtr = d_data;
cpyParm.extent = extent;
cpyParm.kind = cudaMemcpyHostToDevice;
cudaMemcpy3D(&cpyParm); myKernel << <, >> > (d_data, extent);
cudaDeviceSynchronize(); printf("\n\tCopy back to Host.\n");
cpyParm = { };
cpyParm.srcPtr = d_data;
cpyParm.dstPtr = make_cudaPitchedPtr((void*)h_data, sizeof(float) * width, width, height);
cpyParm.extent = extent;
cpyParm.kind = cudaMemcpyDeviceToHost;
cudaMemcpy3D(&cpyParm); for (int i = ; i < width*height*depth; i++)
{
printf("\t%f", h_data[i]);
if ((i + ) % width == )
printf("\n");
if ((i + ) % (width*height) == )
printf("\n");
} free(h_data);
cudaFree(d_data.ptr);
getchar();
return ;
}

● 输出结果:

        Alloc memory.
Copy to Device.
0.000000 1.000000
2.000000 3.000000
4.000000 5.000000 6.000000 7.000000
8.000000 9.000000
10.000000 11.000000 12.000000 13.000000
14.000000 15.000000
16.000000 17.000000 18.000000 19.000000
20.000000 21.000000
22.000000 23.000000 Copy back to Host.
1.000000 2.000000
3.000000 4.000000
5.000000 6.000000 7.000000 8.000000
9.000000 10.000000
11.000000 12.000000 13.000000 14.000000
15.000000 16.000000
17.000000 18.000000 19.000000 20.000000
21.000000 22.000000
23.000000 24.000000

二维数组 cudaMallocPitch() 和三维数组 cudaMalloc3D() 的使用的更多相关文章

  1. c# 基础之数组(包含三维数组)

    public enum ChessType { White = , None=, Black=, } class Program { static void Main(string[] args) { ...

  2. 【opencv】 solvepnp 和 solvepnpRansac 求解 【空间三维坐标系 到 图像二维坐标系】的 三维旋转R 和 三维平移 T 【opencv2使用solvepnp求解rt不准的问题】

    参考: pnp问题 与 solvepnp函数:https://www.jianshu.com/p/b97406d8833c 对图片进行二维仿射变换cv2.warpAffine() or 对图片进行二维 ...

  3. PHP 把MYSQL重复ID 二维数组重组为三维数组

    应用场景 MYSQL在使用关联查询时,比如 产品表 与 产品图片表关联,一个产品多张产品图片,关联查询结果如下: $arr=[['id'=>1,'img'=>'img1'],['id'=& ...

  4. C语言之二维数组

    二维数组 还是一个数组,只不过数组中得每一个元素又是一个数组 1). 声明语法 类型 数组名[行][列]; 例:  int nums[2][3];//2行3列的二维数组,保存的数据类型是int类型 c ...

  5. C语言数组篇(五)多级指针和二维数组指针的区别

    多级指针   以二级指针为例 二级指针的由来是 指针数组 的指针形式. int *p[10] 读取的顺序是 p[] --> 10个空间的数组 * p[] --> 这10个空间的数组里面存放 ...

  6. [poj2155]Matrix(二维树状数组)

    Matrix Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 25004   Accepted: 9261 Descripti ...

  7. JavaScript -- 定义二维数组

    方法一:直接定义并且初始化,这种遇到数量少的情况可以用var _TheArray = [["0-1","0-2"],["1-1"," ...

  8. js二维数组定义和初始化的三种方法总结

    js二维数组定义和初始化的三种方法总结 方法一:直接定义并且初始化,这种遇到数量少的情况可以用var _TheArray = [["0-1","0-2"],[& ...

  9. [Swift]多维数组的表示和存储:N维数组映射到一维数组(一一对应)!

    数组:有序的元素序列. 若将有限个类型相同的变量的集合命名,那么这个名称为数组名.组成数组的各个变量称为数组的分量,也称为数组的元素,有时也称为下标变量.用于区分数组的各个元素的数字编号称为下标.数组 ...

随机推荐

  1. HDU 1233:还是畅通工程(最小生成树)

    还是畅通工程 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Subm ...

  2. Centos(linux)下的Python

    Centos(linux)下安装python3(python2和python3共存) yum -y install lrzsz 首先安装lrzsz工具,lrzsz是一款在linux里可代替ftp上传和 ...

  3. 结构体内的函数与bfs的情景变量

    关于结构体内的函数,太难的尚且不会用,下面是一个简单一点的结构体内函数的代码 定义这样一个结构体函数之后就能像如下这样使用了 以上为结构体内的简单函数,下面重点来了,关于bfs找最短路由于需要避免走回 ...

  4. Where is Silverlight now?

    Some time ago, I wrote an article about the comparison between HTML5 and Silverlight. That article w ...

  5. JVM 详解

    概念 数据类型 Java 虚拟机中,数据类型可以分为两类:基本类型和引用类型.基本类型的变量保存原始值,即:他代表的值就是数值本身:而引用类型的变量保存引用值.“引用值”代表了某个对象的引用,而不是对 ...

  6. JUC集合之 CopyOnWriteArrayList

    CopyOnWriteArrayList介绍 它相当于线程安全的ArrayList.和ArrayList一样,它是个可变数组:但是和ArrayList不同的时,它具有以下特性: 它最适合于具有以下特征 ...

  7. 二、Jmeter录制脚本过程及Could not create script recorder报错、您的连接不是私密连接报错

    两个报错:Could not create script recorder报错和您的连接不是私密连接报错 1.录制过程 * 打开jmeter * 点击Templated,选择Recoding模版 * ...

  8. go test 初始化--- TestMain的使用

    go test 功能,提高了开发和测试的效率. 有时会遇到这样的场景: 进行测试之前需要初始化操作(例如打开连接),测试结束后,需要做清理工作(例如关闭连接)等等.这个时候就可以使用TestMain( ...

  9. centos7下svn的安装与配置

    1.环境 centos7 2.安装svnyum -y install subversion 3.配置 建立版本库目录mkdir /www/svndata svnserve -d -r /www/svn ...

  10. Mac 平台安装 Android Studio 集成 Android SDK

    最近写个小工具,给周边一些朋友用用(类似任务执行),一开始 任务执行用Python写的,部署在云端,有一些局限(资金开销):现准备写一个任务端,需要用到的人直接装个客户端就行,于是准备写个客户端版本的 ...