PSCF v1.2
rpg/fts/montecarlo/ForceBiasMove.tpp
1#ifndef RPG_FORCE_BIAS_MOVE_TPP
2#define RPG_FORCE_BIAS_MOVE_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 "ForceBiasMove.h"
12#include "McMove.h"
13#include <rpg/fts/montecarlo/McSimulator.h>
14#include <rpg/fts/compressor/Compressor.h>
15#include <rpg/fts/VecOpFts.h>
16#include <rpg/System.h>
17#include <prdc/cuda/resources.h>
18#include <pscf/cuda/CudaRandom.h>
19#include <util/param/ParamComposite.h>
20#include <util/random/Random.h>
21
22namespace Pscf {
23namespace Rpg {
24
25 using namespace Util;
26
27 /*
28 * Constructor.
29 */
30 template <int D>
32 : McMove<D>(simulator),
33 w_(),
34 dwc_(),
35 mobility_(0.0)
36 { setClassName("ForceBiasMove"); }
37
38 /*
39 * Destructor, empty default implementation.
40 */
41 template <int D>
44
45 /*
46 * ReadParameters, empty default implementation.
47 */
48 template <int D>
49 void ForceBiasMove<D>::readParameters(std::istream &in)
50 {
51 //Read the probability
52 readProbability(in);
53
54 // Read Brownian dynamics mobility parameter
55 read(in, "mobility", mobility_);
56
57 // Allocate memory for private containers
58 int nMonomer = system().mixture().nMonomer();
59 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
60 w_.allocate(nMonomer);
61 for (int i=0; i < nMonomer; ++i) {
62 w_[i].allocate(meshDimensions);
63 }
64 dc_.allocate(nMonomer-1);
65 dwc_.allocate(nMonomer-1);
66 for (int i=0; i < nMonomer - 1; ++i) {
67 dc_[i].allocate(meshDimensions);
68 dwc_[i].allocate(meshDimensions);
69 }
70
71 }
72
73 template <int D>
75 {
76 // Check array capacities
77 int meshSize = system().domain().mesh().size();
78 IntVec<D> dimensions = system().domain().mesh().dimensions();
79 int nMonomer = system().mixture().nMonomer();
80 UTIL_CHECK(w_.capacity() == nMonomer);
81 for (int i=0; i < nMonomer; ++i) {
82 UTIL_CHECK(w_[i].capacity() == meshSize);
83 }
84 UTIL_CHECK(dc_.capacity() == nMonomer-1);
85 UTIL_CHECK(dwc_.capacity() == nMonomer-1);
86 for (int i=0; i < nMonomer - 1; ++i) {
87 UTIL_CHECK(dc_[i].capacity() == meshSize);
88 UTIL_CHECK(dwc_[i].capacity() == meshSize);
89 }
90
92 biasField_.allocate(dimensions);
93 gaussianField_.allocate(dimensions);
94 }
95
96 /*
97 * Attempt unconstrained move
98 */
99 template <int D>
101 {
102 totalTimer_.start();
103 incrementNAttempt();
104
105 // Preconditions
106 UTIL_CHECK(simulator().hasWc());
107 UTIL_CHECK(simulator().hasDc());
108 UTIL_CHECK(simulator().hasHamiltonian());
109
110 // Array sizes and indices
111 const int nMonomer = system().mixture().nMonomer();
112 const int meshSize = system().domain().mesh().size();
113 int i, j;
114
115
116 // Get current Hamiltonian
117 double oldHamiltonian = simulator().hamiltonian();
118
119 // Save current state
120 simulator().saveState();
121
122 // Clear both eigen-components of the fields and hamiltonian
123 simulator().clearData();
124
125 attemptMoveTimer_.start();
126
127 // Copy current W fields from parent system into wc_
128 for (i = 0; i < nMonomer; ++i) {
129 VecOp::eqV(w_[i], system().w().rgrid(i));
130 }
131
132 // Copy current derivative fields from into member variable dc_
133 for (i = 0; i < nMonomer - 1; ++i) {
134 VecOp::eqV(dc_[i], simulator().dc(i));
135 }
136
137 // Constants for dynamics
138 const double vSystem = system().domain().unitCell().volume();
139 const double vNode = vSystem/double(meshSize);
140 double a = -1.0*mobility_;
141 double b = sqrt(2.0*mobility_/vNode);
142
143 // Constants for normal distribution
144 double stddev = 1.0;
145 double mean = 0;
146
147 // Modify local variables dwc_ and wc_
148 // Loop over eigenvectors of projected chi matrix
149 double evec;
150 for (j = 0; j < nMonomer - 1; ++j) {
151 RField<D> & dwc = dwc_[j];
152
153 // Generate normal distributed random floating point numbers
154 cudaRandom().normal(gaussianField_, stddev, mean);
155
156 // dwc
157 VecOp::addVcVc(dwc, dc_[j], a, gaussianField_, b);
158
159 // Loop over monomer types
160 for (i = 0; i < nMonomer; ++i) {
161 RField<D> const & dwc = dwc_[j];
162 RField<D> & wn = w_[i];
163 evec = simulator().chiEvecs(j,i);
164 VecOp::addEqVc(wn, dwc, evec);
165 }
166
167 }
168
169 // Set modified fields in parent system
170 system().setWRGrid(w_);
171 simulator().clearData();
172
173 attemptMoveTimer_.stop();
174
175 // Call compressor
176 compressorTimer_.start();
177 int compress = simulator().compressor().compress();
178 UTIL_CHECK(system().hasCFields());
179 compressorTimer_.stop();
180
181 bool isConverged = false;
182 if (compress != 0){
183 incrementNFail();
184 simulator().restoreState();
185 } else {
186 isConverged = true;
187
188 // Compute eigenvector components of current fields
189 componentTimer_.start();
190 simulator().computeWc();
191 UTIL_CHECK(system().hasCFields());
192 simulator().computeCc();
193 simulator().computeDc();
194 componentTimer_.stop();
195
196 // Evaluate new Hamiltonian
197 hamiltonianTimer_.start();
198 simulator().computeHamiltonian();
199 double newHamiltonian = simulator().hamiltonian();
200 double dH = newHamiltonian - oldHamiltonian;
201
202 // Compute force bias
203 double bias = 0.0;
204 for (j = 0; j < nMonomer - 1; ++j) {
205 RField<D> const & di = dc_[j];
206 RField<D> const & df = simulator().dc(j);
207 RField<D> const & dwc = dwc_[j];
208
209 VecOpFts::computeForceBias(biasField_, di, df, dwc, mobility_);
210 bias += Reduce::sum(biasField_);
211 }
212 bias *= vNode;
213 hamiltonianTimer_.stop();
214
215 // Accept or reject move
216 decisionTimer_.start();
217 bool accept = false;
218 double weight = exp(bias - dH);
219 accept = random().metropolis(weight);
220 if (accept) {
221 incrementNAccept();
222 simulator().clearState();
223 } else {
224 simulator().restoreState();
225 }
226 decisionTimer_.stop();
227 }
228
229 totalTimer_.stop();
230 return isConverged;
231 }
232
233 /*
234 * Trivial default implementation - do nothing
235 */
236 template <int D>
239
240 template<int D>
241 void ForceBiasMove<D>::outputTimers(std::ostream& out)
242 {
243 // Output timing results, if requested.
244 out << "\n";
245 out << "ForceBiasMove time contributions:\n";
247 }
248
249}
250}
251#endif
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Field of real double precision values on an FFT mesh.
void setClassName(const char *className)
Set class name string.
void setup()
Setup before the beginning of each simulation run.
void output()
Output statistics for this move (at the end of simulation)
ForceBiasMove(McSimulator< D > &simulator)
Constructor.
void readParameters(std::istream &in)
Read required parameters from file.
bool move()
Attempt and accept or reject force bias Monte-Carlo move.
void outputTimers(std::ostream &out)
Return real move times contributions.
McMove is an abstract base class for Monte Carlo moves.
virtual void outputTimers(std::ostream &out)
Log output timing results.
virtual void setup()
Setup before the beginning of each simulation run.
Monte-Carlo simulation coordinator.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
cudaReal sum(DeviceArray< cudaReal > const &in)
Compute sum of array elements (GPU kernel wrapper).
Definition Reduce.cu:480
void eqV(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, const int beginIdA, const int beginIdB, const int n)
Vector assignment, a[i] = b[i], kernel wrapper (cudaReal).
Definition VecOp.cu:1020
void addVcVc(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, cudaReal const c, DeviceArray< cudaReal > const &d, cudaReal const e)
Vector addition w/ coefficient, a[i] = (b[i]*c) + (d[i]*e), kernel wrapper.
Definition VecOpMisc.cu:304
void addEqVc(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, cudaReal const c)
Vector addition in-place w/ coefficient, a[i] += b[i] * c, kernel wrapper.
Definition VecOpMisc.cu:343
void computeForceBias(DeviceArray< cudaReal > &result, DeviceArray< cudaReal > const &di, DeviceArray< cudaReal > const &df, DeviceArray< cudaReal > const &dwc, cudaReal mobility)
Compute force bias.
Definition VecOpFts.cu:133
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.