PSCF v1.2
cpu/RFieldDft.tpp
1#ifndef PRDC_CPU_R_FIELD_DFT_TPP
2#define PRDC_CPU_R_FIELD_DFT_TPP
3
4/*
5* PSCF Package
6*
7* Copyright 2016 - 2022, The Regents of the University of Minnesota
8* Distributed under the terms of the GNU General Public License.
9*/
10
11#include "RFieldDft.h"
12
13namespace Pscf {
14namespace Prdc {
15namespace Cpu {
16
17 using namespace Util;
18
22 template <int D>
24 : FftwDArray<fftw_complex>(),
25 meshDimensions_(0),
26 dftDimensions_(0)
27 {}
28
29 /*
30 * Destructor.
31 */
32 template <int D>
35
36 /*
37 * Copy constructor.
38 *
39 * Allocates new memory and copies all elements by value.
40 */
41 template <int D>
43 : FftwDArray<fftw_complex>(),
44 meshDimensions_(0),
45 dftDimensions_(0)
46 {
47 if (!other.isAllocated()) {
48 UTIL_THROW("Other Field must be allocated.");
49 }
50 allocate(other.meshDimensions_);
51 for (int i = 0; i < capacity_; ++i) {
52 data_[i][0] = other.data_[i][0];
53 data_[i][1] = other.data_[i][1];
54 }
55 }
56
57 /*
58 * Assignment, element-by-element.
59 *
60 * This operator will allocate memory if not allocated previously.
61 *
62 * \throw Exception if other Field is not allocated.
63 * \throw Exception if both Fields are allocated with unequal capacities.
64 *
65 * \param other the rhs Field
66 */
67 template <int D>
69 {
70 // Check for self assignment
71 if (this == &other) return *this;
72
73 // Precondition
74 if (!other.isAllocated()) {
75 UTIL_THROW("Other RFieldDft must be allocated.");
76 }
77
78 // Allocate if necessary, check dimensions
79 if (!isAllocated()) {
80 allocate(other.meshDimensions_);
81 }
82 UTIL_CHECK(capacity_ == other.capacity_);
83 UTIL_CHECK(meshDimensions_ == other.meshDimensions_);
84 UTIL_CHECK(dftDimensions_ == other.dftDimensions_);
85
86 // Copy elements
87 for (int i = 0; i < capacity_; ++i) {
88 data_[i][0] = other.data_[i][0];
89 data_[i][1] = other.data_[i][1];
90 }
91
92 return *this;
93 }
94
95 /*
96 * Allocate the underlying C array for an FFT grid.
97 */
98 template <int D>
99 void RFieldDft<D>::allocate(IntVec<D> const & meshDimensions)
100 {
101 int size = 1;
102 for (int i = 0; i < D; ++i) {
103 UTIL_CHECK(meshDimensions[i] > 0);
104 meshDimensions_[i] = meshDimensions[i];
105 if (i < D - 1) {
106 dftDimensions_[i] = meshDimensions[i];
107 size *= meshDimensions[i];
108 } else {
109 dftDimensions_[i] = (meshDimensions[i]/2 + 1);
110 size *= dftDimensions_[i];
111 }
112 }
114 }
115
116 /*
117 * Dellocate the underlying C array and clear dimensions.
118 */
119 template <int D>
121 {
123 for (int i = 0; i < D; ++i) {
124 meshDimensions_[i] = 0;
125 dftDimensions_[i] = 0;
126 }
127 }
128
129}
130}
131}
132#endif
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Dynamic array with data aligned for use with FFTW library.
Definition FftwDArray.h:33
void allocate(int capacity)
Allocate the underlying C array.
virtual void deallocate()
Dellocate the underlying C array.
bool isAllocated() const
Return true if the FftwDArray has been allocated, false otherwise.
Definition FftwDArray.h:102
Fourier transform of a real field on an FFT mesh.
virtual void deallocate()
Deallocate underlying C array and clear mesh dimensions.
void allocate(IntVec< D > const &meshDimensions)
Allocate the underlying C array and set mesh dimensions.
virtual ~RFieldDft()
Destructor.
RFieldDft< D > & operator=(RFieldDft< D > const &other)
Assignment operator.
RFieldDft()
Default constructor.
fftw_complex * data_
Definition Array.h:109
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Definition global.h:51
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.