画像の回転方法を行っています。2 つの行列と回転度が必要です。元の行列を度数だけ回転し、回転した行列に格納します。次の「通常の」コード(CPU用-このサイトから取得- http://sinepost.wordpress.com/2012/07/24/image-rotation/)があり、正常に機能しています。
static void RotateImage(unsigned char original[RAW_HEIGHT][RAW_WIDTH] , unsigned char rotated[RAW_HEIGHT][RAW_WIDTH] , int degrees)
{
double centerX = RAW_WIDTH/2;
double centerY = RAW_HEIGHT/2;
for(int x = 0; x< RAW_HEIGHT;x++)
{
for (int y = 0; y < RAW_WIDTH; y++)
{
double dir = calculateDirection(x-centerX,y-centerY);
double mag = calculateMagnitude(x-centerX,y-centerY);
dir-=degrees;
int origX = (int)(centerX + calculateX(dir,mag));
int origY = (int)(centerY + calculateY(dir,mag));
if (origX >= 0 && origX < RAW_HEIGHT && origY >= 0 && origY < RAW_WIDTH)
{
rotated[x][y] = original[origX][origY];
}
}
}
}
このコードを CUDA コードに転送したいと思います。これが私のバージョンです:
#define RAW_WIDTH 1600*3
#define RAW_HEIGHT 1200
unsigned char *dev_original_image;
unsigned char *dev_rotated_image;
__global__ void rotatePicture(unsigned char *original, unsigned char *rotated, int degrees)
{
int x = threadIdx.x + blockDim.x * blockIdx.x;
int y = threadIdx.y + blockDim.y * blockIdx.y;
int offset_rotated = x + y * blockDim.x * gridDim.x;
double centerX = 2400.0;
double centerY = 600.0;
double dir = (atan2(y-centerY,x-centerX))*180/3.14159265;
double mag = sqrt((x-centerX)*(x-centerX) + (y-centerY)*(y-centerY));
dir = dir - degrees;
int origX = (int)(centerX + cos((dir*3.14159265/180)) * mag);
int origY = (int)(centerY + sin((dir*3.14159265/180)) * mag);
int offset_original = origX + origY * blockDim.x * gridDim.x;
if(offset_original > 0 && offset_original < RAW_HEIGHT*RAW_WIDTH)
*(rotated + offset_rotated) = *(original + offset_original);
}
しかし、CPU部分と同じ結果にはなりません。問題はCUDA kerenlの引数を渡すことにあると思います。それらを 2D 配列として渡していますが、これは問題ありませんか? 誰かが私にこれを説明できますか?これが私のkerenl構成と呼び出しです:
dim3 BlockPerGrid(450,400,1);
dim3 ThreadsPerGrid(8,4,1);
cudaMalloc((void**)&dev_original_image,sizeof(unsigned char)*RAW_HEIGHT*RAW_WIDTH);
cudaMalloc((void**)&dev_rotated_image,sizeof(unsigned char)*RAW_HEIGHT*RAW_WIDTH);
cudaMemcpy(dev_original_image, raw_image2D, sizeof(unsigned char)*RAW_HEIGHT*RAW_WIDTH,cudaMemcpyHostToDevice);
cudaMemcpy(dev_rotated_image, raw_image2D_rotated, sizeof(unsigned char)*RAW_HEIGHT*RAW_WIDTH, cudaMemcpyHostToDevice);
rotatePicture<<<BlockPerGrid,ThreadsPerGrid>>>(dev_original_image,dev_rotated_image, deg);
アドバイスありがとうございます!
注: コードを修正したところ、動作は改善されましたが、まだ正しくありません。