PSCF v1.2
rpc/fts/brownian/PredCorrBdStep.tpp
1#ifndef RPC_PRED_CORR_BD_STEP_TPP
2#define RPC_PRED_CORR_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 "PredCorrBdStep.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 wp_(),
32 wf_(),
33 dci_(),
34 eta_(),
35 dwc_(),
36 dwp_(),
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 PredCorrBdStep<D>::readParameters(std::istream &in)
52 {
53 read(in, "mobility", mobility_);
54
55 // Allocate memory for private containers
56 int nMonomer = system().mixture().nMonomer();
57 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
58 wp_.allocate(nMonomer);
59 wf_.allocate(nMonomer);
60 for (int i=0; i < nMonomer; ++i) {
61 wp_[i].allocate(meshDimensions);
62 wf_[i].allocate(meshDimensions);
63 }
64 dci_.allocate(nMonomer-1);
65 eta_.allocate(nMonomer-1);
66 for (int i=0; i < nMonomer - 1; ++i) {
67 dci_[i].allocate(meshDimensions);
68 eta_[i].allocate(meshDimensions);
69 }
70 dwc_.allocate(meshDimensions);
71 dwp_.allocate(meshDimensions);
72 }
73
74 template <int D>
76 {
77 // Check array capacities
78 int meshSize = system().domain().mesh().size();
79 int nMonomer = system().mixture().nMonomer();
80 UTIL_CHECK(wp_.capacity() == nMonomer);
81 UTIL_CHECK(wf_.capacity() == nMonomer);
82 for (int i=0; i < nMonomer; ++i) {
83 UTIL_CHECK(wp_[i].capacity() == meshSize);
84 UTIL_CHECK(wf_[i].capacity() == meshSize);
85 }
86 UTIL_CHECK(dci_.capacity() == nMonomer-1);
87 UTIL_CHECK(eta_.capacity() == nMonomer-1);
88 for (int i=0; i < nMonomer - 1; ++i) {
89 UTIL_CHECK(dci_[i].capacity() == meshSize);
90 UTIL_CHECK(eta_[i].capacity() == meshSize);
91 }
92 UTIL_CHECK(dwc_.capacity() == meshSize);
93 UTIL_CHECK(dwp_.capacity() == meshSize);
94 }
95
96 template <int D>
98 {
99 // Array sizes and indices
100 const int nMonomer = system().mixture().nMonomer();
101 const int meshSize = system().domain().mesh().size();
102 int i, j, k;
103
104 // Save current state
105 simulator().saveState();
106
107 // Copy current W fields from parent system
108 for (i = 0; i < nMonomer; ++i) {
109 wp_[i] = system().w().rgrid(i);
110 wf_[i] = wp_[i];
111 }
112
113 // Store initial value of pressure field at all grid points in dwp_
114 dwp_ = simulator().wc(nMonomer-1);
115
116 // Define constants used for step
117 const double vSystem = system().domain().unitCell().volume();
118 const double a = -1.0*mobility_;
119 const double b = sqrt(2.0*mobility_*double(meshSize)/vSystem);
120
121 // Construct all random displacement (noise) components
122 for (j = 0; j < nMonomer - 1; ++j) {
123 RField<D> & eta = eta_[j];
124 for (k = 0; k < meshSize; ++k) {
125 eta[k] = b*random().gaussian();
126 }
127 }
128
129 // Compute predicted state wp_, and store initial force dci_
130
131 // Loop over eigenvectors of projected chi matrix
132 double evec;
133 for (j = 0; j < nMonomer - 1; ++j) {
134 RField<D> const & dc = simulator().dc(j);
135 RField<D> const & eta = eta_[j];
136 RField<D> & dci = dci_[j];
137 for (k = 0; k < meshSize; ++k) {
138 dwc_[k] = a*dc[k] + eta[k];
139 dci[k] = dc[k];
140 }
141 // Loop over monomer types
142 for (i = 0; i < nMonomer; ++i) {
143 RField<D> & wp = wp_[i];
144 evec = simulator().chiEvecs(j,i);
145 for (k = 0; k < meshSize; ++k) {
146 wp[k] += evec*dwc_[k];
147 }
148 }
149 }
150
151 // Set modified system fields at predicted state wp_
152 system().setWRGrid(wp_);
153
154 // Set function return value to indicate failure by default
155 bool isConverged = false;
156
157 // Enforce incompressibility at predicted state
158 int compress = simulator().compressor().compress();
159 if (compress != 0){
160 simulator().restoreState();
161 } else {
162 UTIL_CHECK(system().hasCFields());
163
164 // Compute components and derivatives at wp_
165 simulator().clearData();
166 simulator().computeWc();
167 simulator().computeCc();
168 simulator().computeDc();
169
170 // Compute change dwp_ in pressure field
171 // Note: On entry, dwp_ is the old pressure field
172 RField<D> const & wp = simulator().wc(nMonomer-1);
173 for (k = 0; k < meshSize; ++k) {
174 dwp_[k] = wp[k] - dwp_[k];
175 }
176
177 // Adjust predicted pressure field to final monomer fields
178 for (i = 0; i < nMonomer; ++i) {
179 RField<D> & wf = wf_[i];
180 for (k = 0; k < meshSize; ++k) {
181 wf[k] += dwp_[k];
182 }
183 }
184
185 // Full step (corrector) change in exchange fields
186 const double ha = 0.5*a;
187 for (j = 0; j < nMonomer - 1; ++j) {
188 RField<D> const & dcp = simulator().dc(j);
189 RField<D> const & dci = dci_[j];
190 RField<D> const & eta = eta_[j];
191 for (k = 0; k < meshSize; ++k) {
192 dwc_[k] = ha*( dci[k] + dcp[k]) + eta[k];
193 }
194 for (i = 0; i < nMonomer; ++i) {
195 RField<D> & wf = wf_[i];
196 evec = simulator().chiEvecs(j,i);
197 for (k = 0; k < meshSize; ++k) {
198 wf[k] += evec*dwc_[k];
199 }
200 }
201 }
202
203 // Set system fields after predictor step
204 system().setWRGrid(wf_);
205
206 // Apply compressor to final state
207 int compress2 = simulator().compressor().compress();
208 if (compress2 != 0){
209 simulator().restoreState();
210 } else {
211 isConverged = true;
212 UTIL_CHECK(system().hasCFields());
213
214 // Compute components and derivatives at final point
215 simulator().clearState();
216 simulator().clearData();
217 simulator().computeWc();
218 simulator().computeCc();
219 simulator().computeDc();
220
221 }
222 }
223
224 // True iff compression was successful after predictor and corrector
225 return isConverged;
226 }
227
228}
229}
230#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.
virtual void readParameters(std::istream &in)
Read required parameters from file.
virtual void setup()
Setup before simulation.
virtual bool step()
Take a single Brownian dynamics step.
PredCorrBdStep(BdSimulator< D > &simulator)
Constructor.
#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.