PSCF v1.2
rpg/scft/sweep/Sweep.tpp
1#ifndef RPG_SWEEP_TPP
2/*
3* PSCF - Polymer Self-Consistent Field Theory
4*
5* Copyright 2016 - 2022, The Regents of the University of Minnesota
6* Distributed under the terms of the GNU General Public License.
7*/
8
9#include "Sweep.h"
10#include <pscf/sweep/SweepTmpl.tpp>
11#include <rpg/System.h>
12#include <rpg/scft/iterator/Iterator.h>
13#include <pscf/inter/Interaction.h>
14
15#include <util/misc/FileMaster.h>
16#include <util/misc/ioUtil.h>
17
18namespace Pscf {
19namespace Rpg {
20
21 using namespace Util;
22
23 // Maximum number of previous states = order of continuation + 1
24 #define RPG_HISTORY_CAPACITY 3
25
26 /*
27 * Default constructor.
28 */
29 template <int D>
31 : SweepTmpl< BasisFieldState<D> >(RPG_HISTORY_CAPACITY),
32 writeCRGrid_(false),
33 writeCBasis_(false),
34 writeWRGrid_(false),
35 systemPtr_(0)
36 {}
37
38 /*
39 * Constructor, creates association with parent system.
40 */
41 template <int D>
43 : SweepTmpl< BasisFieldState<D> >(RPG_HISTORY_CAPACITY),
44 writeCRGrid_(false),
45 writeCBasis_(false),
46 writeWRGrid_(false),
47 systemPtr_(&sys)
48 {
49 // Get specialized sweep parameters from Iterator
50 if (system().hasIterator()) {
51 addParameterTypes(system().iterator().getParameterTypes());
52 }
53 }
54
55 /*
56 * Destructor.
57 */
58 template <int D>
61
62 /*
63 * Set association with a parent system.
64 */
65 template <int D>
67 { systemPtr_ = &system; }
68
69 /*
70 * Read parameters
71 */
72 template <int D>
73 void Sweep<D>::readParameters(std::istream& in)
74 {
75 // Call the base class's readParameters function.
76 SweepTmpl< BasisFieldState<D> >::readParameters(in);
77
78 // Read optional flags indicating which field types to output
79 readOptional(in, "writeCRGrid", writeCRGrid_);
80 readOptional(in, "writeCBasis", writeCBasis_);
81 readOptional(in, "writeWRGrid", writeWRGrid_);
82 }
83
84 /*
85 * Check allocation of one state object, allocate if necessary.
86 */
87 template <int D>
89 {
90 UTIL_CHECK(hasSystem());
91 state.setSystem(system());
92 state.allocate();
93 state.unitCell() = system().unitCell();
94 }
95
96 /*
97 * Setup operations at the beginning of a sweep.
98 */
99 template <int D>
101 {
102 initialize();
103 checkAllocation(trial_);
104
105 // Open log summary file
106 std::string fileName = baseFileName_;
107 fileName += "sweep.log";
108 system().fileMaster().openOutputFile(fileName, logFile_);
109 };
110
111 /*
112 * Set non-adjustable system parameters to new values.
113 *
114 * \param s path length coordinate, in range [0,1]
115 */
116 template <int D>
118 {
119 // Empty default implementation allows Sweep<D> to be compiled.
120 UTIL_THROW("Calling unimplemented function Sweep::setParameters");
121 };
122
123 /*
124 * Create guess for adjustable variables by polynomial extrapolation.
125 */
126 template <int D>
127 void Sweep<D>::extrapolate(double sNew)
128 {
129 UTIL_CHECK(historySize() > 0);
130
131 // If historySize() == 1, do nothing: Use previous system state
132 // as trial for the new state.
133
134 if (historySize() > 1) {
135 UTIL_CHECK(historySize() <= historyCapacity());
136
137 // Does the iterator allow a flexible unit cell ?
138 bool isFlexible = system().iterator().isFlexible();
139
140 // Compute coefficients of polynomial extrapolation to sNew
141 setCoefficients(sNew);
142
143 // Set extrapolated trial w fields
144 double coeff;
145 int nMonomer = system().mixture().nMonomer();
146 int nBasis = system().basis().nBasis();
147 DArray<double>* newFieldPtr;
148 DArray<double>* oldFieldPtr;
149 int i, j, k;
150 for (i=0; i < nMonomer; ++i) {
151 newFieldPtr = &(trial_.field(i));
152
153 // Previous state k = 0 (most recent)
154 oldFieldPtr = &state(0).field(i);
155 coeff = c(0);
156 for (j=0; j < nBasis; ++j) {
157 (*newFieldPtr)[j] = coeff*(*oldFieldPtr)[j];
158 }
159
160 // Previous states k >= 1 (older)
161 for (k = 1; k < historySize(); ++k) {
162 oldFieldPtr = &state(k).field(i);
163 coeff = c(k);
164 for (j=0; j < nBasis; ++j) {
165 (*newFieldPtr)[j] += coeff*(*oldFieldPtr)[j];
166 }
167 }
168 }
169
170 // Make sure unitCellParameters_ is up to date with system
171 // (if we are sweeping in a lattice parameter, then the system
172 // parameters will be up-to-date but unitCellParameters_ wont be)
173 FSArray<double, 6> oldParameters = unitCellParameters_;
174 unitCellParameters_ = system().unitCell().parameters();
175
176 // If isFlexible, then extrapolate the flexible unit cell parameters
177 if (isFlexible) {
178
179 double coeff;
180 double parameter;
181
182 #if 0
183 const FSArray<int,6> indices = system().iterator().flexibleParams();
184 const int nParameter = indices.size();
185
186 // Add contributions from all previous states
187 for (k = 0; k < historySize(); ++k) {
188 coeff = c(k);
189 for (int i = 0; i < nParameter; ++i) {
190 if (k == 0) {
191 unitCellParameters_[indices[i]] = 0;
192 }
193 parameter = state(k).unitCell().parameter(indices[i]);
194 unitCellParameters_[indices[i]] += coeff*parameter;
195 }
196 }
197 #endif
198
199 // Add contributions from all previous states
200 const int nParameter = unitCellParameters_.size();
201 for (k = 0; k < historySize(); ++k) {
202 coeff = c(k);
203 for (int i = 0; i < nParameter; ++i) {
204 if (k == 0) {
205 unitCellParameters_[i] = 0.0;
206 }
207 parameter = state(k).unitCell().parameter(i);
208 unitCellParameters_[i] += coeff*parameter;
209 }
210 }
211
212 }
213
214 // Reset trial_.unitCell() object
215 trial_.unitCell().setParameters(unitCellParameters_);
216
217 // Check if any unit cell parameters have changed
218 bool newCellParams(true);
219 for (int i = 0; i < oldParameters.size(); i++) {
220 if (fabs(oldParameters[i] - unitCellParameters_[i]) < 1e-10) {
221 newCellParams = false;
222 break;
223 }
224 }
225
226 // Transfer data from trial_ state to parent system
227 trial_.setSystemState(newCellParams);
228 }
229
230 };
231
232 /*
233 * Call current iterator to solve SCFT problem.
234 *
235 * Return 0 for sucessful solution, 1 on failure to converge.
236 */
237 template <int D>
238 int Sweep<D>::solve(bool isContinuation)
239 { return system().iterate(isContinuation); };
240
241 /*
242 * Reset system to previous solution after iterature failure.
243 *
244 * The implementation of this function should reset the system state
245 * to correspond to that stored in state(0).
246 */
247 template <int D>
249 {
250 bool isFlexible = system().iterator().isFlexible();
251 state(0).setSystemState(isFlexible);
252 }
253
254 /*
255 * Update state(0) and output data after successful convergence
256 *
257 * The implementation of this function should copy the current
258 * system state into state(0) and output any desired information
259 * about the current converged solution.
260 */
261 template <int D>
263 {
264 state(0).setSystem(system());
265 state(0).getSystemState();
266
267 // Output converged solution to several files
268 outputSolution();
269
270 // Output summary to log file
271 outputSummary(logFile_);
272
273 };
274
275 template <int D>
277 {
278 std::ofstream out;
279 std::string outFileName;
280 std::string indexString = toString(nAccept() - 1);
281
282 // Open parameter file, with thermodynamic properties at end
283 outFileName = baseFileName_;
284 outFileName += indexString;
285 outFileName += ".stt";
286 system().fileMaster().openOutputFile(outFileName, out);
287
288 // Write data file, with thermodynamic properties at end
289 system().writeParamNoSweep(out);
290 out << std::endl;
291 system().writeThermo(out);
292 out.close();
293
294 // Write w fields
295 outFileName = baseFileName_;
296 outFileName += indexString;
297 outFileName += "_w";
298 outFileName += ".bf";
299 system().writeWBasis(outFileName);
300
301 // Optionally write c rgrid files
302 if (writeCRGrid_) {
303 outFileName = baseFileName_;
304 outFileName += indexString;
305 outFileName += "_c";
306 outFileName += ".rf";
307 system().writeCRGrid(outFileName);
308 }
309
310 // Optionally write c basis files
311 if (writeCBasis_) {
312 outFileName = baseFileName_;
313 outFileName += indexString;
314 outFileName += "_c";
315 outFileName += ".bf";
316 system().writeCBasis(outFileName);
317 }
318
319 // Optionally write w rgrid files
320 if (writeWRGrid_) {
321 outFileName = baseFileName_;
322 outFileName += indexString;
323 outFileName += "_w";
324 outFileName += ".rf";
325 system().writeWRGrid(outFileName);
326 }
327 }
328
329 template <int D>
330 void Sweep<D>::outputSummary(std::ostream& out)
331 {
332 int i = nAccept() - 1;
333 double sNew = s(0);
334 out << Int(i,5) << Dbl(sNew)
335 << Dbl(system().fHelmholtz(),16)
336 << Dbl(system().pressure(),16);
337 out << std::endl;
338 }
339
340 template <int D>
342 { logFile_.close(); }
343
344} // namespace Rpg
345} // namespace Pscf
346#endif
FieldState for fields in symmetry-adapted basis format.
const UnitCell< D > & unitCell() const
Get UnitCell (i.e., lattice type and parameters) by const reference.
void setSystem(System< D > &system)
Set association with System, after default construction.
Solve a sequence of problems along a line in parameter space.
Definition rpg/System.h:39
virtual void getSolution()
Update state(0) and output data after successful convergence.
virtual int solve(bool isContinuation)
Call current iterator to solve SCFT problem.
virtual void readParameters(std::istream &in)
Read parameters from param file.
virtual void checkAllocation(BasisFieldState< D > &state)
Check allocation state of fields in one state, allocate if necessary.
virtual void extrapolate(double sNew)
Create a guess for adjustable variables by continuation.
virtual void setup()
Setup operation at the beginning of a sweep.
virtual void reset()
Reset system to previous solution after iterature failure.
virtual void setParameters(double sNew)=0
Set non-adjustable system parameters to new values.
Sweep()
Default Constructor.
System< D > & system()
Return the parent system by reference.
void setSystem(System< D > &system)
Set association with parent System.
virtual void cleanup()
Cleanup operation at the beginning of a sweep.
Main class for calculations that represent one system.
Definition rpg/System.h:107
Solve a sequence of problems along a path through parameter space.
Definition SweepTmpl.h:28
void addParameterTypes(GArray< ParameterType > paramTypes)
Dynamically allocatable contiguous array template.
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
int size() const
Return logical size of this array (i.e., number of elements).
Definition FSArray.h:207
Wrapper for an int, for formatted ostream output.
Definition Int.h:37
#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
std::string toString(int n)
Return string representation of an integer.
Definition ioUtil.cpp:52
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.