PSCF v1.2
rpc/field/WFieldContainer.tpp
1#ifndef RPC_W_FIELD_CONTAINER_TPP
2#define RPC_W_FIELD_CONTAINER_TPP
3
4/*
5* PSCF - Polymer Self-Consistent Field Theory
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 "WFieldContainer.h"
12#include <rpc/field/FieldIo.h>
13
14namespace Pscf {
15namespace Rpc
16{
17
18 using namespace Util;
19 using namespace Pscf::Prdc;
20 using namespace Pscf::Prdc::Cpu;
21
22 /*
23 * Constructor.
24 */
25 template <int D>
27 : basis_(),
28 rgrid_(),
29 fieldIoPtr_(0),
30 meshDimensions_(),
31 meshSize_(0),
32 nBasis_(0),
33 nMonomer_(0),
34 isAllocatedRGrid_(false),
35 isAllocatedBasis_(false),
36 hasData_(false),
37 isSymmetric_(false)
38 {}
39
40 /*
41 * Destructor.
42 */
43 template <int D>
46
47 /*
48 * Create an association with a FieldIo object.
49 */
50 template <int D>
52 { fieldIoPtr_ = &fieldIo; }
53
54 /*
55 * Set the stored value of nMonomer (this may only be called once).
56 */
57 template <int D>
59 {
60 UTIL_CHECK(nMonomer_ == 0);
61 UTIL_CHECK(nMonomer > 0);
62 nMonomer_ = nMonomer;
63 }
64
65 /*
66 * Allocate memory for fields in r-grid format.
67 */
68 template <int D>
69 void
71 {
72 UTIL_CHECK(nMonomer_ > 0);
73
74 // If already allocated, deallocate.
75 if (isAllocatedRGrid_) {
76 deallocateRGrid();
77 }
78
79 // Store mesh dimensions
80 meshDimensions_ = meshDimensions;
81 meshSize_ = 1;
82 for (int i = 0; i < D; ++i) {
83 UTIL_CHECK(meshDimensions[i] > 0);
84 meshSize_ *= meshDimensions[i];
85 }
86
87 // Allocate arrays
88 rgrid_.allocate(nMonomer_);
89 for (int i = 0; i < nMonomer_; ++i) {
90 rgrid_[i].allocate(meshDimensions);
91 }
92 isAllocatedRGrid_ = true;
93
94 }
95
96 /*
97 * De-allocate memory for fields in r-grid format
98 */
99 template <int D>
101 {
102 UTIL_CHECK(isAllocatedRGrid_);
103 UTIL_CHECK(nMonomer_ > 0);
104 for (int i = 0; i < nMonomer_; ++i) {
105 rgrid_[i].deallocate();
106 }
107 rgrid_.deallocate();
108 meshDimensions_ = 0;
109 meshSize_ = 0;
110 isAllocatedRGrid_ = false;
111 }
112
113 /*
114 * Allocate memory for fields in basis format.
115 */
116 template <int D>
118 {
119 UTIL_CHECK(nMonomer_ > 0);
120 UTIL_CHECK(nBasis > 0);
121
122 // If already allocated, deallocate.
123 if (isAllocatedBasis_) {
124 deallocateBasis();
125 }
126
127 // Allocate
128 nBasis_ = nBasis;
129 basis_.allocate(nMonomer_);
130 for (int i = 0; i < nMonomer_; ++i) {
131 basis_[i].allocate(nBasis);
132 }
133 isAllocatedBasis_ = true;
134
135 }
136
137 /*
138 * De-allocate memory for fields in basis format.
139 */
140 template <int D>
142 {
143 UTIL_CHECK(isAllocatedBasis_);
144 UTIL_CHECK(nMonomer_ > 0);
145 for (int i = 0; i < nMonomer_; ++i) {
146 basis_[i].deallocate();
147 }
148 basis_.deallocate();
149 nBasis_ = 0;
150 isAllocatedBasis_ = false;
151 }
152
153 /*
154 * Allocate memory for all fields.
155 */
156 template <int D>
157 void WFieldContainer<D>::allocate(int nMonomer, int nBasis,
158 IntVec<D> const & meshDimensions)
159 {
160 setNMonomer(nMonomer);
161 allocateRGrid(meshDimensions);
162 allocateBasis(nBasis);
163 }
164
165 /*
166 * Set new w-field values.
167 */
168 template <int D>
169 void
171 {
172 UTIL_CHECK(fields.capacity() == nMonomer_);
173
174 // Update system w fields (array basis_)
175 for (int i = 0; i < nMonomer_; ++i) {
176 DArray<double> const & f = fields[i];
177 DArray<double> & w = basis_[i];
178 UTIL_CHECK(f.capacity() == nBasis_);
179 UTIL_CHECK(w.capacity() == nBasis_);
180 for (int j = 0; j < nBasis_; ++j) {
181 w[j] = f[j];
182 }
183 }
184
185 // Update system grid fields (array rgrid_)
186 fieldIoPtr_->convertBasisToRGrid(basis_, rgrid_);
187
188 hasData_ = true;
189 isSymmetric_ = true;
190 }
191
192 /*
193 * Set new field values, using r-grid fields as inputs.
194 */
195 template <int D>
197 bool isSymmetric)
198 {
199 UTIL_CHECK(fields.capacity() == nMonomer_);
200
201 // Update system grid fields (array rgrid_ )
202 for (int i = 0; i < nMonomer_; ++i) {
203 RField<D> const & f = fields[i];
204 RField<D>& w = rgrid_[i];
205 UTIL_CHECK(f.capacity() == meshSize_);
206 UTIL_CHECK(w.capacity() == meshSize_);
207 for (int j = 0; j < meshSize_; ++j) {
208 w[j] = f[j];
209 }
210 }
211
212 // If field isSymmetric, update basis fields
213 if (isSymmetric) {
214 fieldIoPtr_->convertRGridToBasis(rgrid_, basis_);
215 }
216
217 hasData_ = true;
218 isSymmetric_ = isSymmetric;
219 }
220
221 /*
222 * Read field component values from input stream, in symmetrized
223 * Fourier format.
224 *
225 * This function also computes and stores the corresponding
226 * r-grid representation. On return, hasData and isSymmetric
227 * are both true.
228 */
229 template <int D>
230 void WFieldContainer<D>::readBasis(std::istream& in,
231 UnitCell<D>& unitCell)
232 {
233 UTIL_CHECK(isAllocatedBasis());
234 fieldIoPtr_->readFieldsBasis(in, basis_, unitCell);
235
236 // Update system wFieldsRGrid
237 fieldIoPtr_->convertBasisToRGrid(basis_, rgrid_);
238
239 hasData_ = true;
240 isSymmetric_ = true;
241 }
242
243 /*
244 * Read field component values from file, in symmetrized
245 * Fourier format.
246 *
247 * This function also computes and stores the corresponding
248 * r-grid representation. On return, hasData and isSymmetric
249 * are both true.
250 */
251 template <int D>
252 void WFieldContainer<D>::readBasis(std::string filename,
253 UnitCell<D>& unitCell)
254 {
255 UTIL_CHECK(isAllocatedBasis());
256 fieldIoPtr_->readFieldsBasis(filename, basis_, unitCell);
257
258 // Update system wFieldsRGrid
259 fieldIoPtr_->convertBasisToRGrid(basis_, rgrid_);
260
261 hasData_ = true;
262 isSymmetric_ = true;
263 }
264
265 /*
266 * Reads fields from an input stream in real-space (r-grid) format.
267 *
268 * If the isSymmetric parameter is true, this function assumes that
269 * the fields are known to be symmetric and so computes and stores
270 * the corresponding basis components. If isSymmetric is false, it
271 * only sets the values in the r-grid format.
272 *
273 * On return, hasData is true and the persistent isSymmetric flag
274 * defined by the class is set to the value of the isSymmetric
275 * input parameter.
276 */
277 template <int D>
278 void WFieldContainer<D>::readRGrid(std::istream& in,
279 UnitCell<D>& unitCell,
280 bool isSymmetric)
281 {
282 UTIL_CHECK(isAllocatedRGrid());
283 fieldIoPtr_->readFieldsRGrid(in, rgrid_, unitCell);
284
285 if (isSymmetric) {
286 fieldIoPtr_->convertRGridToBasis(rgrid_, basis_);
287 }
288
289 hasData_ = true;
290 isSymmetric_ = isSymmetric;
291 }
292
293 /*
294 * Reads fields from a file in real-space (r-grid) format.
295 *
296 * If the isSymmetric parameter is true, this function assumes that
297 * the fields are known to be symmetric and so computes and stores
298 * the corresponding basis components. If isSymmetric is false, it
299 * only sets the values in the r-grid format.
300 *
301 * On return, hasData is true and the persistent isSymmetric flag
302 * defined by the class is set to the value of the isSymmetric
303 * input parameter.
304 */
305 template <int D>
306 void WFieldContainer<D>::readRGrid(std::string filename,
307 UnitCell<D>& unitCell,
308 bool isSymmetric)
309 {
310 UTIL_CHECK(isAllocatedRGrid());
311 fieldIoPtr_->readFieldsRGrid(filename, rgrid_, unitCell);
312
313 if (isSymmetric) {
314 fieldIoPtr_->convertRGridToBasis(rgrid_, basis_);
315 }
316
317 hasData_ = true;
318 isSymmetric_ = isSymmetric;
319 }
320
321} // namespace Rpc
322} // namespace Pscf
323#endif
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Field of real double precision values on an FFT mesh.
Base template for UnitCell<D> classes, D=1, 2 or 3.
Definition rpg/System.h:34
File input/output operations and format conversions for fields.
void allocate(int nMonomer, int nBasis, IntVec< D > const &dimensions)
Allocate memory for all fields.
void allocateRGrid(IntVec< D > const &dimensions)
Allocate or re-allocate memory for fields in rgrid format.
void readBasis(std::istream &in, UnitCell< D > &unitCell)
Read field component values from input stream, in symmetrized Fourier format.
void deallocateBasis()
De-allocate fields in basis format.
void allocateBasis(int nBasis)
Allocate or re-allocate memory for fields in basis format.
void deallocateRGrid()
De-allocate fields in rgrid format.
void setFieldIo(FieldIo< D > const &fieldIo)
Create association with FieldIo (store pointer).
void setRGrid(DArray< RField< D > > const &fields, bool isSymmetric=false)
Set fields values in real-space (r-grid) format.
void readRGrid(std::istream &in, UnitCell< D > &unitCell, bool isSymmetric=false)
Reads fields from an input stream in real-space (r-grid) format.
void setBasis(DArray< DArray< double > > const &fields)
Set field component values, in symmetrized Fourier format.
void setNMonomer(int nMonomer)
Set stored value of nMonomer.
int capacity() const
Return allocated size.
Definition Array.h:159
Dynamically allocatable contiguous array template.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
Fields and FFTs for periodic boundary conditions (CPU)
Definition CField.cpp:12
Periodic fields and crystallography.
Definition CField.cpp:11
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.