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