PSCF v1.2
rpc/fts/analyzer/FourthOrderParameter.tpp
1#ifndef RPC_FOURTH_ORDER_PARAMETER_TPP
2#define RPC_FOURTH_ORDER_PARAMETER_TPP
3
4#include "FourthOrderParameter.h"
5
6#include <rpc/fts/simulator/Simulator.h>
7#include <rpc/System.h>
8
9#include <prdc/cpu/RField.h>
10#include <prdc/crystal/shiftToMinimum.h>
11
12#include <pscf/mesh/MeshIterator.h>
13#include <pscf/math/IntVec.h>
14
15#include <util/containers/DArray.h>
16#include <util/param/ParamComposite.h>
17#include <util/misc/FileMaster.h>
18#include <util/misc/ioUtil.h>
19#include <util/format/Int.h>
20#include <util/format/Dbl.h>
21#include <util/global.h>
22
23#include <fftw3.h>
24
25#include <iostream>
26#include <complex>
27#include <vector>
28#include <numeric>
29#include <cmath>
30#include <set>
31
32namespace Pscf {
33namespace Rpc {
34
35 using namespace Util;
36 using namespace Pscf::Prdc;
37
38 /*
39 * Constructor.
40 */
41 template <int D>
43 System<D>& system)
44 : AverageAnalyzer<D>(simulator, system),
45 kSize_(1),
46 isInitialized_(false)
47 { setClassName("FourthOrderParameter"); }
48
49 /*
50 * Destructor.
51 */
52 template <int D>
55
56 /*
57 * FourthOrderParameter setup
58 */
59 template <int D>
61 {
63
64 IntVec<D> const & dimensions = system().domain().mesh().dimensions();
65
66 // Compute Fourier space dimension
67 for (int i = 0; i < D; ++i) {
68 if (i < D - 1) {
69 kMeshDimensions_[i] = dimensions[i];
70 kSize_ *= dimensions[i];
71 } else {
72 kMeshDimensions_[i] = dimensions[i]/2 + 1;
73 kSize_ *= (dimensions[i]/2 + 1);
74 }
75 }
76
77 // Allocate variables
78 if (!isInitialized_){
79 wK_.allocate(dimensions);
80 prefactor_.allocate(kSize_);
81 for (int i = 0; i < kSize_; ++i){
82 prefactor_[i] = 0;
83 }
84 }
85
86 isInitialized_ = true;
87
88 if (!isInitialized_) {
89 UTIL_THROW("Error: object is not initialized");
90 }
91
92 computePrefactor();
93 }
94
95 template <int D>
97 {
98 UTIL_CHECK(system().w().hasData());
99
100 // For AB diblock
101 const int nMonomer = system().mixture().nMonomer();
102 UTIL_CHECK(nMonomer == 2);
103
104 if (!simulator().hasWc()){
105 simulator().computeWc();
106 }
107
108 MeshIterator<D> itr;
109 itr.setDimensions(kMeshDimensions_);
110 std::vector<double> psi(kSize_);
111
112 // Conver W_(r) to fourier mode W_(k)
113 system().domain().fft().forwardTransform(simulator().wc(0), wK_);
114
115 for (itr.begin(); !itr.atEnd(); ++itr) {
116 std::complex<double> wK(wK_[itr.rank()][0], wK_[itr.rank()][1]);
117 psi[itr.rank()] = std::norm(wK) * std::norm(wK);
118 psi[itr.rank()] *= prefactor_[itr.rank()];
119 }
120
121 // Get sum over all wavevectors
122 FourthOrderParameter_ = std::accumulate(psi.begin(), psi.end(), 0.0);
123 FourthOrderParameter_ = std::pow(FourthOrderParameter_, 0.25);
124
125 return FourthOrderParameter_;
126
127 #if 0
128 // Debugging output
129 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
130 UnitCell<D> const & unitCell = system().domain().unitCell();
131 IntVec<D> G;
132 IntVec<D> Gmin;
133 IntVec<D> nGmin;
134 double kSq;
135 std::vector<double> k(kSize_);
136
137 // Calculate GminList
138 for (itr.begin(); !itr.atEnd(); ++itr){
139 G = itr.position();
140 Gmin = shiftToMinimum(G, meshDimensions, unitCell);
141 kSq = unitCell.ksq(Gmin);
142 k[itr.rank()] = kSq;
143 }
144
145 auto maxIt = std::max_element(psi.begin(), psi.end());
146
147 // Calculate the index of the maximum element
148 size_t maxIndex = std::distance(psi.begin(), maxIt);
149 double kmax = k[maxIndex];
150
151 Log::file() << std::endl;
152 for (itr.begin(); !itr.atEnd(); ++itr){
153 if (k[itr.rank()] == kmax){
154 G = itr.position();
155 Gmin = shiftToMinimum(G, meshDimensions, unitCell);
156 Log::file() << "ksq: " << k[itr.rank()] << std::endl;
157 Log::file() << " G: " << G<< std::endl;
158 Log::file() << " Gmin: " << Gmin<< std::endl;
159 Log::file() << " prefactor: " << prefactor_[itr.rank()]<< std::endl;
160 Log::file() << " psi: " << psi[itr.rank()]<< std::endl;
161 }
162
163 }
164 #endif
165
166 }
167
168 template <int D>
169 void FourthOrderParameter<D>::outputValue(int step, double value)
170 {
171 if (simulator().hasRamp() && nSamplePerOutput() == 1) {
172 double chi= system().interaction().chi(0,1);
173
174 UTIL_CHECK(outputFile_.is_open());
175 outputFile_ << Int(step);
176 outputFile_ << Dbl(chi);
177 outputFile_ << Dbl(value);
178 outputFile_ << "\n";
179 } else {
181 }
182 }
183
184 template <int D>
186 {
187 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
188 UnitCell<D> const & unitCell = system().domain().unitCell();
189 IntVec<D> G;
190 IntVec<D> Gmin;
191 IntVec<D> nGmin;
192 DArray<IntVec<D>> GminList;
193 GminList.allocate(kSize_);
194 MeshIterator<D> itr(kMeshDimensions_);
195 MeshIterator<D> searchItr(kMeshDimensions_);
196
197 // Calculate GminList
198 for (itr.begin(); !itr.atEnd(); ++itr){
199 G = itr.position();
200 Gmin = shiftToMinimum(G, meshDimensions, unitCell);
201 GminList[itr.rank()] = Gmin;
202 }
203
204 // Compute prefactor for each G wavevector
205 for (itr.begin(); !itr.atEnd(); ++itr){
206 bool inverseFound = false;
207
208 // If prefactor of current wavevector has not been assigned
209 if (prefactor_[itr.rank()] == 0){
210 Gmin = GminList[itr.rank()];
211
212 // Compute inverse of wavevector
213 nGmin.negate(Gmin);
214
215 // Search for inverse of wavevector
216 searchItr = itr;
217 for (; !searchItr.atEnd(); ++searchItr){
218 if (nGmin == GminList[searchItr.rank()]){
219 prefactor_[itr.rank()] = 1.0/2.0;
220 prefactor_[searchItr.rank()] = 1.0/2.0;
221 inverseFound = true;
222 }
223 }
224
225 if (inverseFound == false){
226 prefactor_[itr.rank()] = 1.0;
227 }
228
229 }
230
231 }
232 }
233
234}
235}
236#endif
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Iterator over points in a Mesh<D>.
int rank() const
Get the rank of current element.
void begin()
Set iterator to the first point in the mesh.
bool atEnd() const
Is this the end (i.e., one past the last point)?
void setDimensions(const IntVec< D > &dimensions)
Set the grid dimensions in all directions.
IntVec< D > position() const
Get current position in the grid, as integer vector.
virtual double ksq(IntVec< D > const &k) const
Compute square magnitude of reciprocal lattice vector.
Base template for UnitCell<D> classes, D=1, 2 or 3.
Definition rpg/System.h:34
Analyze averages and block averages of several real variables.
virtual void setup()
Setup before loop.
virtual void outputValue(int step, double value)
Output a sampled or block average value.
void setClassName(const char *className)
Set class name string.
FourthOrderParameter(Simulator< D > &simulator, System< D > &system)
Constructor.
void computePrefactor()
Compute prefactor for each Fourier wavevector.
virtual void setup()
Setup before simulation loop.
virtual void outputValue(int step, double value)
Output a sampled or block average value.
virtual double compute()
Compute and return the derivative of H w/ respect to chi.
Field theoretic simulator (base class).
Definition rpc/System.h:38
Main class for SCFT or PS-FTS simulation of one system.
Definition rpc/System.h:100
Vec< D, T > & negate(const Vec< D, T > &v)
Return negative of vector v.
Definition Vec.h:520
Dynamically allocatable contiguous array template.
void allocate(int capacity)
Allocate the underlying C array.
Definition DArray.h:199
Wrapper for a double precision number, for formatted ostream output.
Definition Dbl.h:40
Wrapper for an int, for formatted ostream output.
Definition Int.h:37
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
File containing preprocessor macros for error handling.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Definition global.h:51
Periodic fields and crystallography.
Definition CField.cpp:11
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.