我的第一个CUDA程序

MatAdd算法

实现两个矩阵对应元素相加

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>

// 矩阵加法函数
void MatAdd(int height, int width)
{
    // 在主机内存中为 A、B 和 C 分配内存
    float* A = (float*)malloc(height * width * sizeof(float));
    float* B = (float*)malloc(height * width * sizeof(float));
    float* C = (float*)malloc(height * width * sizeof(float));

    // 初始化输入矩阵 A 和 B
    for (int i = 0; i < height; i++)
    {
        for (int j = 0; j < width; j++)
        {
            // 例如,初始化为 i+j 的和
            *(A + i * width + j) = i + j;
            *(B + i * width + j) = i - j;
        }
    }

    // 执行矩阵加法运算,将结果存储在矩阵 C 中
    for (int i = 0; i < height; i++)
    {
        for (int j = 0; j < width; j++)
        {
            *(C + i * width + j) = *(A + i * width + j) + *(B + i * width + j);
        }
    }

    // 输出结果矩阵 C
    printf("Matrix C (result of A + B):\n");
    for (int i = 0; i < height; i++)
    {
        for (int j = 0; j < width; j++)
        {
            printf("%f ", *(C + i * width + j));
        }
        printf("\n");
    }

    // 释放分配的内存
    free(A);
    free(B);
    free(C);
}

int main()
{
    int height = 3; // 矩阵的高度
    int width = 3;  // 矩阵的宽度

    // 执行矩阵加法
    MatAdd(height, width);

    return 0;
}

输出:

cpp 复制代码
Matrix C (result of A + B):
0.000000 0.000000 0.000000 
2.000000 2.000000 2.000000 
4.000000 4.000000 4.000000 

MatAdd算法的GPU实现

MatAdd算法的GPU实现

  • CPU端为输入矩阵A和B、输出矩阵C分配空间,并进行初始化
  • CPU端分配设备端内存,并将A和B传输到GPU上
  • 定义数据和线程的映射关系,并确定线程的开启数量和组织方式
  1. 每个线程负责输出矩阵C的一个元素的计算,全局ID为(i,j)的线程计算索引为(i,j)的矩阵元素
  2. 当矩阵规模为width*height时,共开启width * height个线程
  3. 每个block包含256个线程,采用(16,16)的组织形式
  • 编写计算kernel,完成计算任务
  • CPU端将计算结果从Device内存拷贝到Host内存

内存分配 数据传输

开启线程 启动kernel 结果返回

GPU kernel

cpp 复制代码
#include <stdio.h>
#include <cuda_runtime.h>

// CUDA 核函数,用于矩阵加法
__global__ void MatAddKernel(float* A, float* B, float* C, int height, int width) {
    // 获取线程的全局ID
    int i = blockIdx.x * blockDim.x + threadIdx.x; // 计算全局行索引
    int j = blockIdx.y * blockDim.y + threadIdx.y; // 计算全局列索引

    // 确保索引在矩阵范围内
    if (i < width && j < height) {
        // 计算当前线程对应的元素索引
        int index = j * width + i;

        // 从矩阵 A 和 B 中读取数据
        float src_data_A = A[index];
        float src_data_B = B[index];

        // 执行加法运算
        float result = src_data_A + src_data_B;

        // 将结果写入矩阵 C
        C[index] = result;
    }
}

