PSCF v1.2
rpg/fts/brownian/LMBdStep.tpp
1#ifndef RPG_LM_BD_STEP_TPP
2#define RPG_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 <rpg/fts/brownian/BdSimulator.h>
14#include <rpg/fts/compressor/Compressor.h>
15#include <rpg/System.h>
16#include <prdc/cuda/VecOp.h>
17#include <pscf/math/IntVec.h>
18#include <pscf/cuda/CudaRandom.h>
19
20namespace Pscf {
21namespace Rpg {
22
23 using namespace Util;
24 using namespace Pscf::Prdc;
25 using namespace Pscf::Prdc::Cuda;
26
27 /*
28 * Constructor.
29 */
30 template <int D>
32 : BdStep<D>(simulator),
33 w_(),
34 etaA_(),
35 etaB_(),
36 dwc_(),
37 etaNewPtr_(0),
38 etaOldPtr_(0),
39 mobility_(0.0)
40 {}
41
42 /*
43 * Destructor, empty default implementation.
44 */
45 template <int D>
48
49 /*
50 * ReadParameters, empty default implementation.
51 */
52 template <int D>
53 void LMBdStep<D>::readParameters(std::istream &in)
54 {
55 read(in, "mobility", mobility_);
56
57 int nMonomer = system().mixture().nMonomer();
58 int meshSize = system().domain().mesh().size();
59 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
60
61 // Allocate memory for private containers
62 w_.allocate(nMonomer);
63 for (int i=0; i < nMonomer; ++i) {
64 w_[i].allocate(meshDimensions);
65 }
66 etaA_.allocate(nMonomer-1);
67 etaB_.allocate(nMonomer-1);
68 for (int i=0; i < nMonomer - 1; ++i) {
69 etaA_[i].allocate(meshDimensions);
70 etaB_[i].allocate(meshDimensions);
71 }
72 dwc_.allocate(meshDimensions);
73 gaussianField_.allocate(meshDimensions);
74 }
75
76 /*
77 * Generate new random displacement values
78 */
79 template <int D>
81 {
82 const int nMonomer = system().mixture().nMonomer();
83 const int meshSize = system().domain().mesh().size();
84
85 // Prefactor b for displacements
86 const double vSystem = system().domain().unitCell().volume();
87 double b = sqrt(0.5*mobility_*double(meshSize)/vSystem);
88
89 // Constants for normal distribution
90 double stddev = 1.0;
91 double mean = 0;
92
93 int j;
94 for (j = 0; j < nMonomer - 1; ++j) {
95 RField<D>& eta = etaNew(j);
96
97 // Generate normal distributed random floating point numbers
98 cudaRandom().normal(gaussianField_, stddev, mean);
99 VecOp::mulVS(eta, gaussianField_, b);
100
101 }
102 }
103
104 template <int D>
105 void LMBdStep<D>::exchangeOldNew()
106 {
107 DArray< RField<D> >* temp;
108 temp = etaOldPtr_;
109 etaOldPtr_ = etaNewPtr_;
110 etaNewPtr_ = temp;
111 }
112
113 /*
114 * Initial setup before main simulation loop.
115 */
116 template <int D>
118 {
119 int nMonomer = system().mixture().nMonomer();
120 int meshSize = system().domain().mesh().size();
121
122 // Check array capacities
123 UTIL_CHECK(etaA_.capacity() == nMonomer-1);
124 UTIL_CHECK(etaB_.capacity() == nMonomer-1);
125 for (int i=0; i < nMonomer - 1; ++i) {
126 UTIL_CHECK(etaA_[i].capacity() == meshSize);
127 UTIL_CHECK(etaB_[i].capacity() == meshSize);
128 }
129 UTIL_CHECK(dwc_.capacity() == meshSize);
130
131 // Initialize pointers
132 etaOldPtr_ = &etaA_;
133 etaNewPtr_ = &etaB_;
134
135 generateEtaNew();
136 exchangeOldNew();
137 }
138
139 /*
140 * One step of Leimkuhler-Matthews BD algorithm.
141 */
142 template <int D>
144 {
145 // Array sizes and indices
146 const int nMonomer = system().mixture().nMonomer();
147 const int meshSize = system().domain().mesh().size();
148 int i, j;
149
150 // Save current state
151 simulator().saveState();
152
153 // Copy current W fields from parent system into w_
154 for (i = 0; i < nMonomer; ++i) {
155 VecOp::eqV(w_[i], system().w().rgrid(i));
156 }
157
158 // Generate new random displacement values
159 generateEtaNew();
160
161 // Take LM step:
162 double a = -1.0*mobility_;
163 double evec;
164
165 // Loop over composition eigenvectors of projected chi matrix
166 for (j = 0; j < nMonomer - 1; ++j) {
167 RField<D> const & etaN = etaNew(j);
168 RField<D> const & etaO = etaOld(j);
169 RField<D> const & dc = simulator().dc(j);
170 VecOp::addVcVcVc(dwc_, etaN, 1.0, etaO, 1.0, dc, a);
171 // Loop over monomer types
172 for (i = 0; i < nMonomer; ++i) {
173 RField<D> & w = w_[i];
174 evec = simulator().chiEvecs(j,i);
175 VecOp::addEqVc(w, dwc_, evec);
176 }
177 }
178
179 // Set modified fields
180 system().setWRGrid(w_);
181
182 // Enforce incompressibility (also solves MDE repeatedly)
183 bool isConverged = false;
184 int compress = simulator().compressor().compress();
185 if (compress != 0){
186 simulator().restoreState();
187 } else{
188 isConverged = true;
189 UTIL_CHECK(system().hasCFields());
190
191 // Compute components and derivatives at wp_
192 simulator().clearState();
193 simulator().clearData();
194 simulator().computeWc();
195 simulator().computeCc();
196 simulator().computeDc();
197
198 // Exchange old and new random fields
199 exchangeOldNew();
200
201 }
202
203 return isConverged;
204
205 }
206
207}
208}
209#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.
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 setup()
Setup before simulation.
virtual void readParameters(std::istream &in)
Read required parameters from file.
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
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 addVcVcVc(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, cudaReal const c, DeviceArray< cudaReal > const &d, cudaReal const e, DeviceArray< cudaReal > const &f, cudaReal const g)
3-vec addition w coeff, a[i] = (b[i]*c) + (d[i]*e) + (f[i]*g), kernel wrapper.
Definition VecOpMisc.cu:322
void mulVS(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, cudaReal const c, const int beginIdA, const int beginIdB, const int n)
Vector multiplication, a[i] = b[i] * c, kernel wrapper (cudaReal).
Definition VecOp.cu:1452
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
Fields, FFTs, and utilities for periodic boundary conditions (CUDA)
Definition CField.cu:12
Periodic fields and crystallography.
Definition CField.cpp:11
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.