PSCF v1.2
rpc/fts/montecarlo/McSimulator.tpp
1#ifndef RPC_MC_SIMULATOR_TPP
2#define RPC_MC_SIMULATOR_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 "McSimulator.h"
12
13#include <rpc/System.h>
14#include <rpc/fts/montecarlo/McMoveFactory.h>
15#include <rpc/fts/analyzer/AnalyzerFactory.h>
16#include <rpc/fts/trajectory/TrajectoryReader.h>
17#include <rpc/fts/trajectory/TrajectoryReaderFactory.h>
18#include <rpc/fts/compressor/Compressor.h>
19#include <rpc/fts/perturbation/PerturbationFactory.h>
20#include <rpc/fts/perturbation/Perturbation.h>
21#include <rpc/fts/ramp/RampFactory.h>
22#include <rpc/fts/ramp/Ramp.h>
23
24#include <util/random/Random.h>
25#include <util/misc/Timer.h>
26#include <util/global.h>
27
28#include <gsl/gsl_eigen.h>
29
30namespace Pscf {
31namespace Rpc {
32
33 using namespace Util;
34
35 /*
36 * Constructor.
37 */
38 template <int D>
40 : Simulator<D>(system),
41 mcMoveManager_(*this, system),
42 analyzerManager_(*this, system),
43 trajectoryReaderFactoryPtr_(0)
44 {
45 setClassName("McSimulator");
46 trajectoryReaderFactoryPtr_
48 }
49
50 /*
51 * Destructor.
52 */
53 template <int D>
55 {
56 if (trajectoryReaderFactoryPtr_) {
57 delete trajectoryReaderFactoryPtr_;
58 }
59 }
60
61 /*
62 * Read parameter file block.
63 */
64 template <int D>
65 void McSimulator<D>::readParameters(std::istream &in)
66 {
67 // Read optional random seed value
68 readRandomSeed(in);
69
70 // Read McMoveManager block
71 readParamCompositeOptional(in, mcMoveManager_);
72
73 // Read optional Compressor, Perturbation and Ramp blocks
74 bool isEnd = false;
75 readCompressor(in, isEnd);
76 readPerturbation(in, isEnd);
77 if (hasMcMoves()) {
78 UTIL_CHECK(hasCompressor());
79 readRamp(in, isEnd);
80 }
81
82 // Read optional AnalyzerManager block
83 Analyzer<D>::baseInterval = 0; // default value
84 readParamCompositeOptional(in, analyzerManager_);
85
86 // Figure out what needs to be saved
87 state_.needsCc = false;
88 state_.needsDc = false;
89 state_.needsHamiltonian = true;
90 if (mcMoveManager_.needsCc()){
91 state_.needsCc = true;
92 }
93 if (mcMoveManager_.needsDc()){
94 state_.needsDc = true;
95 }
96
97 // Initialize Simulator<D> base class
99
100 }
101
102 /*
103 * Initialize just prior to a run.
104 */
105 template <int D>
106 void McSimulator<D>::setup(int nStep)
107 {
108 UTIL_CHECK(system().w().hasData());
109
110 // Eigenanalysis of the projected chi matrix.
111 analyzeChi();
112
113 if (hasPerturbation()) {
114 perturbation().setup();
115 }
116
117 if (hasRamp()) {
118 ramp().setup(nStep);
119 }
120
121 // Solve MDE and compute c-fields for the intial state
122 system().compute();
123
124 // Compress the initial state (adjust pressure-like field)
125 if (hasCompressor()) {
126 compressor().compress();
127 compressor().clearTimers();
128 }
129
130 // Compute field components and Hamiltonian
131 computeWc();
132 if (state_.needsCc || state_.needsDc) {
133 computeCc();
134 }
135 if (state_.needsDc) {
136 computeDc();
137 }
138 computeHamiltonian();
139
140 mcMoveManager_.setup();
141 if (analyzerManager_.size() > 0){
142 analyzerManager_.setup();
143 }
144
145 }
146
147 /*
148 * Perform a field theoretic MC simulation of nStep steps.
149 */
150 template <int D>
152 {
153 UTIL_CHECK(hasMcMoves());
154 UTIL_CHECK(hasCompressor());
155
156 // Initial setup
157 setup(nStep);
158
159 Log::file() << std::endl;
160
161 // Initialize timer, iStep_, and ramp (if any)
162 Timer timer;
163 Timer analyzerTimer;
164 timer.start();
165 iStep_ = 0;
166 if (hasRamp()) {
167 ramp().setParameters(iStep_);
168 }
169
170 // Analysis initial step (if any)
171 analyzerTimer.start();
172 analyzerManager_.sample(iStep_);
173 analyzerTimer.stop();
174
175 // Main Monte Carlo loop
176 for (iTotalStep_ = 0; iTotalStep_ < nStep; ++iTotalStep_) {
177
178 // Choose and attempt an McMove
179 bool converged;
180 converged = mcMoveManager_.chooseMove().move();
181
182 if (converged){
183 iStep_++;
184
185 if (hasRamp()) {
186 ramp().setParameters(iStep_);
187 }
188
189 // Analysis (if any)
190 analyzerTimer.start();
191 if (Analyzer<D>::baseInterval != 0) {
192 if (iStep_ % Analyzer<D>::baseInterval == 0) {
193 if (analyzerManager_.size() > 0) {
194 analyzerManager_.sample(iStep_);
195 }
196 }
197 }
198 analyzerTimer.stop();
199
200 } else{
201 Log::file() << "Step: "<< iTotalStep_
202 << " failed to converge" << "\n";
203 }
204 }
205
206 timer.stop();
207 double time = timer.time();
208 double analyzerTime = analyzerTimer.time();
209
210 // Output results of move statistics to files
211 mcMoveManager_.output();
213 analyzerManager_.output();
214 }
215
216 // Output results of ramp
217 if (hasRamp()){
218 ramp().output();
219 }
220
221 // Output times for the simulation run
222 Log::file() << std::endl;
223 Log::file() << "nStep " << nStep << std::endl;
224 if (iStep_ != nStep){
225 Log::file() << "nFail Step " << (nStep - iStep_) << std::endl;
226 }
227 Log::file() << "Total run time " << time
228 << " sec" << std::endl;
229 double rStep = double(nStep);
230 Log::file() << "time / nStep " << time / rStep
231 << " sec" << std::endl;
232 Log::file() << "Analyzer run time " << analyzerTime
233 << " sec" << std::endl;
234 Log::file() << std::endl;
235
236 // Output number of times MDE has been solved for the simulation run
237 outputMdeCounter(Log::file());
238
239 // Print McMove acceptance statistics
240 long attempt;
241 long accept;
242 long fail;
243 using namespace std;
244 Log::file() << "Move Statistics:" << endl << endl;
245 Log::file() << setw(20) << left << "Move Name"
246 << setw(10) << right << "Attempted"
247 << setw(10) << right << "Accepted"
248 << setw(13) << right << "AcceptRate"
249 << setw(10) << right << "Failed"
250 << setw(13) << right << "FailRate"
251 << endl;
252 int nMove = mcMoveManager_.size();
253 for (int iMove = 0; iMove < nMove; ++iMove) {
254 attempt = mcMoveManager_[iMove].nAttempt();
255 accept = mcMoveManager_[iMove].nAccept();
256 fail = mcMoveManager_[iMove].nFail();
257 Log::file() << setw(20) << left
258 << mcMoveManager_[iMove].className()
259 << setw(10) << right << attempt
260 << setw(10) << accept
261 << setw(13) << fixed << setprecision(5)
262 << ( attempt == 0 ? 0.0 : double(accept)/double(attempt) )
263 << setw(10) << fail
264 << setw(13) << fixed << setprecision(5)
265 << ( attempt == 0 ? 0.0 : double(fail)/double(attempt) )
266 << endl;
267 }
268 Log::file() << endl;
269
270 }
271
272 /*
273 * Open, read and analyze a trajectory file
274 */
275 template <int D>
276 void McSimulator<D>::analyze(int min, int max,
277 std::string classname,
278 std::string filename)
279 {
280 // Preconditions
281 UTIL_CHECK(min >= 0);
282 UTIL_CHECK(max >= min);
284 UTIL_CHECK(analyzerManager_.size() > 0);
285
286 // Construct TrajectoryReader
287 TrajectoryReader<D>* trajectoryReaderPtr;
288 trajectoryReaderPtr = trajectoryReaderFactory().factory(classname);
289 if (!trajectoryReaderPtr) {
290 std::string message;
291 message = "Invalid TrajectoryReader class name " + classname;
292 UTIL_THROW(message.c_str());
293 }
294
295 // Open trajectory file
296 trajectoryReaderPtr->open(filename);
297 trajectoryReaderPtr->readHeader();
298
299 // Main loop over trajectory frames
300 Timer timer;
301 bool hasFrame;
302 timer.start();
303 hasFrame = trajectoryReaderPtr->readFrame();
304
305 for (iStep_ = 0; iStep_ <= max && hasFrame; ++iStep_) {
306 if (hasFrame) {
307 clearData();
308
309 // Initialize analyzers
310 if (iStep_ == min) {
311 setup(iStep_);
312 }
313
314 // Sample property values only for iStep >= min
315 if (iStep_ >= min) {
316 analyzerManager_.sample(iStep_);
317 }
318 }
319
320 hasFrame = trajectoryReaderPtr->readFrame();
321 }
322 timer.stop();
323 Log::file() << "end main loop" << std::endl;
324 int nFrames = iStep_ - min;
325 trajectoryReaderPtr->close();
326 delete trajectoryReaderPtr;
327
328 // Output results of all analyzers to output files
329 analyzerManager_.output();
330
331 // Output number of frames and times
332 Log::file() << std::endl;
333 Log::file() << "# of frames " << nFrames << std::endl;
334 Log::file() << "run time " << timer.time()
335 << " sec" << std::endl;
336 Log::file() << "time / frame " << timer.time()/double(nFrames)
337 << " sec" << std::endl;
338 Log::file() << std::endl;
339
340 }
341
342 /*
343 * Output McMoveManager timer results.
344 */
345 template<int D>
346 void McSimulator<D>::outputTimers(std::ostream& out)
347 {
348 compressor().outputTimers(out);
349 out << "\n";
350 out << "MC move time contributions:\n";
351 mcMoveManager_.outputTimers(out);
352 out << "\n";
353 }
354
355 /*
356 * Clear all McMoveManager timers.
357 */
358 template<int D>
360 { mcMoveManager_.clearTimers(); }
361
362}
363}
364#endif
Abstract base for periodic output and/or analysis actions.
Monte-Carlo simulation coordinator.
virtual void analyze(int min, int max, std::string classname, std::string filename)
Read and analyze a trajectory file.
void setClassName(const char *className)
Set class name string.
virtual void readParameters(std::istream &in)
Read parameters file block for an MC simulation.
virtual void outputTimers(std::ostream &out)
Output timing results.
virtual void clearTimers()
Clear timers.
void simulate(int nStep)
Perform a field theoretic Monte-Carlo simulation.
McSimulator(System< D > &system)
Constructor.
Field theoretic simulator (base class).
Definition rpc/System.h:38
System< D > & system()
Get parent system by reference.
void allocate()
Allocate required memory.
Main class for SCFT or PS-FTS simulation of one system.
Definition rpc/System.h:100
Factory for subclasses of TrajectoryReader.
Trajectory file reader (base class).
virtual bool readFrame()=0
Read a single frame.
virtual void close()=0
Close the trajectory file.
virtual void open(std::string filename)=0
Open trajectory file and read header, if any.
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
Wall clock timer.
Definition Timer.h:35
void start(TimePoint begin)
Start timing from an externally supplied time.
Definition Timer.cpp:49
double time()
Return the accumulated time, in seconds.
Definition Timer.cpp:37
void stop(TimePoint end)
Stop the clock at an externally supplied time.
Definition Timer.cpp:73
File containing preprocessor macros for error handling.
#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.