PSCF v1.2
AmIteratorTmpl.tpp
1#ifndef PSCF_AM_ITERATOR_TMPL_TPP
2#define PSCF_AM_ITERATOR_TMPL_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 <pscf/inter/Interaction.h>
12#include <pscf/math/LuSolver.h>
13#include "NanException.h"
14#include <util/containers/FArray.h>
15#include <util/format/Dbl.h>
16#include <util/format/Int.h>
17#include <util/misc/Timer.h>
18#include <util/misc/FileMaster.h>
19#include <cmath>
20
21namespace Pscf
22{
23
24 using namespace Util;
25
26 // Public member functions
27
28 /*
29 * Constructor
30 */
31 template <typename Iterator, typename T>
33 : errorType_("relNormResid"),
34 epsilon_(0),
35 lambda_(0),
36 r_(0.9),
37 maxItr_(200),
38 maxHist_(50),
39 nBasis_(0),
40 totalItr_(0),
41 nElem_(0),
42 verbose_(1),
43 isAllocatedAM_(false)
44 { setClassName("AmIteratorTmpl"); }
45
46 /*
47 * Destructor
48 */
49 template <typename Iterator, typename T>
52
53 /*
54 * Read parameter file block.
55 */
56 template <typename Iterator, typename T>
58 {
59 // Error tolerance for scalar error. Stop when error < epsilon.
60 read(in, "epsilon", epsilon_);
61
62 // Maximum number of iterations (optional parameter)
63 // Default set in constructor (200) or by prior call to setMaxItr
64 readOptional(in, "maxItr", maxItr_);
65
66 // Maximum number of previous states in history (optional)
67 // Default set in constructor (50) or by prior call to setMaxHist
68 readOptional(in, "maxHist", maxHist_);
69
70 // Verbosity level of error reporting, values 0-3 (optional)
71 // Initialized to 1 by default in constructor
72 // verbose_ = 0 => FTS
73 // verbose_ = 1 => concise
74 // verbose_ = 2 => report all error measures at end
75 // verbose_ = 3 => report all error measures every iteration
76 readOptional(in, "verbose", verbose_);
77
78 // Ramping parameter for correction step
79 // Default set to 0.9. lambda = 1 - r_^Nh for Nh < maxHist
80 readOptional(in, "correctionRamp", r_);
82 #ifdef PSCF_AM_TEST
83 // Default set to be false
84 readOptional(in, "hasAmTest", hasAmTest_);
85 #endif
86 }
87
88 /*
89 * Iteratively solve a fixed-point problem.
90 */
91 template <typename Iterator, typename T>
92 int AmIteratorTmpl<Iterator,T>::solve(bool isContinuation)
93 {
94 // Initialization and allocate operations on entry to loop.
95 setup(isContinuation);
96
97 // Preconditions for generic Anderson-mixing (AM) algorithm.
98 UTIL_CHECK(hasInitialGuess());
99 UTIL_CHECK(isAllocatedAM_);
100
101 // Start overall timer
102 timerTotal_.start();
103
104 // Solve MDE for initial state
105 timerMDE_.start();
106 evaluate();
107 timerMDE_.stop();
108
109 // Iterative loop
110 nBasis_ = fieldBasis_.size();
111 for (itr_ = 0; itr_ < maxItr_; ++itr_) {
112
113 // Append current field to fieldHists_ ringbuffer
114 getCurrent(temp_);
115 fieldHists_.append(temp_);
116
117 timerAM_.start();
119 if (verbose_ > 2) {
120 Log::file() << "------------------------------- \n";
121 }
122
123 if (verbose_ > 0){
124 Log::file() << " Iteration " << Int(itr_,5);
125 }
126
127 lambda_ = computeLambda(r_);
128
129 // Compute current residual vector, store in temp_
130 timerResid_.start();
131 getResidual(temp_);
132
133 // Append current residual to resHists_ ring buffer
134 resHists_.append(temp_);
135 timerResid_.stop();
137 // Compute scalar error used to test convergence
138 timerError_.start();
139 try {
140 error_ = computeError(verbose_);
141 } catch (const NanException&) {
142 Log::file() << ", error = NaN" << std::endl;
143 break; // Exit loop if a NanException is caught
144 }
145 if (verbose_ > 0 && verbose_ < 3) {
146 Log::file() << ", error = " << Dbl(error_, 15)
147 << std::endl;
148 }
149 timerError_.stop();
150
151 // Output additional details of this iteration to the log file
152 outputToLog();
153
154 #ifdef PSCF_AM_TEST
155 // Compute errors and ratios used for algorithm testing
156 if (hasAmTest_){
157 // Compute errors and ratios used for algorithm testing
158 correctionError_ = computeError(0);
159 if (itr_ > 0) {
160 projectionRatio_ += projectionError_/preError_;
161 correctionRatio_ += correctionError_/preError_;
162 testCounter ++;
164 preError_ = correctionError_;
165 }
166 #endif
167
168 // Check for convergence
169 if (error_ < epsilon_) {
170
171 // Stop timers
172 timerAM_.stop();
173 timerTotal_.stop();
174
175 if (verbose_ > 2) {
176 Log::file() << "-------------------------------\n";
177 }
178
179 if (verbose_ > 0) {
180 Log::file() << " Converged\n";
181 }
182
183 // Output error report if not done previously
184 if (verbose_ == 2) {
185 Log::file() << "\n";
186 computeError(2);
187 }
188
189 totalItr_ += itr_;
190
191 // Successful completion (i.e., converged within tolerance)
192 return 0;
193
194 } else {
195
196 // Compute optimal coefficients for basis vectors
197 timerCoeff_.start();
198 computeResidCoeff();
199 timerCoeff_.stop();
200
201 // Compute updated field and update the system
202 timerOmega_.start();
203 updateGuess();
204 timerOmega_.stop();
205
206 timerAM_.stop();
207
208 // Perform the main calculation of the parent system -
209 // Solve MDEs, compute phi's, compute stress if needed
210 timerMDE_.start();
211 evaluate();
212 timerMDE_.stop();
213 }
214
215 }
216
217 // Failure: Counter itr_ reached maxItr_ without converging
218 timerTotal_.stop();
219 Log::file() << "Iterator failed to converge.\n";
220 return 1;
221
222 }
223
224 // Protected member functions
225
226 /*
227 * Set value of maxItr.
228 */
229 template <typename Iterator, typename T>
231 { maxItr_ = maxItr; }
232
233 /*
234 * Set value of maxHist (number of retained previous states)
235 */
236 template <typename Iterator, typename T>
238 { maxHist_ = maxHist; }
239
240 /*
241 * Set and validate value of error type string.
242 */
243 template <typename Iterator, typename T>
245 {
246 errorType_ = errorType;
247
248 // Check validity, normalize to standard form
249 bool isValid = isValidErrorType();
250
251 // If not valid, throw Exception with error message
252 if (!isValid) {
253 std::string msg = "Invalid iterator error type [";
254 msg += errorType;
255 msg += "] input in AmIteratorTmpl::setErrorType";
256 UTIL_THROW(msg.c_str());
257 }
258 }
259
260 /*
261 * Read and validate optional errorType string parameter.
262 */
263 template <typename Iterator, typename T>
265 {
266 // Note: errorType_ is initialized to "relNormResid" in constructor
267
268 // Read optional errorType_ string if present
269 readOptional(in, "errorType", errorType_);
270
271 // Check validity, normalize to standard form
272 bool isValid = isValidErrorType();
273
274 // If not valid, throw Exception with error message
275 if (!isValid) {
276 std::string msg = "Invalid iterator error type [";
277 msg += errorType_;
278 msg += "] in parameter file";
279 UTIL_THROW(msg.c_str());
280 }
281
282 }
283
284 /*
285 * Check validity of errorType_ string and normalize if valid.
286 */
287 template <typename Iterator, typename T>
289 {
290 // Process possible synonyms
291 if (errorType_ == "norm") errorType_ = "normResid";
292 if (errorType_ == "rms") errorType_ = "rmsResid";
293 if (errorType_ == "max") errorType_ = "maxResid";
294 if (errorType_ == "relNorm") errorType_ = "relNormResid";
295
296 // Check value
297 bool valid;
298 valid = (errorType_ == "normResid"
299 || errorType_ == "rmsResid"
300 || errorType_ == "maxResid"
301 || errorType_ == "relNormResid");
302 return valid;
303
304 }
305
306 /*
307 * Allocate memory required by the Anderson-Mixing algorithm, if needed.
308 * (protected, non-virtual)
309 */
310 template <typename Iterator, typename T>
312 {
313 // If already allocated, do nothing and return
314 if (isAllocatedAM_) return;
315
316 // Compute and set number of elements in a residual vector
317 nElem_ = nElements();
318
319 // Allocate ring buffers
320 fieldHists_.allocate(maxHist_+1);
321 resHists_.allocate(maxHist_+1);
322 fieldBasis_.allocate(maxHist_);
323 resBasis_.allocate(maxHist_);
324
325 // Allocate arrays used in iteration
326 fieldTrial_.allocate(nElem_);
327 resTrial_.allocate(nElem_);
328 temp_.allocate(nElem_);
329
330 // Allocate arrays/matrices used in coefficient calculation
331 U_.allocate(maxHist_, maxHist_);
332 v_.allocate(maxHist_);
333 coeffs_.allocate(maxHist_);
334
335 isAllocatedAM_ = true;
336 }
337
338 /*
339 * Clear all history and basis vector data.
340 */
341 template <typename Iterator, typename T>
343 {
344 if (!isAllocatedAM_) return;
345
346 // Clear histories and bases (ring buffers)
347 if (verbose_ > 0) {
348 Log::file() << "Clearing AM field history and basis vectors.\n";
349 }
350 resHists_.clear();
351 fieldHists_.clear();
352 resBasis_.clear();
353 fieldBasis_.clear();
354
355 return;
356 }
357
358 // Private non-virtual member functions
359
360 /*
361 * Compute optimal coefficients of basis vectors.
362 */
363 template <typename Iterator, typename T>
365 {
366 // If first iteration and history is empty
367 // then initialize U, v and coeff arrays
368 if (itr_ == 0 && nBasis_ == 0) {
369 int m, n;
370 for (m = 0; m < maxHist_; ++m) {
371 v_[m] = 0.0;
372 coeffs_[m] = 0.0;
373 for (n = 0; n < maxHist_; ++n) {
374 U_(m, n) = 0.0;
375 }
376 }
377 }
378
379 // Do nothing else on first iteration
380 if (itr_ == 0) return;
381
382 // Update fieldBasis_, resBasis_, U_ matrix and v_ vector
383 if (fieldHists_.size() > 1) {
384
385 // Update basis spanning differences of past field vectors
386 updateBasis(fieldBasis_, fieldHists_);
387
388 // Update basis spanning differences of past residual vectors
389 updateBasis(resBasis_, resHists_);
390
391 // Update nBasis_
392 nBasis_ = fieldBasis_.size();
393 UTIL_CHECK(fieldBasis_.size() == nBasis_);
394
395 // Update the U matrix and v vector.
396 updateU(U_, resBasis_, nBasis_);
397 }
398
399 UTIL_CHECK(nBasis_ > 0);
400
401 // Update v_ vector
402 updateV(v_, resHists_[0], resBasis_, nBasis_);
403
404 // Solve matrix equation problem to compute coefficients
405 // that minmize the L2 norm of the residual vector.
406 if (nBasis_ == 1) {
407 // Solve explicitly for coefficient
408 coeffs_[0] = v_[0] / U_(0,0);
409 } else
410 if (nBasis_ < maxHist_) {
411
412 // Create temporary smaller version of U_, v_, coeffs_ .
413 // This is done to avoid reallocating U_ with each iteration.
414 DMatrix<double> tempU;
415 DArray<double> tempv,tempcoeffs;
416 tempU.allocate(nBasis_,nBasis_);
417 tempv.allocate(nBasis_);
418 tempcoeffs.allocate(nBasis_);
419 for (int i = 0; i < nBasis_; ++i) {
420 tempv[i] = v_[i];
421 for (int j = 0; j < nBasis_; ++j) {
422 tempU(i,j) = U_(i,j);
423 }
424 }
425
426 // Solve matrix equation
427 LuSolver solver;
428 solver.allocate(nBasis_);
429 solver.computeLU(tempU);
430 solver.solve(tempv,tempcoeffs);
431
432 // Transfer solution to full-sized member variable
433 for (int i = 0; i < nBasis_; ++i) {
434 coeffs_[i] = tempcoeffs[i];
435 }
436
437 } else
438 if (nBasis_ == maxHist_) {
439 LuSolver solver;
440 solver.allocate(maxHist_);
441 solver.computeLU(U_);
442 solver.solve(v_, coeffs_);
443 }
444
445 return;
446 }
447
448 template <typename Iterator, typename T>
449 void AmIteratorTmpl<Iterator,T>::updateGuess()
450 {
451
452 // Set field and residual trial vectors to current values
453 setEqual(fieldTrial_, fieldHists_[0]);
454 setEqual(resTrial_, resHists_[0]);
455
456 // Add linear combinations of field and residual basis vectors
457 if (nBasis_ > 0) {
458
459 // Combine basis vectors into trial guess and predicted residual
460 addHistories(fieldTrial_, fieldBasis_, coeffs_, nBasis_);
461 addHistories(resTrial_, resBasis_, coeffs_, nBasis_);
462
463 }
464
465 #ifdef PSCF_AM_TEST
466 // Additional direct computation of error after projection step
467 if (hasAmTest_){
468 timerOmega_.stop();
469
470 // Additional MDE solution
471 timerMDE_.start();
472 update(fieldTrial_);
473 evaluate();
474 timerMDE_.stop();
475
476 // Residual computation
477 timerResid_.start();
478 getResidual(temp_);
479 timerResid_.stop();
480
481 // Compute error after projection
482 timerError_.start();
483 projectionError_ = computeError(temp_, fieldTrial_, errorType_, 0);
484 timerError_.stop();
485 timerOmega_.start();
486 }
487 #endif
488
489 // Correction (or "mixing") step
490 addPredictedError(fieldTrial_, resTrial_,lambda_);
491
492 // Update system using new trial field
493 update(fieldTrial_);
494
495
496 return;
497 }
498
499 // Private virtual member functions
500
501 /*
502 * Initialize just before entry to iterative loop.
503 * (virtual)
504 */
505 template <typename Iterator, typename T>
506 void AmIteratorTmpl<Iterator,T>::setup(bool isContinuation)
507 {
508 if (!isAllocatedAM()) {
509 allocateAM();
510 } else {
511 // Clear residual and field history buffers
512 resHists_.clear();
513 fieldHists_.clear();
514 if (!isContinuation) {
515 // Clear bases iff not a continuation
516 resBasis_.clear();
517 fieldBasis_.clear();
518 }
519 }
520 }
521
526 template <typename Iterator, typename T>
528 {
529 double lambda;
530 if (nBasis_ < maxHist_) {
531 lambda = 1.0 - pow(r, nBasis_ + 1);
532 } else {
533 lambda = 1.0;
534 }
535 return lambda;
536 }
537
538 /*
539 * Compute L2 norm of a vector.
540 */
541 template <typename Iterator, typename T>
543 {
544 double normSq = dotProduct(a, a);
545 return sqrt(normSq);
546 }
547
548 // Update entire U matrix
549 template <typename Iterator, typename T>
550 void
552 RingBuffer<T> const & resBasis,
553 int nHist)
554 {
555 // Update matrix U by shifting elements diagonally
556 int maxHist = U.capacity1();
557 for (int m = maxHist-1; m > 0; --m) {
558 for (int n = maxHist-1; n > 0; --n) {
559 U(m,n) = U(m-1,n-1);
560 }
561 }
562
563 // Compute U matrix's new row 0 and col 0
564 for (int m = 0; m < nHist; ++m) {
565 double dotprod = dotProduct(resBasis[0],resBasis[m]);
566 U(m,0) = dotprod;
567 U(0,m) = dotprod;
568 }
569 }
570
571 template <typename Iterator, typename T>
572 void
573 AmIteratorTmpl<Iterator, T>::updateV(DArray<double> & v,
574 T const & resCurrent,
575 RingBuffer<T> const & resBasis,
576 int nHist)
577 {
578 for (int m = 0; m < nHist; ++m) {
579 v[m] = dotProduct(resCurrent, resBasis[m]);
580 }
581 }
582
583 template <typename Iterator, typename T>
584 double AmIteratorTmpl<Iterator,T>::computeError(T&residTrial, T&fieldTrial,
585 std::string errorType,
586 int verbose)
587 {
588 double error = 0.0;
589
590 // Find max residual vector element
591 double maxRes = maxAbs(residTrial);
592
593 // Find norm of residual vector
594 double normRes = norm(residTrial);
595
596 // Check if calculation has diverged (normRes will be NaN)
597 UTIL_CHECK(!std::isnan(normRes));
598
599 // Find root-mean-squared residual element value
600 double rmsRes = normRes/sqrt(nElements());
601
602 // Find norm of residual vector relative to field
603 double normField = norm(fieldTrial);
604 double relNormRes = normRes/normField;
605
606 // Set error value
607 if (errorType == "maxResid") {
608 error = maxRes;
609 } else if (errorType == "normResid") {
610 error = normRes;
611 } else if (errorType == "rmsResid") {
612 error = rmsRes;
613 } else if (errorType == "relNormResid") {
614 error = relNormRes;
615 } else {
616 UTIL_THROW("Invalid iterator error type in parameter file.");
617 }
618
619 if (verbose > 1) {
620 Log::file() << "\n";
621 Log::file() << "Max Residual = " << Dbl(maxRes,15) << "\n";
622 Log::file() << "Residual Norm = " << Dbl(normRes,15) << "\n";
623 Log::file() << "RMS Residual = " << Dbl(rmsRes,15) << "\n";
624 Log::file() << "Relative Norm = " << Dbl(relNormRes,15)
625 << std::endl;
626 }
627
628 return error;
629 }
630
631 template <typename Iterator, typename T>
633 {
634 return computeError(resHists_[0], fieldHists_[0], errorType_, verbose);
635 }
636
637 template <typename Iterator, typename T>
639 {
640 // Output timing results, if requested.
641 double total = timerTotal_.time();
642 out << "\n";
643 out << " ";
644 out << "Total" << std::setw(22)<< "Per Iteration"
645 << std::setw(9) << "Fraction" << "\n";
646 out << "MDE solution: "
647 << Dbl(timerMDE_.time(), 9, 3) << " s, "
648 << Dbl(timerMDE_.time()/totalItr_, 9, 3) << " s, "
649 << Dbl(timerMDE_.time()/total, 9, 3) << "\n";
650 out << "residual computation: "
651 << Dbl(timerResid_.time(), 9, 3) << " s, "
652 << Dbl(timerResid_.time()/totalItr_, 9, 3) << " s, "
653 << Dbl(timerResid_.time()/total, 9, 3) << "\n";
654 out << "mixing coefficients: "
655 << Dbl(timerCoeff_.time(), 9, 3) << " s, "
656 << Dbl(timerCoeff_.time()/totalItr_, 9, 3) << " s, "
657 << Dbl(timerCoeff_.time()/total, 9, 3) << "\n";
658 out << "checking convergence: "
659 << Dbl(timerError_.time(), 9, 3) << " s, "
660 << Dbl(timerError_.time()/totalItr_, 9, 3) << " s, "
661 << Dbl(timerError_.time()/total, 9, 3) << "\n";
662 out << "updating guess: "
663 << Dbl(timerOmega_.time(), 9, 3) << " s, "
664 << Dbl(timerOmega_.time()/totalItr_, 9, 3) << " s, "
665 << Dbl(timerOmega_.time()/total, 9, 3)<< "\n";
666 out << "total time: "
667 << Dbl(total, 9, 3) << " s, "
668 << Dbl(total/totalItr_, 9, 3) << " s \n";
669 out << "\n";
670
671 #ifdef PSCF_AM_TEST
672 if (hasAmTest_){
673 out << "Average Projection Step Reduction Ratio: "
674 << Dbl(projectionRatio_/testCounter, 3, 3)<< "\n";
675 out << "Average Correction Step Reduction Ratio: "
676 << Dbl(correctionRatio_/testCounter, 3, 3)<< "\n";
677 }
678 #endif
679 }
680
681 template <typename Iterator, typename T>
683 {
684 timerMDE_.clear();
685 timerResid_.clear();
686 timerError_.clear();
687 timerCoeff_.clear();
688 timerOmega_.clear();
689 timerAM_.clear();
690 timerTotal_.clear();
691 totalItr_ = 0;
692 }
693
694}
695#endif
Template for Anderson mixing iterator algorithm.
virtual void setup(bool isContinuation)
Initialize just before entry to iterative loop.
void setMaxHist(int maxHist)
Set value of maxHist (number of retained previous states)
virtual double computeLambda(double r)
Compute mixing parameter for correction step of Anderson mixing.
virtual double computeError(T &residTrial, T &fieldTrial, std::string errorType, int verbose)
Compute and return error used to test for convergence.
void allocateAM()
Allocate memory required by AM algorithm, if necessary.
virtual void clear()
Clear information about history.
void setErrorType(std::string errorType)
Set and validate value of errorType string.
void clearTimers()
Clear timers.
~AmIteratorTmpl()
Destructor.
void readErrorType(std::istream &in)
Read and validate the optional errorType string parameter.
void setMaxItr(int maxItr)
Set value of maxItr.
AmIteratorTmpl()
Constructor.
virtual bool isValidErrorType()
Checks if a string is a valid error type.
void readParameters(std::istream &in)
Read all parameters and initialize.
int solve(bool isContinuation=false)
Iterate to a solution.
void outputTimers(std::ostream &out)
Log output timing results.
virtual double norm(T const &hist)
Find the L2 norm of a vector.
Exception thrown when not-a-number (NaN) is encountered.
Dynamically allocatable contiguous array template.
void allocate(int capacity)
Allocate the underlying C array.
Definition DArray.h:199
Dynamically allocated Matrix.
Definition DMatrix.h:25
void allocate(int capacity1, int capacity2)
Allocate memory for a matrix.
Definition DMatrix.h:170
Wrapper for a double precision number, for formatted ostream output.
Definition Dbl.h:40
Wrapper for an int, for formatted ostream output.
Definition Int.h:37
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
int capacity1() const
Get number of rows (range of the first array index).
Definition Matrix.h:136
Class for storing history of previous values in an array.
Definition RingBuffer.h:27
#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.