mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
fftwTemplates.hpp
Go to the documentation of this file.
1/** \file fftwTemplates.hpp
2 * \brief Declares and defines templatized wrappers for the fftw library
3 * \ingroup ft_files
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2015, 2016, 2017 Jared R. Males (jaredmales@gmail.com)
10//
11// This file is part of mxlib.
12//
13// mxlib is free software: you can redistribute it and/or modify
14// it under the terms of the GNU General Public License as published by
15// the Free Software Foundation, either version 3 of the License, or
16// (at your option) any later version.
17//
18// mxlib is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU General Public License for more details.
22//
23// You should have received a copy of the GNU General Public License
24// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
25//***********************************************************************//
26
27#ifndef fftwTemplates_hpp
28#define fftwTemplates_hpp
29
30#include <complex>
31#include <vector>
32
33#include <fftw3.h>
34
35namespace mx
36{
37namespace math
38{
39namespace ft
40{
41
42/** \addtogroup fftw_templates
43 *
44 * A templatized interface to the fftw library. Accesses the correct fftw types and
45 * functions based on the provided input and output types.
46 */
47
48/** \defgroup fftw_template_types Types
49 * \brief Types and type specifications for FFTW Templates.
50 *
51 * \ingroup fftw_templates
52 *
53 * @{
54 */
55
56/// The complex float data type.
57typedef std::complex<float> complexFT;
58
59/// The complex double data type.
60typedef std::complex<double> complexDT;
61
62/// The complex long double data type.
63typedef std::complex<long double> complexLT;
64
65#ifdef HASQUAD
66/// The complex __float128 data type.
67typedef std::complex<__float128> complexQT;
68#endif
69
70//****** Plans ******//
71
72/** \defgroup fftw_template_plan_specs Plan Specification
73 * \brief Plan specifications for FFTW Templates.
74 *
75 * \ingroup fftw_template_types
76 *
77 * @{
78 */
79
80/// Specify the type of the plan based on the real type of the data
81/**
82 *
83 * \tparam realT is the real floating point type.
84 */
85template <typename realT>
86struct fftwPlanSpec;
87
88// For doxygen only:
89#ifdef __DOXY_ONLY__
90template <typename realT>
92{
93 /// Specializations typedef planT as fftwf_plan, fftw_plan, fftwl_plan, or fftwq_plan.
95};
96#endif
97
98/// Specialization of fftwPlanSpec for float
99template <>
101{
102 /// Specifies fftwf_plan as planT
104};
105
106/// Specialization of fftwPlanSpec for double
107template <>
109{
110 /// Specifies fftw_plan as planT
111 typedef fftw_plan planT; /// Specifies fftwf_plan as planT
112};
113
114/// Specialization of fftwPlanSpec for long double
115template <>
117{
118 /// Specifies fftwl_plan as planT
120};
121
122#ifdef HASQUAD
123/// Specialization of fftwPlanSpec for __float128
124template <>
126{
127 /// Specifies fftwq_plan as planT
129};
130#endif
131
132///@}
133
134//******* Type Specification *********//
135
136/** \defgroup fftw_template_type_specs Type Specification
137 * \brief Type specifications for FFTW Templates.
138 *
139 * \ingroup fftw_template_types
140 *
141 * @{
142 */
143
144/// A structure specifying various types based on the FFT input and output data types.
145/** Provides various typedefs to specify which fftw functions to compile.
146 *
147 * See the specializations:
148 * \li fftwTypeSpec<complexFT,complexFT>
149 * \li fftwTypeSpec<float, complexFT>
150 * \li fftwTypeSpec<complexFT, float>
151 * \li fftwTypeSpec<complexDT,complexDT>
152 * \li fftwTypeSpec<double, complexDT>
153 * \li fftwTypeSpec<complexDT, double>
154 * \li fftwTypeSpec<complexLT,complexLT>
155 * \li fftwTypeSpec<long double, complexLT>
156 * \li fftwTypeSpec<complexLT, long double>
157 * \li fftwTypeSpec<complexQT,complexQT>
158 * \li fftwTypeSpec<__float128, complexQT>
159 * \li fftwTypeSpec<complexQT, __float128>
160 *
161 * \tparam _inputDataT is the data type of the input array
162 * \tparam _outputDataT is the data type of the output array
163 *
164 */
165template <typename _inputDataT, typename _outputDataT>
166struct fftwTypeSpec;
167
168// For doxygen only:
169#ifdef __DOXY_ONLY__
170template <typename _inputDataT, typename _outputDataT>
172{
173 typedef _realT realT; ///< The real data type (_realT is actually defined in specializations).
174 typedef std::complex<realT> complexT; ///< The complex data type
175 typedef _inputDataT inputDataT; ///< The input array data type
176 typedef _outputDataT outputDataT; ///< The output array data type
177 typedef fftwPlanSpec<realT>::planT planT; ///< The plan type
178};
179#endif
180
181/// Specialization of fftwTypeSpec for complex-float input and complex-float output.
182/**
183 */
184template <>
186{
187 typedef float realT; ///< The real data type
188 typedef complexFT complexT; ///< The complex data type
189 typedef complexFT inputDataT; ///< The input array data type
190 typedef complexFT outputDataT; ///< The output array data type
191 typedef fftwPlanSpec<float>::planT planT; ///< The plan type
192};
193
194/// Specialization of fftwTypeSpec for real-float input and complex-float output
195template <>
197{
198 typedef float realT; ///< The real data type
199 typedef complexFT complexT; ///< The complex data type
200 typedef float inputDataT; ///< The input array data type
201 typedef complexFT outputDataT; ///< The output array data type
202 typedef fftwPlanSpec<float>::planT planT; ///< The plan type
203};
204
205/// Specialization of fftwTypeSpec for complex-float input and real-float output
206template <>
208{
209 typedef float realT; ///< The real data type
210 typedef complexFT complexT;
211 typedef complexFT inputDataT;
212 typedef float outputDataT;
214};
215
216/// Specialization of fftwTypeSpec for complex-double input and complex-double output
217template <>
219{
220 typedef double realT; ///< The real data type
221 typedef complexDT complexT;
222 typedef complexDT inputDataT;
223 typedef complexDT outputDataT;
225};
226
227/// Specialization of fftwTypeSpec for real-double input and complex-double output
228template <>
230{
231 typedef double realT; ///< The real data type
232 typedef complexDT complexT;
233 typedef double inputDataT;
234 typedef complexDT outputDataT;
236};
237
238/// Specialization of fftwTypeSpec for complex-double input and double output
239template <>
241{
242 typedef double realT; ///< The real data type
243 typedef complexDT complexT;
244 typedef complexDT inputDataT;
245 typedef double outputDataT;
247};
248
249/// Specialization of fftwTypeSpec for complex-long-double input and complex-long-double output
250template <>
252{
253 typedef long double realT; ///< The real data type
254 typedef complexLT complexT;
255 typedef complexLT inputDataT;
256 typedef complexLT outputDataT;
258};
259
260/// Specialization of fftwTypeSpec for real-long-double input and complex-long-double output
261template <>
263{
264 typedef long double realT; ///< The real data type
265 typedef complexLT complexT;
266 typedef long double inputDataT;
267 typedef complexLT outputDataT;
269};
270
271/// Specialization of fftwTypeSpec for complex-long-double input and real-long-double output
272template <>
274{
275 typedef long double realT; ///< The real data type
276 typedef complexLT complexT;
277 typedef complexLT inputDataT;
278 typedef long double outputDataT;
280};
281
282#ifdef HASQUAD
283/// Specialization of fftwTypeSpec for complex-quad input and complex-quad output
284template <>
286{
287 typedef __float128 realT; ///< The real data type
288 typedef complexQT complexT;
289 typedef complexQT inputDataT;
290 typedef complexQT outputDataT;
292};
293
294/// Specialization of fftwTypeSpec for real-quad input and complex-quad output
295template <>
297{
298 typedef __float128 realT; ///< The real data type
299 typedef complexQT complexT;
300 typedef __float128 inputDataT;
301 typedef complexQT outputDataT;
303};
304
305/// Specialization of fftwTypeSpec for complex-quad input and real-quad output
306template <>
308{
309 typedef __float128 realT; ///< The real data type
310 typedef complexQT complexT;
311 typedef complexQT inputDataT;
312 typedef __float128 outputDataT;
314};
315#endif
316
317///@}
318///@}
319
320/** \defgroup fftw_template_wisdom Wisdom
321 * \brief Wrappers for working with fftw wisdom.
322 *
323 * \ingroup fftw_templates
324 *
325 * @{
326 */
327
328/// Template wrapper for fftwX_import_system_wisdom();
329/**
330 * \retval 0 on failure
331 * \retval 1 on success
332 *
333 * \tparam realT the real floating point type.
334 */
335template <typename realT>
337
338// specializations don't need to be doxygen-ed:
339
340// Template wrapper for fftwf_import_system_wisdom();
341template <>
342int fftw_import_system_wisdom<float>();
343
344// Template wrapper for fftw_import_system_wisdom();
345template <>
346int fftw_import_system_wisdom<double>();
347
348// Template wrapper for fftwl_import_system_wisdom();
349template <>
350int fftw_import_system_wisdom<long double>();
351
352#ifdef HASQUAD
353// Template wrapper for fftwq_import_system_wisdom();
354template <>
355int fftw_import_system_wisdom<__float128>();
356#endif
357
358/// Template wrapper for fftwX_import_wisdom_from_filename(const char *);
359/**
360 * \param filename is the name of the wisdom file
361 *
362 * \retval 0 on failure
363 * \retval 1 on success
364 *
365 * \tparam realT the real floating point type.
366 */
367template <typename realT>
369
370template <>
371int fftw_import_wisdom_from_filename<float>( const char *filename );
372
373template <>
374int fftw_import_wisdom_from_filename<double>( const char *filename );
375
376template <>
377int fftw_import_wisdom_from_filename<long double>( const char *filename );
378
379#ifdef HASQUAD
380template <>
381int fftw_import_wisdom_from_filename<__float128>( const char *filename );
382#endif
383
384/// Template wrapper for fftwX_export_wisdom_to_filename(const char *);
385/**
386 * \param filename is the name of the wisdom file
387 *
388 * \retval 0 on failure
389 * \retval 1 on success
390 *
391 * \tparam realT the real floating point type.
392 */
393template <typename realT>
395
396template <>
397int fftw_export_wisdom_to_filename<float>( const char *filename );
398
399template <>
400int fftw_export_wisdom_to_filename<double>( const char *filename );
401
402template <>
403int fftw_export_wisdom_to_filename<long double>( const char *filename );
404
405#ifdef HASQUAD
406template <>
407int fftw_export_wisdom_to_filename<__float128>( const char *filename );
408#endif
409
410///@} fftw_template_wisdom
411
412/** \defgroup fftw_template_alloc Allocation
413 * \brief Wrappers for fftw memory allocation and de-allocation functions.
414 *
415 * \note Here the call to sizeof() is implicit, so you only specify the number of elements, not the number of bytes.
416 *
417 * \ingroup fftw_templates
418 *
419 * @{
420 */
421
422//************ Allocation ************************//
423
424/// Call to fftw_malloc, with type cast.
425/**
426 * \note Here the call to sizeof() is implicit, so you only specify the number of elements, not the number of bytes.
427 *
428 * \param n is the number of type realT elements, not the number of bytes.
429 */
430template <typename realT>
431realT *fftw_malloc( size_t n );
432
433template <>
434float *fftw_malloc<float>( size_t n );
435
436template <>
437complexFT *fftw_malloc<complexFT>( size_t n );
438
439template <>
440double *fftw_malloc<double>( size_t n );
441
442template <>
443complexDT *fftw_malloc<complexDT>( size_t n );
444
445template <>
446long double *fftw_malloc<long double>( size_t n );
447
448template <>
449complexLT *fftw_malloc<complexLT>( size_t n );
450
451#ifdef HASQUAD
452template <>
453__float128 *fftw_malloc<__float128>( size_t n );
454
455template <>
456complexQT *fftw_malloc<complexQT>( size_t n );
457#endif
458
459//************ De-Allocation ************************//
460
461/// Call to fftw_free.
462/**
463 * \param p is a pointer to the array to de-allocate.
464 */
465template <typename realT>
466void fftw_free( realT *p );
467
468template <>
469void fftw_free<float>( float *p );
470
471template <>
472void fftw_free<complexFT>( complexFT *p );
473
474template <>
475void fftw_free<double>( double *p );
476
477template <>
478void fftw_free<complexDT>( complexDT *p );
479
480template <>
481void fftw_free<long double>( long double *p );
482
483template <>
484void fftw_free<complexLT>( complexLT *p );
485
486#ifdef HASQUAD
487template <>
488void fftw_free<__float128>( __float128 *p );
489
490template <>
491void fftw_free<complexQT>( complexQT *p );
492#endif
493
494///@} fftw_template_alloc
495
496/** \defgroup fftw_template_plans Planning
497 * \brief Wrappers for working with fftw plans.
498 *
499 * \ingroup fftw_templates
500 *
501 * @{
502 */
503
504/// Wrapper for fftwX_make_planner_thread_safe()
505/**
506 * \tparam realT the real floating point type
507 */
508template <typename realT>
510
511template <>
512void fftw_make_planner_thread_safe<float>();
513
514template <>
515void fftw_make_planner_thread_safe<double>();
516
517template <>
518void fftw_make_planner_thread_safe<long double>();
519
520#ifdef HASQUAD
521template <>
522void fftw_make_planner_thread_safe<__float128>();
523#endif
524
525//********** Threads ****************//
526/// Tell the FFTW planner how many threads to use.
527/**
528 * \tparam realT the real floating point type.
529 */
530template <typename realT>
531void fftw_plan_with_nthreads( int nthreads /**< [in] the number of threads. Set to 1 to disable threads. */ );
532
533template <>
534void fftw_plan_with_nthreads<float>( int nthreads );
535
536template <>
537void fftw_plan_with_nthreads<double>( int nthreads );
538
539template <>
540void fftw_plan_with_nthreads<long double>( int nthreads );
541
542#ifdef HASQUAD
543template <>
544void fftw_plan_with_nthreads<__float128>( int nthreads );
545#endif
546
547//********* Plan Creation **************//
548
549/// Wrapper for the fftwX_plan_dft functions
550/**
551 * \param n is a vector of ints containing the size of each dimension.
552 * \param in is the input data array
553 * \param out is the output data array
554 * \param sign specifies forward or backwards, i.e. FFTW_FORWARD or FFTW_BACKWARD.
555 * \param flags other fftw flags
556 *
557 * \returns an fftw plan for the types specified.
558 *
559 * \tparam inputDataT the data type of the input array
560 * \tparam outputDataT the data type of the output array
561 */
562template <typename inputDataT, typename outputDataT>
564fftw_plan_dft( std::vector<int> n, inputDataT *in, outputDataT *out, int sign, unsigned flags );
565
566template <>
568fftw_plan_dft<complexFT, complexFT>( std::vector<int> n, complexFT *in, complexFT *out, int sign, unsigned flags );
569
570template <>
572fftw_plan_dft<float, complexFT>( std::vector<int> n, float *in, complexFT *out, int sign, unsigned flags );
573
574template <>
576fftw_plan_dft<complexFT, float>( std::vector<int> n, complexFT *in, float *out, int sign, unsigned flags );
577
578template <>
580fftw_plan_dft<complexDT, complexDT>( std::vector<int> n, complexDT *in, complexDT *out, int sign, unsigned flags );
581
582template <>
584fftw_plan_dft<double, complexDT>( std::vector<int> n, double *in, complexDT *out, int sign, unsigned flags );
585
586template <>
588fftw_plan_dft<complexDT, double>( std::vector<int> n, complexDT *in, double *out, int sign, unsigned flags );
589
590template <>
592fftw_plan_dft<complexLT, complexLT>( std::vector<int> n, complexLT *in, complexLT *out, int sign, unsigned flags );
593
594template <>
596fftw_plan_dft<long double, complexLT>( std::vector<int> n, long double *in, complexLT *out, int sign, unsigned flags );
597
598template <>
600fftw_plan_dft<complexLT, long double>( std::vector<int> n, complexLT *in, long double *out, int sign, unsigned flags );
601
602#ifdef HASQUAD
603template <>
605fftw_plan_dft<complexQT, complexQT>( std::vector<int> n, complexQT *in, complexQT *out, int sign, unsigned flags );
606
607template <>
609fftw_plan_dft<__float128, complexQT>( std::vector<int> n, __float128 *in, complexQT *out, int sign, unsigned flags );
610
611template <>
613fftw_plan_dft<complexQT, __float128>( std::vector<int> n, complexQT *in, __float128 *out, int sign, unsigned flags );
614#endif
615
616/********* Cleanup *************/
617
618/// Cleanup persistent planner data.
619template <typename realT>
621
622template <>
623void fftw_cleanup<float>();
624
625template <>
626void fftw_cleanup<double>();
627
628template <>
629void fftw_cleanup<long double>();
630
631#ifdef HASQUAD
632template <>
633void fftw_cleanup<__float128>();
634#endif
635
636/********* Thread Cleanup *************/
637
638/// Cleanup persistent planner data and threads data.
639template <typename realT>
641
642template <>
643void fftw_cleanup_threads<float>();
644
645template <>
646void fftw_cleanup_threads<double>();
647
648template <>
649void fftw_cleanup_threads<long double>();
650
651#ifdef HASQUAD
652template <>
653void fftw_cleanup_threads<__float128>();
654#endif
655
656///@} fftw_template_plans
657
658/** \defgroup fftw_template_exec Execution
659 * \brief Wrappers for executing with fftw plans.
660 *
661 * \ingroup fftw_templates
662 *
663 * @{
664 */
665
666//********* Plan Execution *************//
667
668/// Execute the given plan on the given arrays
669/**
670 * \param plan the pre-planned plan
671 * \param in the input data array
672 * \param out the output data array
673 *
674 * \tparam inputDataT the data type of the input array
675 * \tparam outputDataT the data type of the output array
676 */
677template <typename inputDataT, typename outputDataT>
678void fftw_execute_dft( typename fftwTypeSpec<inputDataT, outputDataT>::planT plan, inputDataT *in, outputDataT *out );
679
680template <>
681void fftw_execute_dft<complexFT, complexFT>( fftwTypeSpec<complexFT, complexFT>::planT plan,
682 complexFT *in,
683 complexFT *out );
684
685template <>
686void fftw_execute_dft<float, complexFT>( fftwTypeSpec<float, complexFT>::planT plan, float *in, complexFT *out );
687
688template <>
689void fftw_execute_dft<complexFT, float>( fftwTypeSpec<complexFT, float>::planT plan, complexFT *in, float *out );
690
691template <>
692void fftw_execute_dft<complexDT, complexDT>( fftwTypeSpec<complexDT, complexDT>::planT plan,
693 complexDT *in,
694 complexDT *out );
695
696template <>
697void fftw_execute_dft<double, complexDT>( fftwTypeSpec<double, complexDT>::planT plan, double *in, complexDT *out );
698
699template <>
700void fftw_execute_dft<complexDT, double>( fftwTypeSpec<complexDT, double>::planT plan, complexDT *in, double *out );
701
702//****** Plan Destruction *********//
703
704/// Destroy the given plan
705/**
706 * \param plan is the plan to destroy
707 *
708 * \tparam realT is the real floating point type of the plan
709 */
710template <typename realT>
712
713template <>
714void fftw_destroy_plan<float>( fftwPlanSpec<float>::planT plan );
715
716template <>
717void fftw_destroy_plan<double>( fftwPlanSpec<double>::planT plan );
718
719template <>
720void fftw_destroy_plan<long double>( fftwPlanSpec<long double>::planT plan );
721
722#ifdef HASQUAD
723template <>
724void fftw_destroy_plan<__float128>( fftwPlanSpec<__float128>::planT plan );
725#endif
726
727///@} fftw_template_exec
728
729} // namespace fft
730} // namespace math
731} // namespace mx
732
733#endif // fftwTemplates_hpp
realT * fftw_malloc(size_t n)
Call to fftw_malloc, with type cast.
void fftw_free(realT *p)
Call to fftw_free.
void fftw_destroy_plan(typename fftwPlanSpec< realT >::planT plan)
Destroy the given plan.
void fftw_execute_dft(typename fftwTypeSpec< inputDataT, outputDataT >::planT plan, inputDataT *in, outputDataT *out)
Execute the given plan on the given arrays.
void fftw_make_planner_thread_safe()
Wrapper for fftwX_make_planner_thread_safe()
void fftw_cleanup_threads()
Cleanup persistent planner data and threads data.
fftwTypeSpec< inputDataT, outputDataT >::planT fftw_plan_dft(std::vector< int > n, inputDataT *in, outputDataT *out, int sign, unsigned flags)
Wrapper for the fftwX_plan_dft functions.
void fftw_cleanup()
Cleanup persistent planner data.
void fftw_plan_with_nthreads(int nthreads)
Tell the FFTW planner how many threads to use.
std::complex< double > complexDT
The complex double data type.
std::complex< __float128 > complexQT
The complex __float128 data type.
std::complex< long double > complexLT
The complex long double data type.
std::complex< float > complexFT
The complex float data type.
int fftw_import_wisdom_from_filename(const char *filename)
Template wrapper for fftwX_import_wisdom_from_filename(const char *);.
int fftw_export_wisdom_to_filename(const char *filename)
Template wrapper for fftwX_export_wisdom_to_filename(const char *);.
int fftw_import_system_wisdom()
Template wrapper for fftwX_import_system_wisdom();.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
The mxlib c++ namespace.
Definition mxError.hpp:106
fftwq_plan planT
Specifies fftwq_plan as planT.
fftw_plan planT
Specifies fftw_plan as planT.
fftwf_plan planT
Specifies fftwf_plan as planT.
fftwl_plan planT
Specifies fftwl_plan as planT.
Specify the type of the plan based on the real type of the data.
fftwX_plan planT
Specializations typedef planT as fftwf_plan, fftw_plan, fftwl_plan, or fftwq_plan.
complexFT inputDataT
The input array data type.
fftwPlanSpec< float >::planT planT
The plan type.
complexFT outputDataT
The output array data type.
float inputDataT
The input array data type.
fftwPlanSpec< float >::planT planT
The plan type.
complexFT outputDataT
The output array data type.
A structure specifying various types based on the FFT input and output data types.
fftwPlanSpec< realT >::planT planT
The plan type.
std::complex< realT > complexT
The complex data type.
_outputDataT outputDataT
The output array data type.
_realT realT
The real data type (_realT is actually defined in specializations).
_inputDataT inputDataT
The input array data type.