PSCF v1.2
rpc/scft/iterator/AmIteratorBasis.tpp
1#ifndef RPC_AM_ITERATOR_BASIS_TPP
2#define RPC_AM_ITERATOR_BASIS_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 "AmIteratorBasis.h"
12#include <rpc/System.h>
13#include <pscf/inter/Interaction.h>
14#include <pscf/iterator/NanException.h>
15#include <util/global.h>
16#include <cmath>
17
18namespace Pscf{
19namespace Rpc {
20
21 using namespace Util;
22
23 // Constructor
24 template <int D>
26 : Iterator<D>(system),
27 imposedFields_(system)
28 {
29 isSymmetric_ = true;
30 setClassName("AmIteratorBasis");
31 }
32
33 // Destructor
34 template <int D>
37
38 // Read parameters from file
39 template <int D>
41 {
42 // Call parent class readParameters
43 AmIteratorTmpl<Iterator<D>, DArray<double> >::readParameters(in);
44 AmIteratorTmpl<Iterator<D>, DArray<double> >::readErrorType(in);
45
46 // Allocate local modified copy of Interaction class
47 interaction_.setNMonomer(system().mixture().nMonomer());
48
49 // Default parameter values
50 isFlexible_ = 1;
51 scaleStress_ = 10.0;
52
53 int np = system().domain().unitCell().nParameter();
54 UTIL_CHECK(np > 0);
55 UTIL_CHECK(np <= 6);
56 UTIL_CHECK(system().domain().unitCell().lattice() != UnitCell<D>::Null);
57
58 // Read optional isFlexible boolean (true by default)
59 readOptional(in, "isFlexible", isFlexible_);
60
61 // Populate flexibleParams_ based on isFlexible_ (all 0s or all 1s),
62 // then optionally overwrite with user input from param file
63 if (isFlexible_) {
64 flexibleParams_.clear();
65 for (int i = 0; i < np; i++) {
66 flexibleParams_.append(true); // Set all values to true
67 }
68 // Read optional flexibleParams_ array to overwrite current array
69 readOptionalFSArray(in, "flexibleParams", flexibleParams_, np);
70 if (nFlexibleParams() == 0) isFlexible_ = false;
71 } else { // isFlexible_ = false
72 flexibleParams_.clear();
73 for (int i = 0; i < np; i++) {
74 flexibleParams_.append(false); // Set all values to false
75 }
76 }
77
78 // Read optional scaleStress value
79 readOptional(in, "scaleStress", scaleStress_);
80
81 // Read optional ImposedFieldsGenerator object
82 readParamCompositeOptional(in, imposedFields_);
83 }
84
85 // Output timing results to log file.
86 template<int D>
87 void AmIteratorBasis<D>::outputTimers(std::ostream& out)
88 {
89 // Output timing results, if requested.
90 out << "\n";
91 out << "Iterator times contributions:\n";
92 AmIteratorTmpl<Iterator<D>, DArray<double> >::outputTimers(out);
93 }
94
95 // Protected virtual function
96
97 // Setup before entering iteration loop
98 template <int D>
99 void AmIteratorBasis<D>::setup(bool isContinuation)
100 {
101 if (imposedFields_.isActive()) {
102 imposedFields_.setup();
103 }
104
105 AmIteratorTmpl<Iterator<D>, DArray<double> >::setup(isContinuation);
106 interaction_.update(system().interaction());
107 }
108
109 // Private virtual functions used to implement AM algorithm
110
111 // Assign one array to another
112 template <int D>
114 DArray<double> const & b)
115 { a = b; }
116
117 // Compute and return inner product of two vectors.
118 template <int D>
119 double AmIteratorBasis<D>::dotProduct(DArray<double> const & a,
120 DArray<double> const & b)
121 {
122 const int n = a.capacity();
123 UTIL_CHECK(b.capacity() == n);
124 double product = 0.0;
125 for (int i = 0; i < n; i++) {
126 // if either value is NaN, throw NanException
127 if (std::isnan(a[i]) || std::isnan(b[i])) {
128 throw NanException("AmIteratorBasis::dotProduct", __FILE__,
129 __LINE__, 0);
130 }
131 product += a[i] * b[i];
132 }
133 return product;
134 }
135
136 // Compute and return maximum element of a vector.
137 template <int D>
138 double AmIteratorBasis<D>::maxAbs(DArray<double> const & a)
139 {
140 const int n = a.capacity();
141 double max = 0.0;
142 double value;
143 for (int i = 0; i < n; i++) {
144 value = a[i];
145 if (std::isnan(value)) { // if value is NaN, throw NanException
146 throw NanException("AmIteratorBasis::dotProduct", __FILE__,
147 __LINE__, 0);
148 }
149 if (fabs(value) > max) {
150 max = fabs(value);
151 }
152 }
153 return max;
154 }
155
156 // Update basis
157 template <int D>
158 void
159 AmIteratorBasis<D>::updateBasis(RingBuffer< DArray<double> > & basis,
160 RingBuffer< DArray<double> > const & hists)
161 {
162 // Make sure at least two histories are stored
163 UTIL_CHECK(hists.size() >= 2);
164
165 const int n = hists[0].capacity();
166 DArray<double> newbasis;
167 newbasis.allocate(n);
168
169 // New basis vector is difference between two most recent states
170 for (int i = 0; i < n; i++) {
171 newbasis[i] = hists[0][i] - hists[1][i];
172 }
173 basis.append(newbasis);
174 }
175
176 // Add linear combination of basis vectors to trial field.
177 template <int D>
178 void
179 AmIteratorBasis<D>::addHistories(DArray<double>& trial,
180 RingBuffer<DArray<double> > const & basis,
181 DArray<double> coeffs,
182 int nHist)
183 {
184 int n = trial.capacity();
185 for (int i = 0; i < nHist; i++) {
186 for (int j = 0; j < n; j++) {
187 // Not clear on the origin of the -1 factor
188 trial[j] += coeffs[i] * -1 * basis[i][j];
189 }
190 }
191 }
192
193 // Add predicted error to field trial.
194 template <int D>
195 void AmIteratorBasis<D>::addPredictedError(DArray<double>& fieldTrial,
196 DArray<double> const & resTrial,
197 double lambda)
198 {
199 int n = fieldTrial.capacity();
200 for (int i = 0; i < n; i++) {
201 fieldTrial[i] += lambda * resTrial[i];
202 }
203 }
204
205 // Private virtual functions to exchange data with parent system
206
207 // Does the system have an initial field guess?
208 template <int D>
209 bool AmIteratorBasis<D>::hasInitialGuess()
210 { return system().w().hasData(); }
211
212 // Compute and return number of elements in a residual vector
213 template <int D>
214 int AmIteratorBasis<D>::nElements()
215 {
216 const int nMonomer = system().mixture().nMonomer();
217 const int nBasis = system().domain().basis().nBasis();
218
219 int nEle = nMonomer*nBasis;
220 if (isFlexible()) {
221 nEle += nFlexibleParams();
222 }
223
224 return nEle;
225 }
226
227 // Get the current w fields and lattice parameters
228 template <int D>
229 void AmIteratorBasis<D>::getCurrent(DArray<double>& curr)
230 {
231 const int nMonomer = system().mixture().nMonomer();
232 const int nBasis = system().domain().basis().nBasis();
233 const DArray< DArray<double> > & currSys = system().w().basis();
234
235 // Straighten out fields into linear arrays
236 for (int i = 0; i < nMonomer; i++) {
237 for (int k = 0; k < nBasis; k++) {
238 curr[i*nBasis+k] = currSys[i][k];
239 }
240 }
241
242 // Add elements associated with unit cell parameters (if any)
243 if (isFlexible()) {
244 const int nParam = system().domain().unitCell().nParameter();
245 const FSArray<double,6> currParam
246 = system().domain().unitCell().parameters();
247 int counter = 0;
248 for (int i = 0; i < nParam; i++) {
249 if (flexibleParams_[i]) {
250 curr[nMonomer*nBasis + counter] = scaleStress_*currParam[i];
251 counter++;
252 }
253 }
254 UTIL_CHECK(counter == nFlexibleParams());
255 }
256
257 }
258
259 // Perform the main system computation (solve the MDE)
260 template <int D>
261 void AmIteratorBasis<D>::evaluate()
262 {
263 // Solve MDEs for current omega field
264 // (computes stress if isFlexible_ == true)
265 system().compute(isFlexible_);
266 }
267
268 // Compute the residual for the current system state
269 template <int D>
270 void AmIteratorBasis<D>::getResidual(DArray<double>& resid)
271 {
272 const int n = nElements();
273 const int nMonomer = system().mixture().nMonomer();
274 const int nBasis = system().domain().basis().nBasis();
275
276 // Initialize residual vector to zero
277 for (int i = 0 ; i < n; ++i) {
278 resid[i] = 0.0;
279 }
280
281 // Compute SCF residual vector elements
282 for (int i = 0; i < nMonomer; ++i) {
283 for (int j = 0; j < nMonomer; ++j) {
284 double chi = interaction_.chi(i,j);
285 double p = interaction_.p(i,j);
286 DArray<double> const & c = system().c().basis(j);
287 DArray<double> const & w = system().w().basis(j);
288 for (int k = 0; k < nBasis; ++k) {
289 int idx = i*nBasis + k;
290 resid[idx] += chi*c[k] - p*w[k];
291 }
292 }
293 }
294
295 // If iterator has mask, account for it in residual values
296 if (system().hasMask()) {
297 DArray<double> const & mask = system().mask().basis();
298 double sumChiInv = interaction_.sumChiInverse();
299 for (int i = 0; i < nMonomer; ++i) {
300 for (int k = 0; k < nBasis; ++k) {
301 int idx = i*nBasis + k;
302 resid[idx] -= mask[k] / sumChiInv;
303 }
304 }
305 }
306
307 // If iterator has external fields, account for them in the values
308 // of the residuals
309 if (system().hasExternalFields()) {
310 for (int i = 0; i < nMonomer; ++i) {
311 for (int j = 0; j < nMonomer; ++j) {
312 double p = interaction_.p(i,j);
313 DArray<double> const & h = system().h().basis(j);
314 for (int k = 0; k < nBasis; ++k) {
315 int idx = i*nBasis + k;
316 resid[idx] += p * h[k];
317 }
318 }
319 }
320 }
321
322 // If not canonical, account for incompressibility
323 if (!system().mixture().isCanonical()) {
324 if (!system().hasMask()) {
325 for (int i = 0; i < nMonomer; ++i) {
326 resid[i*nBasis] -= 1.0 / interaction_.sumChiInverse();
327 }
328 }
329 } else {
330 // Explicitly set homogeneous residual components
331 for (int i = 0; i < nMonomer; ++i) {
332 resid[i*nBasis] = 0.0;
333 }
334 }
335
336 // If variable unit cell, compute stress residuals
337 if (isFlexible()) {
338 const int nParam = system().domain().unitCell().nParameter();
339
340 // Combined -1 factor and stress scaling here. This is okay:
341 // - residuals only show up as dot products (U, v, norm)
342 // or with their absolute value taken (max), so the
343 // sign on a given residual vector element is not relevant
344 // as long as it is consistent across all vectors
345 // - The scaling is applied here and to the unit cell param
346 // storage, so that updating is done on the same scale,
347 // and then undone right before passing to the unit cell.
348
349 int counter = 0;
350 for (int i = 0; i < nParam ; i++) {
351 if (flexibleParams_[i]) {
352 double stress = system().mixture().stress(i);
353
354 // Correct stress to account for effect of imposed fields
355 if (imposedFields_.isActive()) {
356 stress = imposedFields_.correctedStress(i,stress);
357 }
358
359 resid[nMonomer*nBasis + counter] = -1 * scaleStress_ * stress;
360 counter++;
361 }
362 }
363 UTIL_CHECK(counter == nFlexibleParams());
364 }
365
366 }
367
368 // Update the current system field coordinates
369 template <int D>
370 void AmIteratorBasis<D>::update(DArray<double>& newGuess)
371 {
372 // Convert back to field format
373 const int nMonomer = system().mixture().nMonomer();
374 const int nBasis = system().domain().basis().nBasis();
375
376 DArray< DArray<double> > wField;
377 wField.allocate(nMonomer);
378
379 // Restructure in format of monomers, basis functions
380 for (int i = 0; i < nMonomer; i++) {
381 wField[i].allocate(nBasis);
382 for (int k = 0; k < nBasis; k++)
383 {
384 wField[i][k] = newGuess[i*nBasis + k];
385 }
386 }
387 // If canonical, explicitly set homogeneous field components
388 if (system().mixture().isCanonical()) {
389 double chi;
390 for (int i = 0; i < nMonomer; ++i) {
391 wField[i][0] = 0.0; // initialize to 0
392 for (int j = 0; j < nMonomer; ++j) {
393 chi = interaction_.chi(i,j);
394 wField[i][0] += chi * system().c().basis(j)[0];
395 }
396 }
397 // If iterator has external fields, include them in homogeneous field
398 if (system().hasExternalFields()) {
399 for (int i = 0; i < nMonomer; ++i) {
400 wField[i][0] += system().h().basis(i)[0];
401 }
402 }
403 }
404 system().setWBasis(wField);
405
406 if (isFlexible()) {
407 const int nParam = system().domain().unitCell().nParameter();
408 const int begin = nMonomer*nBasis;
409
410 FSArray<double,6> parameters;
411 parameters = system().domain().unitCell().parameters();
412
413 double coeff = 1.0 / scaleStress_;
414 int counter = 0;
415 for (int i = 0; i < nParam; i++) {
416 if (flexibleParams_[i]) {
417 parameters[i] = coeff * newGuess[begin + counter];
418 counter++;
419 }
420 }
421 UTIL_CHECK(counter == nFlexibleParams());
422
423 system().setUnitCell(parameters);
424 }
425
426 // Update imposed fields if needed
427 if (imposedFields_.isActive()) {
428 imposedFields_.update();
429 }
430 }
431
432 // Output relevant system details to the iteration log.
433 template<int D>
434 void AmIteratorBasis<D>::outputToLog()
435 {
436 if (isFlexible() && verbose() > 1) {
437 const int nParam = system().domain().unitCell().nParameter();
438 const int nMonomer = system().mixture().nMonomer();
439 const int nBasis = system().domain().basis().nBasis();
440 int counter = 0;
441 for (int i = 0; i < nParam; i++) {
442 if (flexibleParams_[i]) {
443 double stress = residual()[nMonomer*nBasis + counter] /
444 (-1.0 * scaleStress_);
445 Log::file()
446 << " Cell Param " << i << " = "
447 << Dbl(system().domain().unitCell().parameters()[i], 15)
448 << " , stress = "
449 << Dbl(stress, 15)
450 << "\n";
451 counter++;
452 }
453 }
454 }
455 }
456
457 // Return specialized sweep parameter types to add to the Sweep object
458 template<int D>
460 {
462 if (imposedFields_.isActive()) {
463 arr = imposedFields_.getParameterTypes();
464 }
465 return arr;
466 }
467
468 // Set the value of a specialized sweep parameter
469 template<int D>
471 double value, bool& success)
472 {
473 if (imposedFields_.isActive()) {
474 imposedFields_.setParameter(name, ids, value, success);
475 } else {
476 success = false;
477 }
478 }
479
480 // Get the value of a specialized sweep parameter
481 template<int D>
482 double AmIteratorBasis<D>::getParameter(std::string name,
483 DArray<int> ids, bool& success)
484 const
485 {
486 if (imposedFields_.isActive()) {
487 return imposedFields_.getParameter(name, ids, success);
488 } else {
489 success = false;
490 return 0.0;
491 }
492 }
493
494}
495}
496#endif
Template for Anderson mixing iterator algorithm.
Exception thrown when not-a-number (NaN) is encountered.
Base template for UnitCell<D> classes, D=1, 2 or 3.
Definition rpg/System.h:34
Rpc implementation of the Anderson Mixing iterator with symmetry.
void setup(bool isContinuation)
Setup iterator just before entering iteration loop.
GArray< ParameterType > getParameterTypes()
Return specialized sweep parameter types to add to the Sweep object.
double getParameter(std::string name, DArray< int > ids, bool &success) const
Get the value of a specialized sweep parameter.
void setClassName(const char *className)
Set class name string.
AmIteratorBasis(System< D > &system)
Constructor.
void readParameters(std::istream &in)
Read all parameters and initialize.
void setParameter(std::string name, DArray< int > ids, double value, bool &success)
Set the value of a specialized sweep parameter.
void outputTimers(std::ostream &out)
Output timing results to log file.
Base class for iterative solvers for SCF equations.
Definition rpc/System.h:34
Main class for SCFT or PS-FTS simulation of one system.
Definition rpc/System.h:100
int capacity() const
Return allocated size.
Definition Array.h:159
Dynamically allocatable contiguous array template.
void allocate(int capacity)
Allocate the underlying C array.
Definition DArray.h:199
Wrapper for a double precision number, for formatted ostream output.
Definition Dbl.h:40
A fixed capacity (static) contiguous array with a variable logical size.
Definition rpg/System.h:28
An automatically growable array, analogous to a std::vector.
Definition GArray.h:34
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
Class for storing history of previous values in an array.
Definition RingBuffer.h:27
File containing preprocessor macros for error handling.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
cudaReal max(DeviceArray< cudaReal > const &in)
Get maximum of array elements (GPU kernel wrapper).
Definition Reduce.cu:569
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.
float product(float a, float b)
Product for float Data.
Definition product.h:22