void MatAdd(int height, int width) {
    // 在主机内存中分配 A、B 和 C
    float* A = (float*)malloc(height * width * sizeof(float));
    float* B = (float*)malloc(height * width * sizeof(float));
    float* C = (float*)malloc(height * width * sizeof(float));

    // 初始化输入矩阵 A 和 B
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            A[i * width + j] = i + j; // 简单初始化,A的元素为行索引+列索引
            B[i * width + j] = i - j; // 简单初始化,B的元素为行索引-列索引
        }
    }

    // 第一步:在设备内存中为矩阵 A、B 和 C 分配内存
    float* d_A;
    cudaMalloc(&d_A, height * width * sizeof(float)); // 分配矩阵 A 的设备内存
    float* d_B;
    cudaMalloc(&d_B, height * width * sizeof(float)); // 分配矩阵 B 的设备内存
    float* d_C;
    cudaMalloc(&d_C, height * width * sizeof(float)); // 分配矩阵 C 的设备内存

    // 第二步:将矩阵 A 和 B 从主机内存复制到设备内存
    cudaMemcpy(d_A, A, height * width * sizeof(float), cudaMemcpyHostToDevice); // 复制 A
    cudaMemcpy(d_B, B, height * width * sizeof(float), cudaMemcpyHostToDevice); // 复制 B

    // 第三步:调用 CUDA 核函数
    dim3 threadsPerBlock(16, 16); // 定义每个块中的线程数
    dim3 numBlocks((width + threadsPerBlock.x - 1) / threadsPerBlock.x, 
                   (height + threadsPerBlock.y - 1) / threadsPerBlock.y); // 计算网格中的块数
    MatAddKernel<<<numBlocks, threadsPerBlock>>>(d_A, d_B, d_C, height, width); // 启动 CUDA 核函数

    // 第四步:将结果从设备内存复制回主机内存
    cudaMemcpy(C, d_C, height * width * sizeof(float), cudaMemcpyDeviceToHost);

    // 输出结果矩阵 C
    printf("Matrix C (result of A + B):\n");
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            printf("%f ", C[i * width + j]);
        }
        printf("\n");
    }

    // 释放设备内存
    cudaFree(d_A);
    cudaFree(d_B);
    cudaFree(d_C);

    // 释放主机内存
    free(A);
    free(B);
    free(C);
}

int main() {
    int height = 3; // 矩阵的高度
    int width = 3;  // 矩阵的宽度

    // 调用矩阵加法函数
    MatAdd(height, width);

    return 0;
}

输出:

cpp 复制代码
Matrix C (result of A + B):
0.000000 0.000000 0.000000 
2.000000 2.000000 2.000000 
4.000000 4.000000 4.000000 

CUDA程序编译

CUDA程序性能测试

使用 CUDA GPU Timers 实际要循环100次求平均值

cpp 复制代码
#include <stdio.h>
#include <cuda_runtime.h>

// CUDA 核函数,用于矩阵加法
__global__ void MatAddKernel(float* A, float* B, float* C, int height, int width) {
    // 获取线程的全局ID
    int i = blockIdx.x * blockDim.x + threadIdx.x; // 计算全局行索引
    int j = blockIdx.y * blockDim.y + threadIdx.y; // 计算全局列索引

    // 确保索引在矩阵范围内
    if (i < width && j < height) {
        // 计算当前线程对应的元素索引
        int index = j * width + i;

        // 从矩阵 A 和 B 中读取数据
        float src_data_A = A[index];
        float src_data_B = B[index];

        // 执行加法运算
        float result = src_data_A + src_data_B;

        // 将结果写入矩阵 C
        C[index] = result;
    }
}

void MatAdd(int height, int width) {
    // 在主机内存中分配 A、B 和 C
    float* A = (float*)malloc(height * width * sizeof(float));
    float* B = (float*)malloc(height * width * sizeof(float));
    float* C = (float*)malloc(height * width * sizeof(float));

    // 初始化输入矩阵 A 和 B
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            A[i * width + j] = i + j; // 简单初始化,A的元素为行索引+列索引
            B[i * width + j] = i - j; // 简单初始化,B的元素为行索引-列索引
        }
    }

    // 第一步:在设备内存中为矩阵 A、B 和 C 分配内存
    float* d_A;
    cudaMalloc(&d_A, height * width * sizeof(float)); // 分配矩阵 A 的设备内存
    float* d_B;
    cudaMalloc(&d_B, height * width * sizeof(float)); // 分配矩阵 B 的设备内存
    float* d_C;
    cudaMalloc(&d_C, height * width * sizeof(float)); // 分配矩阵 C 的设备内存

    // 第二步:将矩阵 A 和 B 从主机内存复制到设备内存
    cudaMemcpy(d_A, A, height * width * sizeof(float), cudaMemcpyHostToDevice); // 复制 A
    cudaMemcpy(d_B, B, height * width * sizeof(float), cudaMemcpyHostToDevice); // 复制 B

    // 第三步:定义事件变量,用于测量核函数执行时间
    cudaEvent_t start, stop;
    float time;
    cudaEventCreate(&start); // 创建开始事件
    cudaEventCreate(&stop);  // 创建停止事件

    // 第四步:调用 CUDA 核函数
    dim3 threadsPerBlock(16, 16); // 定义每个块中的线程数
    dim3 numBlocks((width + threadsPerBlock.x - 1) / threadsPerBlock.x, 
                   (height + threadsPerBlock.y - 1) / threadsPerBlock.y); // 计算网格中的块数

    cudaEventRecord(start, 0); // 记录开始时间
    MatAddKernel<<<numBlocks, threadsPerBlock>>>(d_A, d_B, d_C, height, width); // 启动 CUDA 核函数
    cudaEventRecord(stop, 0);  // 记录结束时间

    cudaEventSynchronize(stop); // 等待事件完成
    cudaEventElapsedTime(&time, start, stop); // 计算核函数执行时间
    printf("Kernel execution time: %f ms\n", time); // 输出核函数执行时间

    // 销毁事件
    cudaEventDestroy(start);
    cudaEventDestroy(stop);

    // 第五步:将结果从设备内存复制回主机内存
    cudaMemcpy(C, d_C, height * width * sizeof(float), cudaMemcpyDeviceToHost);

    // 输出结果矩阵 C
    printf("Matrix C (result of A + B):\n");
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            printf("%f ", C[i * width + j]);
        }
        printf("\n");
    }

    // 释放设备内存
    cudaFree(d_A);
    cudaFree(d_B);
    cudaFree(d_C);

    // 释放主机内存
    free(A);
    free(B);
    free(C);
}

