PSCF v1.2
SweepTmpl.tpp
1#ifndef PSCF_SWEEP_TMPL_TPP
2#define PSCF_SWEEP_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 "SweepTmpl.h"
12#include <util/misc/Log.h>
13
14namespace Pscf {
15
16 using namespace Util;
17
18 /*
19 * Constructor
20 */
21 template <class State>
22 SweepTmpl<State>::SweepTmpl(int historyCapacity)
23 : ns_(0),
24 baseFileName_(),
25 historyCapacity_(historyCapacity),
26 historySize_(0),
27 nAccept_(0),
28 reuseState_(true)
29 { setClassName("SweepTmpl"); }
30
31 /*
32 * Destructor.
33 */
34 template <class State>
38 /*
39 * Read parameters.
40 */
41 template <class State>
42 void SweepTmpl<State>::readParameters(std::istream& in)
43 {
44 // Default values
45 baseFileName_ = "";
46
47 // Read parameters
48 read<int>(in, "ns", ns_);
49 readOptional<std::string>(in, "baseFileName", baseFileName_);
50 readOptional<int>(in, "historyCapacity", historyCapacity_);
51 readOptional<bool>(in, "reuseState", reuseState_);
52
53 // Allocate required arrays
54 UTIL_CHECK(historyCapacity_ > 0);
55 states_.allocate(historyCapacity_);
56 stateHistory_.allocate(historyCapacity_);
57 sHistory_.allocate(historyCapacity_);
58 c_.allocate(historyCapacity_);
59 }
60
61 template <class State>
63 {
64
65 // Compute and output ds
66 double ds = 1.0/double(ns_);
67 double ds0 = ds;
68 Log::file() << std::endl;
69 Log::file() << "ns = " << ns_ << std::endl;
70 Log::file() << "ds = " << ds << std::endl;
71
72 // Initial setup, before a sweep
73 setup();
74
75 // Solve for initial state of sweep
76 double sNew = 0.0;
77 Log::file() << std::endl;
78 Log::file() << "===========================================\n";
79 Log::file() << "Attempt s = " << sNew << std::endl;
80
81 int error;
82 bool isContinuation = false; // False on first step
83 error = solve(isContinuation);
84
85 if (error) {
86 UTIL_THROW("Failure to converge initial state of sweep");
87 } else {
88 accept(sNew);
89 }
90
91 // Loop over states on path
92 bool finished = false; // Are we finished with the loop?
93 while (!finished) {
94
95 // Loop over iteration attempts, with decreasing ds as needed
96 error = 1;
97 while (error) {
98
99 // Set a new contour variable value sNew
100 sNew = s(0) + ds;
101 Log::file() << std::endl;
102 Log::file() << "===========================================\n";
103 Log::file() << "Attempt s = " << sNew << std::endl;
104
105 // Set non-adjustable system parameters to new values
106 setParameters(sNew);
107
108 // Guess new state variables by polynomial extrapolation.
109 // This function must both compute the extrapolation and
110 // set initial guess values in the parent system.
111 extrapolate(sNew);
112
113 // Attempt iterative SCFT solution
114 isContinuation = reuseState_;
115 error = solve(isContinuation);
116
117 // Process success or failure
118 if (error) {
119 Log::file() << "Backtrack and halve sweep step size:"
120 << std::endl;
121
122 // Upon failure, reset state to last converged solution
123 reset();
124
125 // Decrease ds by half
126 ds *= 0.50;
127 if (ds < 0.1*ds0) {
128 UTIL_THROW("Sweep decreased ds too many times.");
129 }
130
131 } else {
132
133 // Upon successful convergence, update history and nAccept
134 accept(sNew);
135
136 }
137 }
138 if (sNew + ds > 1.0000001) {
139 finished = true;
140 }
141 }
142 Log::file() << "===========================================\n";
143
144 // Clean up after end of the entire sweep
145 cleanup();
146
147 }
148
149 template <class State>
150 void SweepTmpl<State>::addParameterType(std::string name, int nId,
151 ParameterModifier& modifier)
152 {
153 // Check if parameterTypes_ already has an element with this name
154 if (parameterTypes_.size() > 0) {
155 for (int i = 0; i < parameterTypes_.size(); ++i) {
156 UTIL_CHECK(parameterTypes_[i].name != name);
157 }
158 }
159
160 ParameterType paramType;
161 paramType.name = name;
162 paramType.nId = nId;
163 paramType.modifierPtr = &modifier;
164 parameterTypes_.append(paramType);
165 }
166
167 template <class State>
169 {
170 // Check if parameterTypes_ already has an element with this name
171 if (parameterTypes_.size() > 0) {
172 for (int i = 0; i < parameterTypes_.size(); ++i) {
173 UTIL_CHECK(parameterTypes_[i].name != paramType.name);
174 }
175 }
176
177 parameterTypes_.append(paramType);
178 }
179
180 template <class State>
182 {
183 for (int i = 0; i < paramTypes.size(); i++) {
184 // Check if parameterTypes_ already has an element with this name
185 if (parameterTypes_.size() > 0) {
186 for (int j = 0; j < parameterTypes_.size(); j++) {
187 UTIL_CHECK(parameterTypes_[j].name != paramTypes[i].name);
188 }
189 }
190
191 parameterTypes_.append(paramTypes[i]);
192 }
193 }
194
195 /*
196 * Initialize history variables (must be called by setup function).
197 */
198 template <class State>
200 {
201 UTIL_CHECK(historyCapacity_ > 0);
202 UTIL_CHECK(states_.capacity() == historyCapacity_);
203 UTIL_CHECK(sHistory_.capacity() == historyCapacity_);
204 UTIL_CHECK(stateHistory_.capacity() == historyCapacity_);
205
206 // Check allocation of all states, allocate if necessary
207 for (int i = 0; i < historyCapacity_; ++i) {
208 checkAllocation(states_[i]);
209 }
210
211 // Set pointers in stateHistory_ to refer to objects in array states_
212 nAccept_ = 0;
213 historySize_ = 0;
214 for (int i = 0; i < historyCapacity_; ++i) {
215 sHistory_[i] = 0.0;
216 stateHistory_[i] = &states_[i];
217 }
218 }
219
220 template <class State>
221 void SweepTmpl<State>::accept(double sNew)
222 {
223
224 // Shift elements of sHistory_
225 for (int i = historyCapacity_ - 1; i > 0; --i) {
226 sHistory_[i] = sHistory_[i-1];
227 }
228 sHistory_[0] = sNew;
229
230 // Shift elements of stateHistory_ (pointers to stored solutions)
231 State* temp;
232 temp = stateHistory_[historyCapacity_-1];
233 for (int i = historyCapacity_ - 1; i > 0; --i) {
234 stateHistory_[i] = stateHistory_[i-1];
235 }
236 stateHistory_[0] = temp;
237
238 // Update counters nAccept_ and historySize_
239 ++nAccept_;
240 if (historySize_ < historyCapacity_) {
241 ++historySize_;
242 }
243
244 // Call getSolution to copy system state to state(0).
245 getSolution();
246 }
247
248 /*
249 * Use Lagrange polynomials to compute coefficients for continuation.
250 */
251 template <class State>
253 {
254 UTIL_CHECK(historySize_ <= historyCapacity_);
255 if (historySize_ == 1) {
256 c_[0] = 1.0;
257 } else {
258 double num, den;
259 int i, j;
260 for (i = 0; i < historySize_; ++i) {
261 num = 1.0;
262 den = 1.0;
263 for (j = 0; j < historySize_; ++j) {
264 if (j != i) {
265 num *= (sNew - s(j));
266 den *= (s(i) - s(j));
267 }
268 }
269 c_[i] = num/den;
270 }
271 }
272 // Theory: The coefficient c_[i] is a Lagrange polynomial
273 // function c_[i](sNew) of sNew that is defined such that
274 // c_[i](s(i)) = 1 and c_[i](s(j)) = 0 for any j != i.
275 // Given a set of values y(i) previously obtained at contour
276 // variable values s(i) for all 0 <= i < historySize_, a
277 // linear combination f(sNew) = \sum_i c_[i](sNew)*y(i) summed
278 // over i = 0, ..., historySize_ - 1 gives a polynomial in
279 // sNew that passes through all previous values, such that
280 // f(sNew) = y(i) for sNew = s(i).
281 }
282
283 /*
284 * Clean up after the end of a sweep (empty default implementation).
285 */
286 template <class State>
289
290} // namespace Pscf
291#endif
Base class allowing subclasses to define sweepable parameters.
Solve a sequence of problems along a path through parameter space.
Definition SweepTmpl.h:28
void addParameterTypes(GArray< ParameterType > paramTypes)
Declare an array of specialized sweep parameter types.
~SweepTmpl()
Destructor.
Definition SweepTmpl.tpp:35
virtual void cleanup()
Clean up operation at the end of a sweep.
void initialize()
Initialize variables that track history of solutions.
void setCoefficients(double sNew)
Compute coefficients of previous states for continuation.
void addParameterType(std::string name, int nId, ParameterModifier &modifier)
Declare a specialized parameter type.
virtual void sweep()
Iterate to solution.
Definition SweepTmpl.tpp:62
virtual void readParameters(std::istream &in)
Read ns and baseFileName parameters.
Definition SweepTmpl.tpp:42
An automatically growable array, analogous to a std::vector.
Definition GArray.h:34
int size() const
Return logical size of this array (i.e., current number of elements).
Definition GArray.h:455
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
void setClassName(const char *className)
Set class name string.
#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.
Declaration of a specialized sweep parameter type.