1#ifndef RPC_FOURTH_ORDER_PARAMETER_TPP
2#define RPC_FOURTH_ORDER_PARAMETER_TPP
4#include "FourthOrderParameter.h"
6#include <rpc/fts/simulator/Simulator.h>
9#include <prdc/cpu/RField.h>
10#include <prdc/crystal/shiftToMinimum.h>
12#include <pscf/mesh/MeshIterator.h>
13#include <pscf/math/IntVec.h>
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>
64 IntVec<D> const & dimensions = system().domain().mesh().dimensions();
67 for (
int i = 0; i < D; ++i) {
69 kMeshDimensions_[i] = dimensions[i];
70 kSize_ *= dimensions[i];
72 kMeshDimensions_[i] = dimensions[i]/2 + 1;
73 kSize_ *= (dimensions[i]/2 + 1);
79 wK_.allocate(dimensions);
80 prefactor_.allocate(kSize_);
81 for (
int i = 0; i < kSize_; ++i){
86 isInitialized_ =
true;
88 if (!isInitialized_) {
89 UTIL_THROW(
"Error: object is not initialized");
101 const int nMonomer = system().mixture().nMonomer();
104 if (!simulator().hasWc()){
105 simulator().computeWc();
110 std::vector<double> psi(kSize_);
113 system().domain().fft().forwardTransform(simulator().wc(0), wK_);
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()];
122 FourthOrderParameter_ = std::accumulate(psi.begin(), psi.end(), 0.0);
123 FourthOrderParameter_ = std::pow(FourthOrderParameter_, 0.25);
125 return FourthOrderParameter_;
129 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
130 UnitCell<D> const & unitCell = system().domain().unitCell();
135 std::vector<double> k(kSize_);
140 Gmin = shiftToMinimum(G, meshDimensions, unitCell);
141 kSq = unitCell.
ksq(Gmin);
145 auto maxIt = std::max_element(psi.begin(), psi.end());
148 size_t maxIndex = std::distance(psi.begin(), maxIt);
149 double kmax = k[maxIndex];
153 if (k[itr.
rank()] == kmax){
155 Gmin = shiftToMinimum(G, meshDimensions, unitCell);
158 Log::file() <<
" Gmin: " << Gmin<< std::endl;
159 Log::file() <<
" prefactor: " << prefactor_[itr.
rank()]<< std::endl;
171 if (simulator().hasRamp() && nSamplePerOutput() == 1) {
172 double chi= system().interaction().chi(0,1);
175 outputFile_ <<
Int(step);
176 outputFile_ <<
Dbl(chi);
177 outputFile_ <<
Dbl(value);
187 IntVec<D> meshDimensions = system().domain().mesh().dimensions();
188 UnitCell<D> const & unitCell = system().domain().unitCell();
200 Gmin = shiftToMinimum(G, meshDimensions, unitCell);
201 GminList[itr.
rank()] = Gmin;
206 bool inverseFound =
false;
209 if (prefactor_[itr.
rank()] == 0){
210 Gmin = GminList[itr.
rank()];
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;
225 if (inverseFound ==
false){
226 prefactor_[itr.
rank()] = 1.0;
An IntVec<D, T> is a D-component vector of elements of integer type T.
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.
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 ~FourthOrderParameter()
Destructor.
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).
Main class for SCFT or PS-FTS simulation of one system.
Vec< D, T > & negate(const Vec< D, T > &v)
Return negative of vector v.
Dynamically allocatable contiguous array template.
void allocate(int capacity)
Allocate the underlying C array.
Wrapper for a double precision number, for formatted ostream output.
Wrapper for an int, for formatted ostream output.
static std::ostream & file()
Get log ostream by reference.
File containing preprocessor macros for error handling.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Periodic fields and crystallography.
PSCF package top-level namespace.
Utility classes for scientific computation.