int main() {
    int height = 3; // 矩阵的高度
    int width = 3;  // 矩阵的宽度

    // 调用矩阵加法函数
    MatAdd(height, width);

    return 0;
}

输出:

cpp 复制代码
Kernel execution time: 0.086016 ms
Matrix C (result of A + B):
0.000000 0.000000 0.000000 
2.000000 2.000000 2.000000 
4.000000 4.000000 4.000000 

使用 CPU Timers 实际要循环100次求平均值

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <sys/time.h> // 包含 gettimeofday 的头文件
#include <cuda_runtime.h>

// CUDA 核函数,用于矩阵加法
__global__ void MatAddKernel(float* A, float* B, float* C, int height, int width) {
    // 获取线程的全局ID
    int i = blockIdx.x * blockDim.x + threadIdx.x; // 计算全局行索引
    int j = blockIdx.y * blockDim.y + threadIdx.y; // 计算全局列索引

    // 确保索引在矩阵范围内
    if (i < width && j < height) {
        // 计算当前线程对应的元素索引
        int index = j * width + i;

        // 从矩阵 A 和 B 中读取数据
        float src_data_A = A[index];
        float src_data_B = B[index];

        // 执行加法运算
        float result = src_data_A + src_data_B;

        // 将结果写入矩阵 C
        C[index] = result;
    }
}

void MatAdd(int height, int width) {
    // 在主机内存中分配 A、B 和 C
    float* A = (float*)malloc(height * width * sizeof(float));
    float* B = (float*)malloc(height * width * sizeof(float));
    float* C = (float*)malloc(height * width * sizeof(float));

    // 初始化输入矩阵 A 和 B
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            A[i * width + j] = i + j; // 简单初始化,A的元素为行索引+列索引
            B[i * width + j] = i - j; // 简单初始化,B的元素为行索引-列索引
        }
    }

    // 第一步:在设备内存中为矩阵 A、B 和 C 分配内存
    float* d_A;
    cudaMalloc(&d_A, height * width * sizeof(float)); // 分配矩阵 A 的设备内存
    float* d_B;
    cudaMalloc(&d_B, height * width * sizeof(float)); // 分配矩阵 B 的设备内存
    float* d_C;
    cudaMalloc(&d_C, height * width * sizeof(float)); // 分配矩阵 C 的设备内存

    // 第二步:将矩阵 A 和 B 从主机内存复制到设备内存
    cudaMemcpy(d_A, A, height * width * sizeof(float), cudaMemcpyHostToDevice); // 复制 A
    cudaMemcpy(d_B, B, height * width * sizeof(float), cudaMemcpyHostToDevice); // 复制 B

    // 第三步:定义时间测量变量
    struct timeval start, end;
    double elapsedTime;

    // 记录开始时间
    gettimeofday(&start, NULL);

    // 启动 CUDA 核函数
    dim3 threadsPerBlock(16, 16); // 定义每个块中的线程数
    dim3 numBlocks((width + threadsPerBlock.x - 1) / threadsPerBlock.x, 
                   (height + threadsPerBlock.y - 1) / threadsPerBlock.y); // 计算网格中的块数

    MatAddKernel<<<numBlocks, threadsPerBlock>>>(d_A, d_B, d_C, height, width); // 启动 CUDA 核函数
    cudaDeviceSynchronize(); // 等待核函数完成

    // 记录结束时间
    gettimeofday(&end, NULL);

    // 计算时间差
    elapsedTime = (end.tv_sec - start.tv_sec) * 1000.0; // 秒转毫秒
    elapsedTime += (end.tv_usec - start.tv_usec) / 1000.0; // 微秒转毫秒

    // 输出核函数执行时间
    printf("Kernel execution time: %f ms\n", elapsedTime);

    // 第四步:将结果从设备内存复制回主机内存
    cudaMemcpy(C, d_C, height * width * sizeof(float), cudaMemcpyDeviceToHost);

    // 输出结果矩阵 C
    printf("Matrix C (result of A + B):\n");
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            printf("%f ", C[i * width + j]);
        }
        printf("\n");
    }

    // 释放设备内存
    cudaFree(d_A);
    cudaFree(d_B);
    cudaFree(d_C);

    // 释放主机内存
    free(A);
    free(B);
    free(C);
}

