1#ifndef RPG_BINARY_STRUCTURE_FACTOR_GRID_TPP
2#define RPG_BINARY_STRUCTURE_FACTOR_GRID_TPP
4#include "BinaryStructureFactorGrid.h"
6#include <rpg/fts/simulator/Simulator.h>
8#include <prdc/crystal/shiftToMinimum.h>
9#include <prdc/cuda/resources.h>
10#include <prdc/cuda/complex.h>
11#include <pscf/mesh/MeshIterator.h>
12#include <util/misc/FileMaster.h>
13#include <util/misc/ioUtil.h>
14#include <util/format/Dbl.h>
15#include <util/format/Int.h>
20#include <unordered_map>
36 simulatorPtr_(&simulator),
37 systemPtr_(&(simulator.system())),
38 isInitialized_(false),
49 readOutputFileName(in);
50 readOptional(in,
"nSamplePerBlock", nSamplePerBlock_);
60 const int nMonomer = system().mixture().nMonomer();
62 UTIL_THROW(
"The BinaryStructureFactorGrid Analyzer is designed specifically for diblock copolymer system. Please verify the number of monomer types in your system.");
66 IntVec<D> const & dimensions = system().mesh().dimensions();
68 wm_.allocate(dimensions);
69 wk_.allocate(dimensions);
73 for (
int i = 0; i < D; ++i) {
75 kMeshDimensions_[i] = dimensions[i];
77 kMeshDimensions_[i] = dimensions[i]/2 + 1;
81 for(
int i = 0; i < D; ++i) {
82 kSize_ *= kMeshDimensions_[i];
86 structureFactors_.allocate(nWave_);
87 accumulators_.allocate(nWave_);
90 qList_.resize(kSize_);
91 RField<D> const & kSq = system().domain().waveList().kSq();
97 qList_[itr.
rank()] = sqrt(
double(kSqHost[itr.
rank()]));
100 isInitialized_ =
true;
103 for (
int i = 0; i < nWave_; ++i) {
104 structureFactors_[i] = 0.0;
105 accumulators_[i].setNSamplePerBlock(nSamplePerBlock_);
106 accumulators_[i].clear();
109 if (!isInitialized_) {
110 UTIL_THROW(
"Error: object is not initialized");
122 if (isAtInterval(iStep)) {
123 IntVec<D> const & dimensions = system().mesh().dimensions();
127 system().w().rgrid(1), -0.5);
130 system().fft().forwardTransform(wm_, wk_);
134 for (
int k = 0; k < wk_.capacity(); k++) {
144 const double vSystem = system().domain().unitCell().volume();
145 const double vMonomer = system().mixture().vMonomer();
146 double n = vSystem / vMonomer;
147 double chi= system().interaction().chi(0,1);
152 structureFactors_[itr.
rank()] = n / (chi * chi) * accumulators_[itr.
rank()].average() - 1.0/(2.0*chi);
155 structureFactors_[itr.
rank()] /= vMonomer;
163 UTIL_CHECK(qList_.capacity() == structureFactors_.capacity());
165 std::map<double, double> SMap;
166 for (
int i = 0; i < qList_.capacity(); ++i) {
167 double q = qList_[i];
168 double s = structureFactors_[i];
173 for (
auto& i : SMap) {
175 double sum = i.second;
176 int count = std::count(qList_.begin(), qList_.end(), q);
177 i.second = sum / count;
181 double tolerance = 1e-5;
182 auto it = SMap.begin();
183 double currentq = it->first;
184 double sumS = it->second;
186 for (++it; it != SMap.end(); ++it) {
187 double key = it->first;
188 double s = it->second;
189 if (std::abs(key - currentq) <= tolerance) {
193 averageSMap_[currentq] = sumS / count;
201 averageSMap_[currentq] = sumS / count;
213 computeStructureFactor();
214 averageStructureFactor();
217 system().fileMaster().openOutputFile(outputFileName(), outputFile_);
218 outputFile_ <<
"\t" <<
"q";
219 outputFile_ <<
"\t" <<
"\t" <<
"S(q)/(\u03C1 N)";
220 outputFile_<< std::endl;
221 for (
const auto& i : averageSMap_) {
223 double averageS = i.second;
224 outputFile_ <<
Dbl(q, 18, 8);
225 outputFile_ <<
Dbl(averageS, 18, 8);
226 outputFile_ << std::endl;
Template for dynamic array stored in host CPU memory.
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.
Field of real double precision values on an FFT mesh.
Abstract base for periodic output and/or analysis actions.
void setup()
Clear accumulators.
void sample(long iStep)
Add particles to BinaryStructureFactor accumulators.
void setClassName(const char *className)
Set class name string.
void averageStructureFactor()
Compute average S(k) over k of equal magnitude.
void computeStructureFactor()
Compute structure factor.
void output()
Output results to predefined output file.
void readParameters(std::istream &in)
Read parameters from file.
BinaryStructureFactorGrid(Simulator< D > &simulator, System< D > &system)
Constructor.
Field theoretic simulator (base class).
Main class for calculations that represent one system.
Wrapper for a double precision number, for formatted ostream output.
#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.
RT absSq(CT const &a)
Return square of absolute magnitude of a complex number.
void addVcVc(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, cudaReal const c, DeviceArray< cudaReal > const &d, cudaReal const e)
Vector addition w/ coefficient, a[i] = (b[i]*c) + (d[i]*e), kernel wrapper.
Fields, FFTs, and utilities for periodic boundary conditions (CUDA)
Periodic fields and crystallography.
PSCF package top-level namespace.
Utility classes for scientific computation.