PSCF v1.2
FourierMove.tpp
1#ifndef RPG_FOURIER_MOVE_TPP
2#define RPG_FOURIER_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 "FourierMove.h"
12#include "McMove.h"
13#include <rpg/fts/VecOpFts.h>
14#include <rpg/System.h>
15#include <prdc/crystal/shiftToMinimum.h>
16#include <prdc/cuda/resources.h>
17#include <pscf/mesh/MeshIterator.h>
18#include <pscf/math/IntVec.h>
19#include <pscf/cuda/CudaRandom.h>
20#include <util/random/Random.h>
21#include <util/param/ParamComposite.h>
22#include <util/global.h>
23
24#include <iostream>
25#include <complex>
26#include <random>
27#include <cmath>
28
29namespace Pscf {
30namespace Rpg
31{
32
33 using namespace Util;
34
35 /*
36 * Constructor.
37 */
38 template <int D>
40 : McMove<D>(simulator),
41 isAllocated_(false)
42 { setClassName("FourierMove"); }
43
44 /*
45 * Destructor, empty default implementation.
46 */
47 template <int D>
50
51 /*
52 * ReadParameters
53 */
54 template <int D>
55 void FourierMove<D>::readParameters(std::istream &in)
56 {
57 //Read the probability
58 readProbability(in);
59 // attampt move range [A, -A]
60 read(in, "A", stepSize_);
61 read(in, "F*", fStar_);
62 read(in, "tau", tau_);
63 read(in, "N", n_);
64 read(in, "volumeFraction", f_);
65 read(in, "statisticalSegmentLength", b_);
66 }
67
68 template <int D>
70 {
72 const int nMonomer = system().mixture().nMonomer();
73 IntVec<D> const & dimensions = system().mesh().dimensions();
74
75 if (!isAllocated_){
76 wFieldTmp_.allocate(nMonomer);
77 randomFieldR_.allocate(dimensions);
78 randomFieldK_.allocate(dimensions);
79 wKGrid_.allocate(nMonomer);
80 wRGrid_.allocate(nMonomer);
81 sqrtSq_.allocate(dimensions);
82 for (int i = 0; i < nMonomer; ++i) {
83 wFieldTmp_[i].allocate(dimensions);
84 wKGrid_[i].allocate(dimensions);
85 wRGrid_[i].allocate(dimensions);
86 }
87 isAllocated_ = true;
88 }
89 computeRgSquare();
90 computeStructureFactor();
91 }
92
96 template <int D>
98 {
99 rgSquare_ = n_ * b_ * b_ / 6.0;
100 }
101
102 /*
103 * Debye function g(f,x)
104 */
105 template <int D>
106 double FourierMove<D>::computeDebye(double f, double x)
107 {
108 double fx = f * x;
109 return 2.0 * (fx + exp(-fx) - 1.0) / (x * x);
110 }
111
112 /*
113 * F(x,f) = g(1,x)/{g(f,x)g(1-f,x) - [g(1,x) - g(f,x) - g(1-f,x)]^2/4}
114 */
115 template <int D>
116 double FourierMove<D>::computeF(double x)
117 {
118 //Only for diblock
119 if (f_ > 0.5){
120 f_ = 1.0 - f_;
121 }
122
123 double g1 = computeDebye(1.0, x);
124 double gf = computeDebye(f_, x);
125 double g1f = computeDebye(1.0 - f_, x);
126 double denominator = gf * g1f - pow(g1 - gf- g1f, 2.0)/4.0;
127 return g1 / denominator;
128 }
129
130 /*
131 * S(q)/N Fredrickson-Helfand structure factor. S(q)/N = 1/(F(x) - F* + tau_)
132 */
133 template <int D>
134 double FourierMove<D>::computeS(double qSquare)
135 {
136 //Input is square magnitude of reciprocal basis vector
137 double x = qSquare * rgSquare_;
138 double denominator = computeF(x) - fStar_ + tau_;
139 // Nbar = Rho^(-2)*a^6* N && Rho = 1/vMonomer && vMonomer = 1/sqrt(Nbar)
140 const double vMonomer = system().mixture().vMonomer();
141 return 1.0 / (vMonomer * denominator);
142 }
143
144 template <int D>
145 void FourierMove<D>::computeStructureFactor()
146 {
147 int meshSize = system().domain().mesh().size();
148 HostDArray<cudaReal> sqTemp(meshSize);
149 MeshIterator<D> itr;
150 itr.setDimensions(system().mesh().dimensions());
151 IntVec<D> G; IntVec<D> Gmin;
152 double qSq; double S_;
153 for (itr.begin(); !itr.atEnd(); ++itr){
154 if (itr.rank() == 0){
155 sqTemp[itr.rank()] = 0;
156 } else {
157 G = itr.position();
158 Gmin = shiftToMinimum(G, system().mesh().dimensions(), system().unitCell());
159 qSq = system().unitCell().ksq(Gmin);
160 // Compute Fredrickson-Helfand structure factor
161 S_ = computeS(qSq);
162 sqTemp[itr.rank()] = sqrt(S_);
163 }
164 }
165 sqrtSq_ = sqTemp; // copy sqTemp to device
166 }
167
168 /*
169 * Attempt unconstrained move
170 */
171 template <int D>
173 {
174
175 const int nMonomer = system().mixture().nMonomer();
176 const int meshSize = system().domain().mesh().size();
177
178 for (int i = 0; i < nMonomer; ++i) {
179 VecOp::eqV(wRGrid_[i], system().w().rgrid(i));
180 }
181
182 // Convert real grid to KGrid format
183 for (int i = 0; i < nMonomer; ++i) {
184 system().fft().forwardTransform(wRGrid_[i], wKGrid_[i]);
185 }
186
187 for (int i = 0; i < nMonomer; ++i){
188 // Generate random number for real part
189 cudaRandom().uniform(randomFieldR_);
190 // Generate random numbers between [-stepSize_,stepSize_]
191 VecOpFts::mcftsScale(randomFieldR_, stepSize_);
192 // Generate random numbers between [-stepSize_*S(q)^(1/2), stepSize_*S(q)^(1/2)]
193 VecOp::mulEqV(randomFieldR_, sqrtSq_);
194
195 // Generate random number for imaginary part
196 cudaRandom().uniform(randomFieldK_);
197 // Generate random numbers between [-stepSize_,stepSize_]
198 VecOpFts::mcftsScale(randomFieldK_, stepSize_);
199 // Generate random numbers between [-stepSize_*S(q)^(1/2), stepSize_*S(q)^(1/2)]
200 VecOp::mulEqV(randomFieldK_, sqrtSq_);
201
202 // Attempt Move in ourier (k-grid) format
203 VecOpFts::fourierMove(wKGrid_[i], randomFieldR_, randomFieldK_);
204 }
205
206 // Convert back to real space (destroys data in wKGrid_)
207 for (int i = 0; i < nMonomer; ++i) {
208 system().fft().inverseTransformUnsafe(wKGrid_[i], wFieldTmp_[i]);
209 }
210 // Update attemptMove
211 system().setWRGrid(wFieldTmp_);
212 }
213
214
215 /*
216 * Trivial default implementation - do nothing
217 */
218 template <int D>
221
222 template<int D>
223 void FourierMove<D>::outputTimers(std::ostream& out)
224 {
225 // Output timing results, if requested.
226 out << "\n";
227 out << "FourierMove time contributions:\n";
229 }
230
231}
232}
233#endif
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
FourierMove is a Monte Carlo move in fourier space.
Definition FourierMove.h:31
FourierMove(McSimulator< D > &simulator)
Constructor.
void setClassName(const char *className)
Set class name string.
void attemptMove()
Attempt unconstrained move.
void setup()
Setup before the beginning of each simulation run.
void output()
Output statistics for this move (at the end of simulation)
~FourierMove()
Destructor.
void outputTimers(std::ostream &out)
Return fourier move times contributions.
void readParameters(std::istream &in)
Read required parameters from file.
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.
File containing preprocessor macros for error handling.
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 mulEqV(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, const int beginIdA, const int beginIdB, const int n)
Vector multiplication in-place, a[i] *= b[i], kernel wrapper (cudaReal).
Definition VecOp.cu:1824
void fourierMove(DeviceArray< cudaComplex > &a, DeviceArray< cudaReal > const &b, DeviceArray< cudaReal > const &c)
Add array b to real part of a and array c to imaginary part of a.
Definition VecOpFts.cu:92
void mcftsScale(DeviceArray< cudaReal > &a, cudaReal const b)
Rescale array a from [0,1] to [-b, b], GPU kernel wrapper.
Definition VecOpFts.cu:77
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.