int main() {
    int height = 3; // 矩阵的高度
    int width = 3;  // 矩阵的宽度

    // 调用矩阵加法函数
    MatAdd(height, width);

    return 0;
}

输出:

cpp 复制代码
Kernel execution time: 0.101000 ms
Matrix C (result of A + B):
0.000000 0.000000 0.000000 
2.000000 2.000000 2.000000 
4.000000 4.000000 4.000000 

MatAdd程序之变一:每个线程处理4个元素

  • CPU端为输入矩阵A和B、输出矩阵C分配空间,并进行初始化
  • CPU端分配设备端内存,并将A和B传输到GPU上
  • 定义数据和线程的映射关系,并确定线程的开启数量和组织方式
  1. 每个线程负责输出矩阵C的四个元素的计算,全局ID为(i,j)的线程计算索引为(i,4*j~4*j+3)的矩阵元素
  2. 当矩阵规模为width*height时,共开启width/4 * height个线程
  3. 每个block包含256个线程,采用(16,16)的组织形式
  • 编写计算kernel,完成计算任务
  • CPU端将计算结果从Device内存拷贝到Host内存
cpp 复制代码
#include <cuda_runtime.h>
#include <iostream>

// 自定义向上取整除法函数
int div_up(int a, int b) {
    return (a + b - 1) / b;
}
// CUDA核函数,执行矩阵加法
__global__ void MatAdd(float *A, float *B, float *C, int height, int width)
{
    // 获取线程的全局ID
    int i = blockIdx.x * blockDim.x + threadIdx.x;
    int j = blockIdx.y * blockDim.y + threadIdx.y;

    // 检查线程是否在有效的计算范围内
    // 每个线程处理4个连续的元素,因此检查4*i是否在宽度范围内
    if ((4 * i) < width && j < height)
    {
        // 获取线程与数据之间的映射关系
        int index = j * width + 4 * i;

        // 从源矩阵读取数据
        float4 src_data_A = *((float4 *)(A + index));
        float4 src_data_B = *((float4 *)(B + index));

        // 执行加法运算
          // 执行加法运算,逐个对float4的分量进行加法
        float4 result;
        result.x = src_data_A.x + src_data_B.x;
        result.y = src_data_A.y + src_data_B.y;
        result.z = src_data_A.z + src_data_B.z;
        result.w = src_data_A.w + src_data_B.w;

        // 将结果写回结果矩阵
        *((float4 *)(C + index)) = result;
    }
}

