Как сделать умножение матриц, не ограничивая размеры входных матриц, кратные размеру блока? - PullRequest
0 голосов
/ 20 февраля 2019

Мне нужно изменить код, чтобы снять ограничение, что размеры входных матриц должны быть кратны размеру блока , но не знаю, как это сделать.

Руководство, которому я следую, говорит, что для этого нужно внести изменения в метод main , но, насколько мне известно, код в main просто заполняет массив и вызывает ядро..

Ниже приведен мой код, это базовый калькулятор умножения матриц.

#include <stdio.h>
#include <stdlib.h>

#include <cuda.h>
#define BLOCK_SIZE 16

// Matrices are stored in row-major order
typedef struct {
int width;
int height;
float* elements;
} Matrix;

__global__ void MatrixMultKern(const Matrix A, const Matrix B, const Matrix C) {
    // Calculate the column index of C and B
    int col = blockIdx.x * blockDim.x + threadIdx.x;
    // Calculate the row index of C and of A
    int row = blockIdx.y * blockDim.y + threadIdx.y;

    if ((row < A.height) && (col < B.width)) {
        float Cvalue = 0;
        // each thread computes one element of the block sub-matrix
        for (int k = 0; k < A.width; ++k) {
            Cvalue += A.elements[row * A.width + k] * B.elements[k*B.width + col];
        }
        C.elements[row * C.width + col] = Cvalue;
    }
}

// Matrix multiplication - Host Code
// Matrix dimensions are assumed to be multiples of BLOCK_SIZE
void MatrixMult(const Matrix h_A, const Matrix h_B, Matrix h_C)
{
    // Load A and B into device memory
    Matrix d_A;
    d_A.width = h_A.width; d_A.height = h_A.height;
    size_t size = h_A.width * h_A.height * sizeof(float);
    cudaMalloc(&d_A.elements, size);
    cudaMemcpy(d_A.elements, h_A.elements, size, cudaMemcpyHostToDevice);

    Matrix d_B;
    d_B.width = h_B.width; d_B.height = h_B.height;
    size = h_B.width * h_B.height * sizeof(float);
    cudaMalloc(&d_B.elements, size);
    cudaMemcpy(d_B.elements, h_B.elements, size, cudaMemcpyHostToDevice);

    // Allocate C in Device memory
    Matrix d_C;
    d_C.width = h_C.width; d_C.height = h_C.height;
    size = h_C.width * h_C.height * sizeof(float);
    cudaMalloc(&d_C.elements, size);

    // Invoke Kernel
    dim3 dimBlock(BLOCK_SIZE, BLOCK_SIZE);
    dim3 dimGrid(d_B.width / dimBlock.x, d_A.height / dimBlock.y);
    MatrixMultKern<<< dimGrid, dimBlock >>>(d_A, d_B, d_C);

    // Read C from Device to Host
    cudaMemcpy(h_C.elements, d_C.elements, size, cudaMemcpyDeviceToHost);

    // Free Device Memory
    cudaFree(d_A.elements);
    cudaFree(d_B.elements);
    cudaFree(d_C.elements);
}

int main(int argc, char* argv[]) {
    Matrix A, B, C;
    // Read Dimensions of A and B
    A.height = atoi(argv[1]);
    A.width = atoi(argv[2]);
    B.height = A.width;
    B.width = atoi(argv[3]);

    //Allocate memory
    A.elements = (float*)malloc(A.width * A.height * sizeof(float));
    B.elements = (float*)malloc(B.width * B.height * sizeof(float));
    C.height = A.height;
    C.width = B.width;
    C.elements = (float*)malloc(C.width * C.height * sizeof(float));

    //populates arrays with random numbers
    for(int i = 0; i < A.height; i++)
        for(int j = 0; j < A.width; j++)
            A.elements[i*A.width + j] = (float)(rand() % 3);
    for(int i = 0; i < B.height; i++)
        for(int j = 0; j < B.width; j++)
            B.elements[i*B.width + j] = (float)(rand() % 2);
    MatrixMult(A, B, C);

    for(int i = 0; i < A.height; i++){
        for(int j = 0; j < A.width; j++)
            printf("%f ", A.elements[i*A.width + j]);
        printf("\n");
    }
    printf("\n");
    for(int i = 0; i < B.height; i++){
        for(int j = 0; j < B.width; j++)
            printf("%f ", B.elements[i*B.width + j]);
        printf("\n");
    }
    printf("\n");
    for(int i = 0; i < C.height; i++){
        for(int j = 0; j < C.width; j++)
            printf("%f ", C.elements[i*C.width + j]);
        printf("\n");
    }
    printf("\n");

    return 0;
}
...