PSCF v1.2
rpg/fts/analyzer/FourthOrderParameter.tpp
1#ifndef RPG_FOURTH_ORDER_PARAMETER_TPP
2#define RPG_FOURTH_ORDER_PARAMETER_TPP
3
4#include "FourthOrderParameter.h"
5
6#include <rpg/fts/simulator/Simulator.h>
7#include <rpg/System.h>
8
9#include <prdc/crystal/shiftToMinimum.h>
10#include <prdc/cuda/resources.h>
11
12#include <pscf/mesh/MeshIterator.h>
13#include <pscf/math/IntVec.h>
14
15#include <util/param/ParamComposite.h>
16#include <util/misc/FileMaster.h>
17#include <util/misc/ioUtil.h>
18#include <util/format/Int.h>
19#include <util/format/Dbl.h>
20#include <util/global.h>
21
22#include <fftw3.h>
23
24#include <iostream>
25#include <complex>
26#include <vector>
27#include <numeric>
28#include <cmath>
29
30namespace Pscf {
31namespace Rpg {
32
33 using namespace Util;
34 using namespace Pscf::Prdc;
35
36 /*
37 * Constructor.
38 */
39 template <int D>
41 System<D>& system)
42 : AverageAnalyzer<D>(simulator, system),
43 kSize_(1),
44 isInitialized_(false)
45 { setClassName("FourthOrderParameter"); }
46
47 /*
48 * Destructor.
49 */
50 template <int D>
53
54 /*
55 * FourthOrderParameter setup
56 */
57 template <int D>
59 {
61
62 IntVec<D> const & dimensions = system().mesh().dimensions();
63
64 // Compute Fourier space dimension
65 for (int i = 0; i < D; ++i) {
66 if (i < D - 1) {
67 kMeshDimensions_[i] = dimensions[i];
68 kSize_ *= dimensions[i];
69 } else {
70 kMeshDimensions_[i] = dimensions[i]/2 + 1;
71 kSize_ *= (dimensions[i]/2 + 1);
72 }
73 }
74
75 // Allocate variables
76 if (!isInitialized_){
77 wc0_.allocate(dimensions);
78 wK_.allocate(dimensions);
79 prefactor_.allocate(kMeshDimensions_);
80 VecOp::eqS(prefactor_, 0);
81 }
82
83 isInitialized_ = true;
84
85 if (!isInitialized_) {
86 UTIL_THROW("Error: object is not initialized");
87 }
88
89 computePrefactor();
90 }
91
92 template <int D>
94 {
95 UTIL_CHECK(system().w().hasData());
96
97 // For AB diblock
98 const int nMonomer = system().mixture().nMonomer();
99 UTIL_CHECK(nMonomer == 2);
100
101 if (!simulator().hasWc()){
102 simulator().computeWc();
103 }
104
105 const int meshSize = system().domain().mesh().size();
106 RField<D> psi;
107 psi.allocate(kMeshDimensions_);
108
109 // Convert W_(r) to fourier mode W_(k)
110 VecOp::eqV(wc0_, simulator().wc(0));
111 system().fft().forwardTransform(wc0_, wK_);
112
113 // psi = |wK_|^4
114 VecOp::sqSqNormV(psi, wK_);
115
116 // W_(k)^4 * weight factor
117 VecOp::mulEqV(psi, prefactor_);
118
119 // Get sum over all wavevectors
120 FourthOrderParameter_ = Reduce::sum(psi);
121 FourthOrderParameter_ = std::pow(FourthOrderParameter_, 0.25);
122
123 return FourthOrderParameter_;
124 }
125
126 template <int D>
127 void FourthOrderParameter<D>::outputValue(int step, double value)
128 {
129 if (simulator().hasRamp() && nSamplePerOutput() == 1) {
130 double chi= system().interaction().chi(0,1);
131
132 UTIL_CHECK(outputFile_.is_open());
133 outputFile_ << Int(step);
134 outputFile_ << Dbl(chi);
135 outputFile_ << Dbl(value);
136 outputFile_ << "\n";
137 } else {
139 }
140 }
141
142 template <int D>
144 {
145 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
146 UnitCell<D> const & unitCell = system().domain().unitCell();
147 HostDArray<cudaReal> prefactor_h(kSize_);
148 for (int i = 0; i < kSize_; ++i){
149 prefactor_h[i] = 0;
150 }
151 IntVec<D> G;
152 IntVec<D> Gmin;
153 IntVec<D> nGmin;
154 DArray<IntVec<D>> GminList;
155 GminList.allocate(kSize_);
156 MeshIterator<D> itr(kMeshDimensions_);
157 MeshIterator<D> searchItr(kMeshDimensions_);
158
159 // Calculate GminList
160 for (itr.begin(); !itr.atEnd(); ++itr){
161 G = itr.position();
162 Gmin = shiftToMinimum(G, meshDimensions, unitCell);
163 GminList[itr.rank()] = Gmin;
164 }
165
166 // Compute weight factor for each G wavevector
167 for (itr.begin(); !itr.atEnd(); ++itr){
168 bool inverseFound = false;
169
170 // If the weight factor of the current wavevector has not been assigned
171 if (prefactor_h[itr.rank()] == 0){
172 Gmin = GminList[itr.rank()];
173
174 // Compute inverse of wavevector
175 nGmin.negate(Gmin);
176
177 // Search for inverse of wavevector
178 searchItr = itr;
179 for (; !searchItr.atEnd(); ++searchItr){
180 if (nGmin == GminList[searchItr.rank()]){
181 prefactor_h[itr.rank()] = 1.0/2.0;
182 prefactor_h[searchItr.rank()] = 1.0/2.0;
183 inverseFound = true;
184 }
185 }
186
187 if (inverseFound == false){
188 prefactor_h[itr.rank()] = 1.0;
189 }
190
191 }
192
193 }
194
195 // Copy the weight factor from cpu(host) to gpu(device)
196 prefactor_ = prefactor_h;
197 }
198
199}
200}
201#endif
Template for dynamic array stored in host CPU memory.
Definition HostDArray.h:43
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)?
IntVec< D > position() const
Get current position in the grid, as integer vector.
Field of real double precision values on an FFT mesh.
void allocate(IntVec< D > const &meshDimensions)
Allocate the underlying C array for an FFT grid.
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.
virtual double compute()
Compute and return the fourth order parameter.
void setClassName(const char *className)
Set class name string.
FourthOrderParameter(Simulator< D > &simulator, System< D > &system)
Constructor.
virtual void outputValue(int step, double value)
Output a sampled or block average value.
virtual void setup()
Setup before simulation loop.
void computePrefactor()
Compute prefactor for each Fourier wavevector.
Field theoretic simulator (base class).
Definition rpg/System.h:41
Main class for calculations that represent one system.
Definition rpg/System.h:107
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
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
cudaReal sum(DeviceArray< cudaReal > const &in)
Compute sum of array elements (GPU kernel wrapper).
Definition Reduce.cu:480
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 sqSqNormV(DeviceArray< cudaReal > &a, DeviceArray< cudaComplex > const &b)
Norm of complex number to the 4th power, a[i] = norm(b[i])^4, kernel wrapper.
Definition VecOpMisc.cu:600
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 eqS(DeviceArray< cudaReal > &a, cudaReal const b, const int beginIdA, const int n)
Vector assignment, a[i] = b, kernel wrapper (cudaReal).
Definition VecOp.cu:1054
Periodic fields and crystallography.
Definition CField.cpp:11
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.