I am porting my c++ code to CUDA & CUBLAS. I use stl::complex for complex computation (i.e. pow, log, exp, etc.) but I didn't see the same functions defined in CuComplex library. I don't know how to create those functions but I found some codes online
#include <iostream>
#include <cublas_v2.h>
#include <cuComplex.h>
using namespace std;
typedef cuDoubleComplex Complex;
#define complex(x, y) make_cuDoubleComplex(x, y)
__host__ __device__ double cabs(const Complex& z) {return cuCabs(z);}
__host__ __device__ double carg(const Complex& z) {return atan2(cuCreal(z), cuCimag(z));}
__host__ __device__ Complex polar(const double &magnitude, const double &angle) {return complex(magnitude*cos(angle), magnitude*sin(angle));}
__host__ __device__ Complex cexp(const Complex& z) {return polar( exp(cuCreal(z)), cuCimag(z));}
__host__ __device__ Complex czlog(const Complex& z) {return complex( ::log(cabs(z)), carg(z) );}
__host__ __device__ Complex cpow(const Complex& z, const int &exponent) {return cexp( cuCmul(czlog(z), complex((double )exponent, 0)) );}
void main(void)
{
Complex z=complex(0.34, 0.56);
cout << cuCreal(cpow(z, 2)) << " " << cuCimag(cpow(z, 2)) << endl;
}
the above results didn't give right answer. Is that anything wrong with the cpow? Is that any better to do the power and other function on complex number?
This is not correct:
The polar angle of a complex number is given by the arctangent of the imaginary part divided by the real part of the complex number. This corresponds to the ratio of the first parameter divided by the second parameter of atan2
Therefore you should use:
I'm not sure about your power function (
cpow
) either. Have you tried DeMoivre's theorem? I don't know computationally the best method, but it seems like the first order of business is to get the right answer.Additional notes:
Here's a worked example based on DeMoivre's theorem:
I'm not suggesting this is thoroughly tested code, but it seems to be on the right track and gives the right answer for your test case.