PSCF v1.2
rpc/fts/brownian/LMBdStep.tpp
1#ifndef RPC_LM_BD_STEP_TPP
2#define RPC_LM_BD_STEP_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 "LMBdStep.h"
12
13#include <rpc/fts/brownian/BdSimulator.h>
14#include <rpc/fts/compressor/Compressor.h>
15#include <rpc/System.h>
16#include <pscf/math/IntVec.h>
17#include <util/random/Random.h>
18
19namespace Pscf {
20namespace Rpc {
21
22 using namespace Util;
23 using namespace Prdc::Cpu;
24
25 /*
26 * Constructor.
27 */
28 template <int D>
30 : BdStep<D>(simulator),
31 w_(),
32 etaA_(),
33 etaB_(),
34 dwc_(),
35 etaNewPtr_(0),
36 etaOldPtr_(0),
37 mobility_(0.0)
38 {}
39
40 /*
41 * Destructor, empty default implementation.
42 */
43 template <int D>
46
47 /*
48 * ReadParameters, empty default implementation.
49 */
50 template <int D>
51 void LMBdStep<D>::readParameters(std::istream &in)
52 {
53 read(in, "mobility", mobility_);
54
55 int nMonomer = system().mixture().nMonomer();
56 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
57
58 // Allocate memory for private containers
59 w_.allocate(nMonomer);
60 for (int i=0; i < nMonomer; ++i) {
61 w_[i].allocate(meshDimensions);
62 }
63 etaA_.allocate(nMonomer-1);
64 etaB_.allocate(nMonomer-1);
65 for (int i=0; i < nMonomer - 1; ++i) {
66 etaA_[i].allocate(meshDimensions);
67 etaB_[i].allocate(meshDimensions);
68 }
69 dwc_.allocate(meshDimensions);
70 }
71
72 /*
73 * Generate new random displacement values
74 */
75 template <int D>
77 {
78 const int nMonomer = system().mixture().nMonomer();
79 const int meshSize = system().domain().mesh().size();
80
81 // Prefactor b for displacements
82 const double vSystem = system().domain().unitCell().volume();
83 const double b = sqrt(0.5*mobility_*double(meshSize)/vSystem);
84
85 int j, k;
86 for (j = 0; j < nMonomer - 1; ++j) {
87 RField<D>& eta = etaNew(j);
88 for (k = 0; k < meshSize; ++k) {
89 eta[k] = b*random().gaussian();
90 }
91 }
92 }
93
94 /*
95 * Exchange pointers to old and new random fields.
96 */
97 template <int D>
98 void LMBdStep<D>::exchangeOldNew()
99 {
100 DArray< RField<D> >* temp;
101 temp = etaOldPtr_;
102 etaOldPtr_ = etaNewPtr_;
103 etaNewPtr_ = temp;
104 }
105
106 /*
107 * Initial setup of stepper before main simulation loop.
108 */
109 template <int D>
111 {
112 int nMonomer = system().mixture().nMonomer();
113 int meshSize = system().domain().mesh().size();
114
115 // Check array capacities
116 UTIL_CHECK(etaA_.capacity() == nMonomer-1);
117 UTIL_CHECK(etaB_.capacity() == nMonomer-1);
118 for (int i=0; i < nMonomer - 1; ++i) {
119 UTIL_CHECK(etaA_[i].capacity() == meshSize);
120 UTIL_CHECK(etaB_[i].capacity() == meshSize);
121 }
122 UTIL_CHECK(dwc_.capacity() == meshSize);
123
124 // Initialize pointers
125 etaOldPtr_ = &etaA_;
126 etaNewPtr_ = &etaB_;
127
128 generateEtaNew();
129 exchangeOldNew();
130 }
131
132 /*
133 * One step of Leimkuhler-Matthews BD algorithm.
134 */
135 template <int D>
137 {
138 // Array sizes and indices
139 const int nMonomer = system().mixture().nMonomer();
140 const int meshSize = system().domain().mesh().size();
141 int i, j, k;
142
143 // Save current state
144 simulator().saveState();
145
146 // Copy current fields to w_
147 for (i = 0; i < nMonomer; ++i) {
148 w_[i] = system().w().rgrid(i);
149 }
150
151 // Generate new random displacement values
152 generateEtaNew();
153
154 // Take LM step:
155 const double a = -1.0*mobility_;
156 double evec;
157 // Loop over composition eigenvectors of projected chi matrix
158 for (j = 0; j < nMonomer - 1; ++j) {
159 RField<D> const & etaN = etaNew(j);
160 RField<D> const & etaO = etaOld(j);
161 RField<D> const & dc = simulator().dc(j);
162 for (k = 0; k < meshSize; ++k) {
163 dwc_[k] = a*dc[k] + etaN[k] + etaO[k];
164 }
165 // Loop over monomer types
166 for (i = 0; i < nMonomer; ++i) {
167 RField<D> & w = w_[i];
168 evec = simulator().chiEvecs(j,i);
169 for (k = 0; k < meshSize; ++k) {
170 w[k] += evec*dwc_[k];
171 }
172 }
173 }
174
175 // Set modified fields
176 system().setWRGrid(w_);
177
178 // Enforce incompressibility (also solves MDE repeatedly)
179 bool isConverged = false;
180 int compress = simulator().compressor().compress();
181 if (compress != 0){
182 simulator().restoreState();
183 } else {
184 isConverged = true;
185 UTIL_CHECK(system().hasCFields());
186
187 // Compute components and derivatives at wp_
188 simulator().clearState();
189 simulator().clearData();
190 simulator().computeWc();
191 simulator().computeCc();
192 simulator().computeDc();
193
194 // Exchange old and new random fields
195 exchangeOldNew();
196
197 }
198
199 return isConverged;
200 }
201
202}
203}
204#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.
Brownian dynamics simulator for PS-FTS.
BdStep is an abstract base class for Brownian dynamics steps.
Leimkuhler-Matthews Brownian dynamics stepper.
virtual bool step()
Take a single Brownian dynamics step.
virtual void readParameters(std::istream &in)
Read required parameters from file.
virtual void setup()
Setup before simulation.
LMBdStep(BdSimulator< D > &simulator)
Constructor.
Dynamically allocatable contiguous array template.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.