Преобразование массива NumPy в cufftComplex
Я пишу сценарий для выполнения FFT с использованием библиотеки cuFFT на основе GPU/CUDA. CuFFT требует, чтобы входные данные были в формате, указанном как "cufftComplex". Однако мои входные данные в формате numpy.complex64. Я использую C-API Python для отправки данных из Python в C. Как я могу конвертировать между двумя форматами? В настоящее время мой код выглядит так:
#include<python2.7/Python.h>
#include<numpy/arrayobject.h>
#include<cufft.h>
void compute_BP(PyObject* inputData, pyObject* OutputData, int Nfft)
{
cuffthandle plan;
cuFFTPlan1d(&plan, Nfft, CUFFT_C2C, CUFFT_INVERSE);
cuFFTExecC2C(plan, inputData, OutputData, CUFFT_INVERSE);
...
}
При компиляции я получаю следующую ошибку:
Ошибка: аргумент типа "PyObject *" несовместим с параметром типа "cufftComplex".
1 ответ
Решение
Заимствуя из моего ответа здесь, вот рабочий пример того, как вы могли бы использовать ctypes
в Python, чтобы запустить функцию из библиотеки cufft в скрипте Python, используя numpy
данные:
$ cat mylib.cpp
#include <cufft.h>
#include <stdio.h>
#include <assert.h>
#include <cuda_runtime_api.h>
extern "C"
void fft(void *input, void *output, size_t N){
cufftHandle plan;
cufftComplex *d_in, *d_out;
size_t ds = N*sizeof(cufftComplex);
cudaMalloc((void **)&d_in, ds);
cudaMalloc((void **)&d_out, ds);
cufftResult res = cufftPlan1d(&plan, N, CUFFT_C2C, 1);
assert(res == CUFFT_SUCCESS);
cudaMemcpy(d_in, input, ds, cudaMemcpyHostToDevice);
res = cufftExecC2C(plan, d_in, d_out, CUFFT_FORWARD);
assert(res == CUFFT_SUCCESS);
cudaMemcpy(output, d_out, ds, cudaMemcpyDeviceToHost);
printf("%s\n", cudaGetErrorString(cudaGetLastError()));
printf("from shared object:\n");
for (int i = 0; i < N; i++)
printf("%.1f + j%.1f, ", ((cufftComplex *)output)[i].x, ((cufftComplex *)output)[i].y);
printf("\n");
}
$ cat t8.py
import ctypes
import os
import sys
import numpy as np
mylib = ctypes.cdll.LoadLibrary('libmylib.so')
N = 4
mydata = np.ones((N), dtype = np.complex64)
myresult = np.zeros((N), dtype = np.complex64)
mylib.fft(ctypes.c_void_p(mydata.ctypes.data), ctypes.c_void_p(myresult.ctypes.data), ctypes.c_size_t(N))
print(myresult)
$ g++ -fPIC -I/usr/local/cuda/include --shared mylib.cpp -L/usr/local/cuda/lib64 -lcufft -lcudart -o libmylib.so
$ LD_LIBRARY_PATH=$LD_LIBRARY_PATH:`pwd` python t8.py
no error
from shared object:
4.0 + j0.0, 0.0 + j0.0, 0.0 + j0.0, 0.0 + j0.0,
[4.+0.j 0.+0.j 0.+0.j 0.+0.j]
$