// 矩阵加法函数,处理矩阵的加法运算
void MatAdd(int height, int width)
{
    float *A, *B, *C;         // 主机内存指针
    float *d_A, *d_B, *d_C;   // 设备内存指针

    // 分配主机内存
    A = (float*)malloc(height * width * sizeof(float));
    B = (float*)malloc(height * width * sizeof(float));
    C = (float*)malloc(height * width * sizeof(float));

    // 初始化矩阵A和B(这里省略具体的初始化代码)
    // 初始化输入矩阵 A 和 B
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            A[i * width + j] = i + j; // 简单初始化,A的元素为行索引+列索引
            B[i * width + j] = i - j; // 简单初始化,B的元素为行索引-列索引
        }
    }
    // 分配设备内存
    cudaMalloc((void**)&d_A, height * width * sizeof(float));
    cudaMalloc((void**)&d_B, height * width * sizeof(float));
    cudaMalloc((void**)&d_C, height * width * sizeof(float));

    // 将数据从主机内存复制到设备内存
    cudaMemcpy(d_A, A, height * width * sizeof(float), cudaMemcpyHostToDevice);
    cudaMemcpy(d_B, B, height * width * sizeof(float), cudaMemcpyHostToDevice);

    // 设置线程块的大小,每个块中有16x16个线程
    dim3 threadsPerBlock(16, 16);

    // 计算需要多少个线程块来覆盖整个矩阵区域
    // 宽度除以4是因为每个线程处理4个元素
    dim3 numBlocks(div_up(width / 4, threadsPerBlock.x), div_up(height, threadsPerBlock.y));

    // 调用核函数,执行矩阵加法运算
    MatAdd<<<numBlocks, threadsPerBlock>>>(d_A, d_B, d_C, height, width);

    // 将计算结果从设备内存复制到主机内存
    cudaMemcpy(C, d_C, height * width * sizeof(float), cudaMemcpyDeviceToHost);
// 输出结果矩阵 C
    printf("Matrix C (result of A + B):\n");
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            printf("%f ", C[i * width + j]);
        }
        printf("\n");
    }
    // 释放设备内存
    cudaFree(d_A);
    cudaFree(d_B);
    cudaFree(d_C);

    // 释放主机内存
    free(A);
    free(B);
    free(C);
}


int main() {
    int height = 8; // 示例矩阵的高度
    int width = 8;  // 示例矩阵的宽度

    // 调用矩阵加法函数
    MatAdd(height, width);

    return 0;
}

输出:

cpp 复制代码
Matrix C (result of A + B):
0.000000 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000 
2.000000 2.000000 2.000000 2.000000 2.000000 2.000000 2.000000 2.000000 
4.000000 4.000000 4.000000 4.000000 4.000000 4.000000 4.000000 4.000000 
6.000000 6.000000 6.000000 6.000000 6.000000 6.000000 6.000000 6.000000 
8.000000 8.000000 8.000000 8.000000 8.000000 8.000000 8.000000 8.000000 
10.000000 10.000000 10.000000 10.000000 10.000000 10.000000 10.000000 10.000000 
12.000000 12.000000 12.000000 12.000000 12.000000 12.000000 12.000000 12.000000 
14.000000 14.000000 14.000000 14.000000 14.000000 14.000000 14.000000 14.000000 

MatAdd程序之变二

矩阵A、B、C都为NxN的方阵,A和B为已知矩阵,C[i][j] = A[i][j] + B[j][i]。

  • CPU端为输入矩阵A和B、输出矩阵C分配空间,并进行初始化
  • CPU端分配设备端内存,并将A和B传输到GPU上
  • 定义数据和线程的映射关系,并确定线程的开启数量和组织方式
  1. 每个线程负责输出矩阵C的一个元素的计算,全局ID为(i,j)的线程计算索引为(i,j)的矩阵元素
  2. 当矩阵规模为width*height时,共开启width * height个线程
  3. 每个block包含256个线程,采用(16,16)的组织形式
  • 编写计算kernel,完成计算任务
  • CPU端将计算结果从Device内存拷贝到Host内存
cpp 复制代码
#include <cuda_runtime.h>
#include <iostream>

#define N 1024 // 矩阵的大小(N x N)

// CUDA核函数,执行矩阵加法运算
__global__ void MatAdd(float *A, float *B, float *C, int height, int width)
{
    // 获取线程的全局ID
    int i = blockIdx.x * blockDim.x + threadIdx.x;
    int j = blockIdx.y * blockDim.y + threadIdx.y;

    // 确保线程在有效的计算范围内
    if(i < width && j < height)
    {
        // 获取矩阵元素的索引
        int index_A = j * width + i;
        int index_B = i * width + j;
        int index_C = index_A;

        // 从矩阵A和B读取数据并计算结果
        float src_data_A = *(A + index_A);
        float src_data_B = *(B + index_B);
        float result = src_data_A + src_data_B;

        // 将结果写入矩阵C
        *(C + index_C) = result;
    }
}

