PSCF v1.2
rpc/fts/brownian/BdSimulator.tpp
1#ifndef RPC_BD_SIMULATOR_TPP
2#define RPC_BD_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 "BdSimulator.h"
12
13#include <rpc/fts/brownian/BdStep.h>
14#include <rpc/fts/brownian/BdStepFactory.h>
15#include <rpc/fts/compressor/Compressor.h>
16#include <rpc/fts/analyzer/AnalyzerFactory.h>
17#include <rpc/fts/trajectory/TrajectoryReader.h>
18#include <rpc/fts/trajectory/TrajectoryReaderFactory.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#include <rpc/System.h>
24
25#include <util/random/Random.h>
26#include <util/misc/Timer.h>
27#include <util/global.h>
28
29namespace Pscf {
30namespace Rpc {
31
32 using namespace Util;
33
34 /*
35 * Constructor.
36 */
37 template <int D>
39 : Simulator<D>(system),
40 analyzerManager_(*this, system),
41 bdStepPtr_(0),
42 bdStepFactoryPtr_(0),
43 trajectoryReaderFactoryPtr_(0)
44 {
45 setClassName("BdSimulator");
46 bdStepFactoryPtr_ = new BdStepFactory<D>(*this);
47 trajectoryReaderFactoryPtr_
49 }
50
51 /*
52 * Destructor.
53 */
54 template <int D>
56 {
57 if (bdStepFactoryPtr_) {
58 delete bdStepFactoryPtr_;
59 }
60 if (bdStepPtr_) {
61 delete bdStepPtr_;
62 }
63 if (trajectoryReaderFactoryPtr_) {
64 delete trajectoryReaderFactoryPtr_;
65 }
66 }
67
68 /*
69 * Read parameter file block for a BD simulator.
70 */
71 template <int D>
72 void BdSimulator<D>::readParameters(std::istream &in)
73 {
74 // Optionally read a random seed value
75 readRandomSeed(in);
76
77 // Optionally read a BdStep
78 bool isEnd = false;
79 std::string className;
80 bdStepPtr_ =
81 bdStepFactoryPtr_->readObjectOptional(in, *this,
82 className,
83 isEnd);
84 if (!hasBdStep() && ParamComponent::echo()) {
85 Log::file() << indent() << " BdStep{ [absent] }\n";
86 }
87
88 // Optionally read Compressor, Perturbation and/or Ramp blocks
89 readCompressor(in, isEnd);
90 if (hasBdStep()) {
91 UTIL_CHECK(hasCompressor());
92 }
93
94 // Optionally Perturbation and/or Ramp blocks
95 readPerturbation(in, isEnd);
96 if (hasBdStep()) {
97 readRamp(in, isEnd);
98 }
99
100 // Optionally read an AnalyzerManager
101 Analyzer<D>::baseInterval = 0; // default value
102 readParamCompositeOptional(in, analyzerManager_);
103
104 // Figure out what variables need to be saved
105 state_.needsCc = false;
106 state_.needsDc = false;
107 state_.needsHamiltonian = false;
108 if (hasBdStep()) {
109 if (bdStep().needsCc()){
110 state_.needsCc = true;
111 }
112 if (bdStep().needsDc()){
113 state_.needsDc = true;
114 }
115 }
116
117 // Allocate memory for Simulator<D> base class
119
120 }
121
122 /*
123 * Setup before main loop of a BD simulation.
124 */
125 template <int D>
126 void BdSimulator<D>::setup(int nStep)
127 {
128 UTIL_CHECK(system().w().hasData());
129
130 // Eigenanalysis of the projected chi matrix.
131 analyzeChi();
132
133 if (hasPerturbation()) {
134 perturbation().setup();
135 }
136
137 if (hasRamp()) {
138 ramp().setup(nStep);
139 }
140
141 // Solve MDE and compute c-fields for the intial state
142 system().compute();
143
144 // Compress the initial state (adjust pressure-like field)
145 if (hasCompressor()) {
146 compressor().compress();
147 compressor().clearTimers();
148 }
149
150 // Compute field components and Hamiltonian for initial state.
151 computeWc();
152 computeCc();
153 computeDc();
154 computeHamiltonian();
155
156 bdStep().setup();
157 if (analyzerManager_.size() > 0){
158 analyzerManager_.setup();
159 }
160
161 }
162
163 /*
164 * Perform a field theoretic MC simulation of nStep steps.
165 */
166 template <int D>
168 {
169 UTIL_CHECK(hasBdStep());
170 UTIL_CHECK(hasCompressor());
171 UTIL_CHECK(system().w().hasData());
172
173 // Initial setup
174 setup(nStep);
175
176 // Main simulation loop
177 Timer timer;
178 Timer analyzerTimer;
179 timer.start();
180 iStep_ = 0;
181 if (hasRamp()) {
182 ramp().setParameters(iStep_);
183 }
184
185 // Analysis for initial state (if any)
186 analyzerTimer.start();
187 if (analyzerManager_.size() > 0){
188 analyzerManager_.sample(iStep_);
189 }
190 analyzerTimer.stop();
191
192 for (iTotalStep_ = 0; iTotalStep_ < nStep; ++iTotalStep_) {
193
194 // Take a step (modifies W fields)
195 bool converged;
196 converged = bdStep().step();
197
198 if (converged){
199 iStep_++;
200
201 if (hasRamp()) {
202 ramp().setParameters(iStep_);
203 }
204
205 // Analysis (if any)
206 analyzerTimer.start();
207 if (Analyzer<D>::baseInterval != 0) {
208 if (analyzerManager_.size() > 0) {
209 if (iStep_ % Analyzer<D>::baseInterval == 0) {
210 analyzerManager_.sample(iStep_);
211 }
212 }
213 }
214 analyzerTimer.stop();
215
216 } else {
217 Log::file() << "Step: "<< iTotalStep_
218 << " failed to converge" << "\n";
219 }
220
221 }
222
223 timer.stop();
224 double time = timer.time();
225 double analyzerTime = analyzerTimer.time();
226
227 // Output results analyzers to files
229 analyzerManager_.output();
230 }
231
232 // Output results of ramp
233 if (hasRamp()){
234 Log::file() << std::endl;
235 ramp().output();
236 }
237
238 // Output times for the simulation run
239 Log::file() << std::endl;
240 Log::file() << "nStep " << nStep << std::endl;
241 if (iStep_ != nStep){
242 Log::file() << "nFail Step " << (nStep - iStep_) << std::endl;
243 }
244 Log::file() << "Total run time " << time
245 << " sec" << std::endl;
246 double rStep = double(nStep);
247 Log::file() << "time / nStep " << time / rStep
248 << " sec" << std::endl;
249 Log::file() << "Analyzer run time " << analyzerTime
250 << " sec" << std::endl;
251 Log::file() << std::endl;
252
253 // Output number of times MDE has been solved for the simulation run
254 Log::file() << "MDE counter "
255 << compressor().mdeCounter() << std::endl;
256 Log::file() << std::endl;
257
258 // Output compressor timer results
259 // compressor().outputTimers(Log::file());
260 }
261
262 /*
263 * Open, read and analyze a trajectory file
264 */
265 template <int D>
266 void BdSimulator<D>::analyze(int min, int max,
267 std::string classname,
268 std::string filename)
269 {
270 // Preconditions
271 UTIL_CHECK(min >= 0);
272 UTIL_CHECK(max >= min);
274 UTIL_CHECK(analyzerManager_.size() > 0);
275
276 // Construct TrajectoryReader
277 TrajectoryReader<D>* trajectoryReaderPtr;
278 trajectoryReaderPtr = trajectoryReaderFactory().factory(classname);
279 if (!trajectoryReaderPtr) {
280 std::string message;
281 message = "Invalid TrajectoryReader class name " + classname;
282 UTIL_THROW(message.c_str());
283 }
284
285 // Open trajectory file
286 trajectoryReaderPtr->open(filename);
287 trajectoryReaderPtr->readHeader();
288
289 // Main loop over trajectory frames
290 Timer timer;
291 bool hasFrame;
292 timer.start();
293 hasFrame = trajectoryReaderPtr->readFrame();
294
295 for (iStep_ = 0; iStep_ <= max && hasFrame; ++iStep_) {
296 if (hasFrame) {
297 clearData();
298
299 // Initialize analyzers
300 if (iStep_ == min) {
301 //analyzerManager_.setup();
302 setup(iStep_);
303 }
304
305 // Sample property values only for iStep >= min
306 if (iStep_ >= min) {
307 analyzerManager_.sample(iStep_);
308 }
309 }
310
311 hasFrame = trajectoryReaderPtr->readFrame();
312 }
313 timer.stop();
314 Log::file() << "end main loop" << std::endl;
315 int nFrames = iStep_ - min;
316 trajectoryReaderPtr->close();
317 delete trajectoryReaderPtr;
318
319 // Output results of all analyzers to output files
320 analyzerManager_.output();
321
322 // Output number of frames and times
323 Log::file() << std::endl;
324 Log::file() << "# of frames " << nFrames << std::endl;
325 Log::file() << "run time " << timer.time()
326 << " sec" << std::endl;
327 Log::file() << "time / frame " << timer.time()/double(nFrames)
328 << " sec" << std::endl;
329 Log::file() << std::endl;
330
331 }
332
333}
334}
335#endif
Abstract base for periodic output and/or analysis actions.
Brownian dynamics simulator for PS-FTS.
BdSimulator(System< D > &system)
Constructor.
void setClassName(const char *className)
Set class name string.
virtual void analyze(int min, int max, std::string classname, std::string filename)
Read and analyze a trajectory file.
virtual void readParameters(std::istream &in)
Read parameter file block for a Brownian dynamics (BD) simulation.
void simulate(int nStep)
Perform a field theoretic Monte-Carlo simulation.
Factory for subclasses of BdStep.
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
static bool echo()
Get echo parameter.
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.