10#ifndef EIGEN_FFT_MODULE_H
11#define EIGEN_FFT_MODULE_H
16#include "../../Eigen/Core"
82#include "../../Eigen/src/Core/util/DisableStupidWarnings.h"
86#ifdef EIGEN_FFTW_DEFAULT
89#include "src/FFT/fftw_impl.h"
93struct default_fft_impl :
public internal::fftw_impl<T> {};
95#elif defined EIGEN_MKL_DEFAULT
97#include "src/FFT/imklfft_impl.h"
100struct default_fft_impl :
public internal::imklfft::imklfft_impl<T> {};
102#elif defined EIGEN_POCKETFFT_DEFAULT
104#include <pocketfft_hdronly.h>
105#include "src/FFT/pocketfft_impl.h"
108struct default_fft_impl :
public internal::pocketfft_impl<T> {};
110#elif defined EIGEN_DUCCFFT_DEFAULT
111#include <ducc0/fft/fft.h>
112#include <ducc0/infra/string_utils.h>
113#include <ducc0/fft/fft.h>
114#include <ducc0/fft/fftnd_impl.h>
115#include "src/FFT/duccfft_impl.h"
118struct default_fft_impl :
public internal::duccfft_impl<T> {};
122#include "src/FFT/kissfft_impl.h"
125struct default_fft_impl :
public internal::kissfft_impl<T> {};
134template <
typename T_SrcMat,
typename T_FftIfc>
136template <
typename T_SrcMat,
typename T_FftIfc>
140template <
typename T_SrcMat,
typename T_FftIfc>
141struct traits<fft_fwd_proxy<T_SrcMat, T_FftIfc> > {
142 typedef typename T_SrcMat::PlainObject ReturnType;
144template <
typename T_SrcMat,
typename T_FftIfc>
145struct traits<fft_inv_proxy<T_SrcMat, T_FftIfc> > {
146 typedef typename T_SrcMat::PlainObject ReturnType;
150template <
typename T_SrcMat,
typename T_FftIfc>
151struct fft_fwd_proxy :
public ReturnByValue<fft_fwd_proxy<T_SrcMat, T_FftIfc> > {
152 typedef DenseIndex Index;
154 fft_fwd_proxy(
const T_SrcMat& src, T_FftIfc& fft, Index nfft) : m_src(src), m_ifc(fft), m_nfft(nfft) {}
156 template <
typename T_DestMat>
157 void evalTo(T_DestMat& dst)
const;
159 Index rows()
const {
return m_src.rows(); }
160 Index cols()
const {
return m_src.cols(); }
163 const T_SrcMat& m_src;
168template <
typename T_SrcMat,
typename T_FftIfc>
169struct fft_inv_proxy :
public ReturnByValue<fft_inv_proxy<T_SrcMat, T_FftIfc> > {
170 typedef DenseIndex Index;
172 fft_inv_proxy(
const T_SrcMat& src, T_FftIfc& fft, Index nfft) : m_src(src), m_ifc(fft), m_nfft(nfft) {}
174 template <
typename T_DestMat>
175 void evalTo(T_DestMat& dst)
const;
177 Index rows()
const {
return m_src.rows(); }
178 Index cols()
const {
return m_src.cols(); }
181 const T_SrcMat& m_src;
186template <
typename T_Scalar,
typename T_Impl = default_fft_impl<T_Scalar> >
189 typedef T_Impl impl_type;
190 typedef DenseIndex Index;
191 typedef typename impl_type::Scalar Scalar;
192 typedef typename impl_type::Complex Complex;
195 static constexpr Flag Default = 0;
196 static constexpr Flag Unscaled = 1;
197 static constexpr Flag HalfSpectrum = 2;
198 static constexpr Flag Speedy = 32767;
200 FFT(
const impl_type& impl = impl_type(), Flag flags = Default) : m_impl(impl), m_flag(flags) {
201 eigen_assert((flags == Default || flags == Unscaled || flags == HalfSpectrum || flags == Speedy) &&
202 "invalid flags argument");
205 inline bool HasFlag(Flag f)
const {
return (m_flag & (
int)f) == f; }
207 inline void SetFlag(Flag f) { m_flag |= (int)f; }
209 inline void ClearFlag(Flag f) { m_flag &= (~(int)f); }
211 inline void fwd(Complex* dst,
const Scalar* src, Index nfft) {
212 m_impl.fwd(dst, src,
static_cast<int>(nfft));
213 if (HasFlag(HalfSpectrum) ==
false) ReflectSpectrum(dst, nfft);
216 inline void fwd(Complex* dst,
const Complex* src, Index nfft) { m_impl.fwd(dst, src,
static_cast<int>(nfft)); }
218#if defined EIGEN_FFTW_DEFAULT || defined EIGEN_POCKETFFT_DEFAULT || defined EIGEN_DUCCFFT_DEFAULT || \
219 defined EIGEN_MKL_DEFAULT
220 inline void fwd2(Complex* dst,
const Complex* src,
int n0,
int n1) { m_impl.fwd2(dst, src, n0, n1); }
223 template <
typename Input_>
224 inline void fwd(std::vector<Complex>& dst,
const std::vector<Input_>& src) {
225 if (NumTraits<Input_>::IsComplex == 0 && HasFlag(HalfSpectrum))
226 dst.resize((src.size() >> 1) + 1);
228 dst.resize(src.size());
229 fwd(&dst[0], &src[0], src.size());
232 template <
typename InputDerived,
typename ComplexDerived>
233 inline void fwd(MatrixBase<ComplexDerived>& dst,
const MatrixBase<InputDerived>& src, Index nfft = -1) {
234 typedef typename ComplexDerived::Scalar dst_type;
235 typedef typename InputDerived::Scalar src_type;
236 EIGEN_STATIC_ASSERT_VECTOR_ONLY(InputDerived)
237 EIGEN_STATIC_ASSERT_VECTOR_ONLY(ComplexDerived)
238 EIGEN_STATIC_ASSERT_SAME_VECTOR_SIZE(ComplexDerived, InputDerived)
240 (internal::is_same<dst_type, Complex>::value),
241 YOU_MIXED_DIFFERENT_NUMERIC_TYPES__YOU_NEED_TO_USE_THE_CAST_METHOD_OF_MATRIXBASE_TO_CAST_NUMERIC_TYPES_EXPLICITLY)
242 EIGEN_STATIC_ASSERT(
int(InputDerived::Flags) &
int(ComplexDerived::Flags) &
DirectAccessBit,
243 THIS_METHOD_IS_ONLY_FOR_EXPRESSIONS_WITH_DIRECT_MEMORY_ACCESS_SUCH_AS_MAP_OR_PLAIN_MATRICES)
245 if (nfft < 1) nfft = src.size();
247 Index dst_size = nfft;
248 if (NumTraits<src_type>::IsComplex == 0 && HasFlag(HalfSpectrum)) {
249 dst_size = (nfft >> 1) + 1;
251 dst.derived().resize(dst_size);
253 if (src.innerStride() != 1 || src.size() < nfft) {
254 Matrix<src_type, 1, Dynamic> tmp;
255 if (src.size() < nfft) {
257 tmp.block(0, 0, src.size(), 1) = src;
261 if (dst.innerStride() != 1) {
262 Matrix<dst_type, 1, Dynamic> out(1, dst_size);
263 fwd(&out[0], &tmp[0], nfft);
266 fwd(&dst[0], &tmp[0], nfft);
269 if (dst.innerStride() != 1) {
270 Matrix<dst_type, 1, Dynamic> out(1, dst_size);
271 fwd(&out[0], &src[0], nfft);
274 fwd(&dst[0], &src[0], nfft);
279 template <
typename InputDerived>
280 inline fft_fwd_proxy<MatrixBase<InputDerived>, FFT<T_Scalar, T_Impl> > fwd(
const MatrixBase<InputDerived>& src,
282 return fft_fwd_proxy<MatrixBase<InputDerived>, FFT<T_Scalar, T_Impl> >(src, *
this, nfft);
285 template <
typename InputDerived>
286 inline fft_inv_proxy<MatrixBase<InputDerived>, FFT<T_Scalar, T_Impl> > inv(
const MatrixBase<InputDerived>& src,
288 return fft_inv_proxy<MatrixBase<InputDerived>, FFT<T_Scalar, T_Impl> >(src, *
this, nfft);
291 inline void inv(Complex* dst,
const Complex* src, Index nfft) {
292 m_impl.inv(dst, src,
static_cast<int>(nfft));
293 if (HasFlag(Unscaled) ==
false) scale(dst, Scalar(1. / nfft), nfft);
296 inline void inv(Scalar* dst,
const Complex* src, Index nfft) {
297 m_impl.inv(dst, src,
static_cast<int>(nfft));
298 if (HasFlag(Unscaled) ==
false) scale(dst, Scalar(1. / nfft), nfft);
301 template <
typename OutputDerived,
typename ComplexDerived>
302 inline void inv(MatrixBase<OutputDerived>& dst,
const MatrixBase<ComplexDerived>& src, Index nfft = -1) {
303 typedef typename ComplexDerived::Scalar src_type;
304 typedef typename ComplexDerived::RealScalar real_type;
305 typedef typename OutputDerived::Scalar dst_type;
306 const bool realfft = (NumTraits<dst_type>::IsComplex == 0);
307 EIGEN_STATIC_ASSERT_VECTOR_ONLY(OutputDerived)
308 EIGEN_STATIC_ASSERT_VECTOR_ONLY(ComplexDerived)
309 EIGEN_STATIC_ASSERT_SAME_VECTOR_SIZE(ComplexDerived, OutputDerived)
311 (internal::is_same<src_type, Complex>::value),
312 YOU_MIXED_DIFFERENT_NUMERIC_TYPES__YOU_NEED_TO_USE_THE_CAST_METHOD_OF_MATRIXBASE_TO_CAST_NUMERIC_TYPES_EXPLICITLY)
313 EIGEN_STATIC_ASSERT(
int(OutputDerived::Flags) &
int(ComplexDerived::Flags) &
DirectAccessBit,
314 THIS_METHOD_IS_ONLY_FOR_EXPRESSIONS_WITH_DIRECT_MEMORY_ACCESS_SUCH_AS_MAP_OR_PLAIN_MATRICES)
317 if (realfft && HasFlag(HalfSpectrum))
318 nfft = 2 * (src.size() - 1);
322 dst.derived().resize(nfft);
325 Index resize_input = (realfft && HasFlag(HalfSpectrum)) ? ((nfft / 2 + 1) - src.size()) : (nfft - src.size());
327 if (src.innerStride() != 1 || resize_input) {
329 Matrix<src_type, 1, Dynamic> tmp;
331 size_t ncopy = (std::min)(src.size(), src.size() + resize_input);
332 tmp.setZero(src.size() + resize_input);
333 if (realfft && HasFlag(HalfSpectrum)) {
335 tmp.head(ncopy) = src.head(ncopy);
336 tmp(ncopy - 1) =
real(tmp(ncopy - 1));
339 nhead = 1 + ncopy / 2 - 1;
340 ntail = ncopy / 2 - 1;
341 tmp.head(nhead) = src.head(nhead);
342 tmp.tail(ntail) = src.tail(ntail);
345 tmp(nhead) = (src(nfft / 2) + src(src.size() - nfft / 2)) * real_type(.5);
347 tmp(nhead) = src(nhead) * real_type(.5);
348 tmp(tmp.size() - nhead) = tmp(nhead);
355 if (dst.innerStride() != 1) {
356 Matrix<dst_type, 1, Dynamic> out(1, nfft);
357 inv(&out[0], &tmp[0], nfft);
360 inv(&dst[0], &tmp[0], nfft);
363 if (dst.innerStride() != 1) {
364 Matrix<dst_type, 1, Dynamic> out(1, nfft);
365 inv(&out[0], &src[0], nfft);
368 inv(&dst[0], &src[0], nfft);
373 template <
typename Output_>
374 inline void inv(std::vector<Output_>& dst,
const std::vector<Complex>& src, Index nfft = -1) {
376 nfft = (NumTraits<Output_>::IsComplex == 0 && HasFlag(HalfSpectrum)) ? 2 * (src.size() - 1) : src.size();
378 inv(&dst[0], &src[0], nfft);
381#if defined EIGEN_FFTW_DEFAULT || defined EIGEN_POCKETFFT_DEFAULT || defined EIGEN_DUCCFFT_DEFAULT || \
382 defined EIGEN_MKL_DEFAULT
383 inline void inv2(Complex* dst,
const Complex* src,
int n0,
int n1) {
384 m_impl.inv2(dst, src, n0, n1);
385 if (HasFlag(Unscaled) ==
false) scale(dst, 1. / (n0 * n1), n0 * n1);
389 inline impl_type& impl() {
return m_impl; }
392 template <
typename T_Data>
393 inline void scale(T_Data* x, Scalar s, Index nx) {
395 for (
int k = 0; k < nx; ++k) *x++ *= s;
397 if (((ptrdiff_t)x) & 15)
398 Matrix<T_Data, Dynamic, 1>::Map(x, nx) *= s;
400 Matrix<T_Data, Dynamic, 1>::MapAligned(x, nx) *= s;
404 inline void ReflectSpectrum(Complex* freq, Index nfft) {
406 Index nhbins = (nfft >> 1) + 1;
407 for (Index k = nhbins; k < nfft; ++k) freq[k] =
conj(freq[nfft - k]);
414template <
typename T_SrcMat,
typename T_FftIfc>
415template <
typename T_DestMat>
416inline void fft_fwd_proxy<T_SrcMat, T_FftIfc>::evalTo(T_DestMat& dst)
const {
417 m_ifc.fwd(dst, m_src, m_nfft);
420template <
typename T_SrcMat,
typename T_FftIfc>
421template <
typename T_DestMat>
422inline void fft_inv_proxy<T_SrcMat, T_FftIfc>::evalTo(T_DestMat& dst)
const {
423 m_ifc.inv(dst, m_src, m_nfft);
428#include "../../Eigen/src/Core/util/ReenableStupidWarnings.h"
const unsigned int DirectAccessBit
Namespace containing all symbols from the Eigen library.
const Eigen::CwiseUnaryOp< Eigen::internal::scalar_real_op< typename Derived::Scalar >, const Derived > real(const Eigen::ArrayBase< Derived > &x)
const Eigen::CwiseUnaryOp< Eigen::internal::scalar_conjugate_op< typename Derived::Scalar >, const Derived > conj(const Eigen::ArrayBase< Derived > &x)