// 主函数,处理矩阵的初始化、调用CUDA核函数以及结果输出
void MatAdd(int height, int width)
{
    // 在主机内存中为A、B和C分配空间
    float* A = (float*)malloc(height * width * sizeof(float));
    float* B = (float*)malloc(height * width * sizeof(float));
    float* C = (float*)malloc(height * width * sizeof(float));

    // 初始化矩阵A和B
    for (int i = 0; i < height; i++) {
        for (int j = 0; j < width; j++) {
            *(A + i * width + j) = static_cast<float>(i + j); // 示例初始化
            *(B + i * width + j) = static_cast<float>(i - j); // 示例初始化
        }
    }

    // 分配设备内存
    float *d_A, *d_B, *d_C;
    cudaMalloc((void**)&d_A, height * width * sizeof(float));
    cudaMalloc((void**)&d_B, height * width * sizeof(float));
    cudaMalloc((void**)&d_C, height * width * sizeof(float));

    // 将数据从主机内存复制到设备内存
    cudaMemcpy(d_A, A, height * width * sizeof(float), cudaMemcpyHostToDevice);
    cudaMemcpy(d_B, B, height * width * sizeof(float), cudaMemcpyHostToDevice);

    // 配置CUDA网格和线程块大小
    dim3 threadsPerBlock(16, 16);
    dim3 numBlocks((width + threadsPerBlock.x - 1) / threadsPerBlock.x,
                   (height + threadsPerBlock.y - 1) / threadsPerBlock.y);

    // 调用CUDA核函数,执行矩阵加法运算
    MatAdd<<<numBlocks, threadsPerBlock>>>(d_A, d_B, d_C, height, width);

    // 将计算结果从设备内存复制回主机内存
    cudaMemcpy(C, d_C, height * width * sizeof(float), cudaMemcpyDeviceToHost);

    // 输出部分计算结果用于验证
    std::cout << "C[0][0] = " << C[0] << std::endl;
    std::cout << "C[N-1][N-1] = " << C[(N-1) * width + (N-1)] << std::endl;

    // 释放设备内存
    cudaFree(d_A);
    cudaFree(d_B);
    cudaFree(d_C);

    // 释放主机内存
    free(A);
    free(B);
    free(C);
}

int main() {
    // 调用矩阵加法函数
    MatAdd(N, N);

    return 0;
}

输出:

cpp 复制代码
C[0][0] = 0
C[N-1][N-1] = 2046

MatAdd程序之变三

  • 矩阵A、B都为MxN的矩阵,矩阵C为(M/2)*(N/2)的矩阵, A和B为已知矩阵,C[i][j] = A[2*i][2*j] *B[2*i][2*j] + A[2*i][2*j+1] *B[2*i][2*j+1] + A[2*i+1][2*j] *B[2*i+1][2*j] + A[2*i+1][2*j +1] *B[2*i+1][2*j+1] 。 CPU端为输入矩阵A和B、输出矩阵C分配空间,并进行初始化
  • CPU端分配设备端内存,并将A和B传输到GPU上
  • 定义数据和线程的映射关系,并确定线程的开启数量和组织方式
  1. 每个线程负责输出矩阵C的一个元素的计算,全局ID为(i,j)的线程计算索引为(i,j)的矩阵元素
  2. 当矩阵规模为(M/2)*(N/2)时,共开启(M/2)*(N/2)个线程,每个线程对应A和B的四个元素
  3. 每个block包含256个线程,采用(16,16)的组织形式
  • 编写计算kernel,完成计算任务
  • CPU端将计算结果从Device内存拷贝到Host内存
cpp 复制代码
#include <iostream>
#include <cuda_runtime.h>

#define BLOCK_SIZE 16

__global__ void MatAdd(float *A, float *B, float *C, int height, int width)
{
    // Get thread ID
    int i = blockIdx.x * blockDim.x + threadIdx.x;
    int j = blockIdx.y * blockDim.y + threadIdx.y;

    if (i < width / 2 && j < height / 2)
    {
        // Get the mapping between thread and data
        int index_A_1 = 2 * j * width + 2 * i;
        int index_A_2 = 2 * j * width + 2 * i + 1;
        int index_A_3 = (2 * j + 1) * width + 2 * i;
        int index_A_4 = (2 * j + 1) * width + 2 * i + 1;

        int index_B_1 = index_A_1; // Indexes for matrix B are the same as A
        int index_B_2 = index_A_2;
        int index_B_3 = index_A_3;
        int index_B_4 = index_A_4;

        int index_C = j * (width / 2) + i;

        // Read data from source matrices
        float A1 = A[index_A_1];
        float A2 = A[index_A_2];
        float A3 = A[index_A_3];
        float A4 = A[index_A_4];

        float B1 = B[index_B_1];
        float B2 = B[index_B_2];
        float B3 = B[index_B_3];
        float B4 = B[index_B_4];

        // Compute
        float result = A1 * B1 + A2 * B2 + A3 * B3 + A4 * B4;

        // Write result
        C[index_C] = result;
    }
}

