| // To use the simple FFT implementation |
| // g++ -o demofft -I.. -Wall -O3 FFT.cpp |
| |
| // To use the FFTW implementation |
| // g++ -o demofft -I.. -DUSE_FFTW -Wall -O3 FFT.cpp -lfftw3 -lfftw3f -lfftw3l |
| |
| #ifdef USE_FFTW |
| #include <fftw3.h> |
| #endif |
| |
| #include <vector> |
| #include <complex> |
| #include <algorithm> |
| #include <iterator> |
| #include <iostream> |
| #include <Eigen/Core> |
| #include <unsupported/Eigen/FFT> |
| |
| using namespace std; |
| using namespace Eigen; |
| |
| template <typename T> |
| T mag2(T a) |
| { |
| return a*a; |
| } |
| template <typename T> |
| T mag2(std::complex<T> a) |
| { |
| return norm(a); |
| } |
| |
| template <typename T> |
| T mag2(const std::vector<T> & vec) |
| { |
| T out=0; |
| for (size_t k=0;k<vec.size();++k) |
| out += mag2(vec[k]); |
| return out; |
| } |
| |
| template <typename T> |
| T mag2(const std::vector<std::complex<T> > & vec) |
| { |
| T out=0; |
| for (size_t k=0;k<vec.size();++k) |
| out += mag2(vec[k]); |
| return out; |
| } |
| |
| template <typename T> |
| vector<T> operator-(const vector<T> & a,const vector<T> & b ) |
| { |
| vector<T> c(a); |
| for (size_t k=0;k<b.size();++k) |
| c[k] -= b[k]; |
| return c; |
| } |
| |
| template <typename T> |
| void RandomFill(std::vector<T> & vec) |
| { |
| for (size_t k=0;k<vec.size();++k) |
| vec[k] = T( rand() )/T(RAND_MAX) - T(.5); |
| } |
| |
| template <typename T> |
| void RandomFill(std::vector<std::complex<T> > & vec) |
| { |
| for (size_t k=0;k<vec.size();++k) |
| vec[k] = std::complex<T> ( T( rand() )/T(RAND_MAX) - T(.5), T( rand() )/T(RAND_MAX) - T(.5)); |
| } |
| |
| template <typename T_time,typename T_freq> |
| void fwd_inv(size_t nfft) |
| { |
| typedef typename NumTraits<T_freq>::Real Scalar; |
| vector<T_time> timebuf(nfft); |
| RandomFill(timebuf); |
| |
| vector<T_freq> freqbuf; |
| static FFT<Scalar> fft; |
| fft.fwd(freqbuf,timebuf); |
| |
| vector<T_time> timebuf2; |
| fft.inv(timebuf2,freqbuf); |
| |
| T_time rmse = mag2(timebuf - timebuf2) / mag2(timebuf); |
| cout << "roundtrip rmse: " << rmse << endl; |
| } |
| |
| template <typename T_scalar> |
| void two_demos(int nfft) |
| { |
| cout << " scalar "; |
| fwd_inv<T_scalar,std::complex<T_scalar> >(nfft); |
| cout << " complex "; |
| fwd_inv<std::complex<T_scalar>,std::complex<T_scalar> >(nfft); |
| } |
| |
| void demo_all_types(int nfft) |
| { |
| cout << "nfft=" << nfft << endl; |
| cout << " float" << endl; |
| two_demos<float>(nfft); |
| cout << " double" << endl; |
| two_demos<double>(nfft); |
| cout << " long double" << endl; |
| two_demos<long double>(nfft); |
| } |
| |
| int main() |
| { |
| demo_all_types( 2*3*4*5*7 ); |
| demo_all_types( 2*9*16*25 ); |
| demo_all_types( 1024 ); |
| return 0; |
| } |