PSCF v1.2
rpg/fts/analyzer/BinaryStructureFactorGrid.tpp
1#ifndef RPG_BINARY_STRUCTURE_FACTOR_GRID_TPP
2#define RPG_BINARY_STRUCTURE_FACTOR_GRID_TPP
3
4#include "BinaryStructureFactorGrid.h"
5
6#include <rpg/fts/simulator/Simulator.h>
7#include <rpg/System.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>
16
17#include <fftw3.h>
18
19#include <vector>
20#include <unordered_map>
21#include <algorithm>
22
23namespace Pscf {
24namespace Rpg {
25
26 using namespace Util;
27 using namespace Pscf::Prdc;
28 using namespace Pscf::Prdc::Cuda;
29
30 /*
31 * Constructor.
32 */
33 template <int D>
35 : Analyzer<D>(),
36 simulatorPtr_(&simulator),
37 systemPtr_(&(simulator.system())),
38 isInitialized_(false),
39 nSamplePerBlock_(1)
40 { setClassName("BinaryStructureFactorGrid"); }
41
42 /*
43 * Read parameters from file, and allocate memory.
44 */
45 template <int D>
47 {
48 readInterval(in);
49 readOutputFileName(in);
50 readOptional(in,"nSamplePerBlock", nSamplePerBlock_);
51 }
52
53 /*
54 * BinaryStructureFactorGrid setup
55 */
56 template <int D>
58 {
59 //Check if the system is AB diblock copolymer
60 const int nMonomer = system().mixture().nMonomer();
61 if (nMonomer != 2) {
62 UTIL_THROW("The BinaryStructureFactorGrid Analyzer is designed specifically for diblock copolymer system. Please verify the number of monomer types in your system.");
63 }
64
65 //Allocate variables
66 IntVec<D> const & dimensions = system().mesh().dimensions();
67 if (!isInitialized_){
68 wm_.allocate(dimensions);
69 wk_.allocate(dimensions);
70 }
71
72 // Compute Fourier space kMeshDimensions_
73 for (int i = 0; i < D; ++i) {
74 if (i < D - 1) {
75 kMeshDimensions_[i] = dimensions[i];
76 } else {
77 kMeshDimensions_[i] = dimensions[i]/2 + 1;
78 }
79 }
80 kSize_ = 1;
81 for(int i = 0; i < D; ++i) {
82 kSize_ *= kMeshDimensions_[i];
83 }
84
85 nWave_ = kSize_;
86 structureFactors_.allocate(nWave_);
87 accumulators_.allocate(nWave_);
88
89 // Convert real grid to KGrid format
90 qList_.resize(kSize_);
91 RField<D> const & kSq = system().domain().waveList().kSq();
92 HostDArray<cudaReal> kSqHost(kSize_);
93 kSqHost = kSq;
95 itr.setDimensions(kMeshDimensions_);
96 for (itr.begin(); !itr.atEnd(); ++itr){
97 qList_[itr.rank()] = sqrt(double(kSqHost[itr.rank()]));
98 }
99
100 isInitialized_ = true;
101
102 // Clear accumulators
103 for (int i = 0; i < nWave_; ++i) {
104 structureFactors_[i] = 0.0;
105 accumulators_[i].setNSamplePerBlock(nSamplePerBlock_);
106 accumulators_[i].clear();
107 }
108
109 if (!isInitialized_) {
110 UTIL_THROW("Error: object is not initialized");
111 }
112 }
113
114 /*
115 * Increment structure factors for all wavevectors and modes.
116 */
117 template <int D>
119 {
120 UTIL_CHECK(system().w().hasData());
121
122 if (isAtInterval(iStep)) {
123 IntVec<D> const & dimensions = system().mesh().dimensions();
124
125 // Compute W: (rgrid(0) - rgrid(1)) / 2
126 VecOp::addVcVc(wm_, system().w().rgrid(0), 0.5,
127 system().w().rgrid(1), -0.5);
128
129 // Convert real grid to KGrid format
130 system().fft().forwardTransform(wm_, wk_);
131
132 HostDArray<cudaComplex> wkCpu(kSize_);
133 wkCpu = wk_; // copy from device to host
134 for (int k = 0; k < wk_.capacity(); k++) {
135 accumulators_[k].sample(absSq<cudaComplex, cudaReal>(wkCpu[k]));
136 }
137 }
138
139 }
140
141 template <int D>
143 {
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);
148 MeshIterator<D> itr;
149 itr.setDimensions(kMeshDimensions_);
150 for (itr.begin(); !itr.atEnd(); ++itr) {
151 // Compute vS(q)
152 structureFactors_[itr.rank()] = n / (chi * chi) * accumulators_[itr.rank()].average() - 1.0/(2.0*chi);
153
154 // Compute S(q)
155 structureFactors_[itr.rank()] /= vMonomer;
156 }
157 }
158
159 // Average S(k) over k of equal magnitude
160 template <int D>
162 {
163 UTIL_CHECK(qList_.capacity() == structureFactors_.capacity());
164
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];
169 SMap[q] += s;
170 }
171
172 // Average structure factor with same magnitude value of q
173 for (auto& i : SMap) {
174 double q = i.first;
175 double sum = i.second;
176 int count = std::count(qList_.begin(), qList_.end(), q);
177 i.second = sum / count;
178 }
179
180 // Average structure factor for q in range of +-tolerance
181 double tolerance = 1e-5;
182 auto it = SMap.begin();
183 double currentq = it->first;
184 double sumS = it->second;
185 int count = 1;
186 for (++it; it != SMap.end(); ++it) {
187 double key = it->first;
188 double s = it->second;
189 if (std::abs(key - currentq) <= tolerance) {
190 sumS += s;
191 count++;
192 } else {
193 averageSMap_[currentq] = sumS / count;
194 currentq = key;
195 sumS = s;
196 count = 1;
197 }
198 }
199
200 // Add last batch of data (last number within tolerance)
201 averageSMap_[currentq] = sumS / count;
202 }
203
204
205
206
207 /*
208 * Output final results to output file.
209 */
210 template <int D>
212 {
213 computeStructureFactor();
214 averageStructureFactor();
215
216 // Output structure factors to one file
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_) {
222 double q = i.first;
223 double averageS = i.second;
224 outputFile_ << Dbl(q, 18, 8);
225 outputFile_ << Dbl(averageS, 18, 8);
226 outputFile_ << std::endl;
227 }
228 outputFile_.close();
229 }
230
231}
232}
233#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)?
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 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 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).
Definition rpg/System.h:41
Main class for calculations that represent one system.
Definition rpg/System.h:107
Wrapper for a double precision number, for formatted ostream output.
Definition Dbl.h:40
#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
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.
Definition VecOpMisc.cu:304
Fields, FFTs, and utilities for periodic boundary conditions (CUDA)
Definition CField.cu:12
Periodic fields and crystallography.
Definition CField.cpp:11
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.