int main()
{
    int M = 4; // Example dimensions for the matrices
    int N = 4;

    int size_A = M * N * sizeof(float);
    int size_B = M * N * sizeof(float);
    int size_C = (M / 2) * (N / 2) * sizeof(float);

    // Allocate host memory
    float *h_A = (float*)malloc(size_A);
    float *h_B = (float*)malloc(size_B);
    float *h_C = (float*)malloc(size_C);

    // Initialize matrices A and B
    for (int i = 0; i < M * N; ++i)
    {
        h_A[i] = i + 1; // Example initialization
        h_B[i] = i + 1;
    }

    // Allocate device memory
    float *d_A, *d_B, *d_C;
    cudaMalloc(&d_A, size_A);
    cudaMalloc(&d_B, size_B);
    cudaMalloc(&d_C, size_C);

    // Copy data from host to device
    cudaMemcpy(d_A, h_A, size_A, cudaMemcpyHostToDevice);
    cudaMemcpy(d_B, h_B, size_B, cudaMemcpyHostToDevice);

    // Define block and grid size
    dim3 blockDim(BLOCK_SIZE, BLOCK_SIZE);
    dim3 gridDim((N / 2 + BLOCK_SIZE - 1) / BLOCK_SIZE, (M / 2 + BLOCK_SIZE - 1) / BLOCK_SIZE);

    // Launch kernel
    MatAdd<<<gridDim, blockDim>>>(d_A, d_B, d_C, M, N);

    // Copy result from device to host
    cudaMemcpy(h_C, d_C, size_C, cudaMemcpyDeviceToHost);

    // Print result
    for (int i = 0; i < M / 2; ++i)
    {
        for (int j = 0; j < N / 2; ++j)
        {
            std::cout << h_C[i * (N / 2) + j] << " ";
        }
        std::cout << std::endl;
    }

    // Free device memory
    cudaFree(d_A);
    cudaFree(d_B);
    cudaFree(d_C);

    // Free host memory
    free(h_A);
    free(h_B);
    free(h_C);

    return 0;
}

输出:

cpp 复制代码
66 138 
546 746 
  • 异构计算整成为当前计算领域的重点方向
  • GPGPU是异构计算的主要形式
  • GPGPU是一款大规模细粒度并行处理器,并行思维是进行GPGPU编程的重要前提
  • NVIDIA是当前GPGPU领域当之无愧的霸主
  • GPGPU编程的重点是定义明确的线程和数据间的映射
相关推荐
IE0610 分钟前
深度学习系列76:流式tts的一个简单实现
人工智能·深度学习
GIS数据转换器14 分钟前
城市生命线安全保障:技术应用与策略创新
大数据·人工智能·安全·3d·智慧城市
C语言魔术师16 分钟前
【小游戏篇】三子棋游戏
前端·算法·游戏
自由自在的小Bird17 分钟前
简单排序算法
数据结构·算法·排序算法
刘好念21 分钟前
[OpenGL]实现屏幕空间环境光遮蔽(Screen-Space Ambient Occlusion, SSAO)
c++·计算机图形学·opengl·glsl
利刃大大1 小时前
【Linux入门】2w字详解yum、vim、gcc/g++、gdb、makefile以及进度条小程序
linux·c语言·vim·makefile·gdb·gcc
C嘎嘎嵌入式开发2 小时前
什么是僵尸进程
服务器·数据库·c++
一水鉴天2 小时前
为AI聊天工具添加一个知识系统 之65 详细设计 之6 变形机器人及伺服跟随
人工智能
我想学LINUX2 小时前
【2024年华为OD机试】 (A卷,100分)- 微服务的集成测试(JavaScript&Java & Python&C/C++)
java·c语言·javascript·python·华为od·微服务·集成测试
雁于飞3 小时前
c语言贪吃蛇(极简版,基本能玩)
c语言·开发语言·笔记·学习·其他·课程设计·大作业