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