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