PSCF v1.2
rpc/fts/analyzer/BinaryStructureFactorGrid.tpp
1#ifndef RPC_BINARY_STRUCTURE_FACTOR_GRID_TPP
2#define RPC_BINARY_STRUCTURE_FACTOR_GRID_TPP
3
4#include "BinaryStructureFactorGrid.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#include <pscf/math/RealVec.h>
15
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 <algorithm>
29
30//#include <unordered_map>
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 Simulator<D>& simulator,
44 System<D>& system)
45 : Analyzer<D>(),
46 simulatorPtr_(&simulator),
47 systemPtr_(&(simulator.system())),
48 isInitialized_(false),
49 nSamplePerBlock_(1),
50 kMeshDimensions_(0),
51 kSize_(0)
52 { setClassName("BinaryStructureFactorGrid"); }
53
54
55 /*
56 * Read parameters from file, and allocate memory.
57 */
58 template <int D>
60 {
61 // Precondition: Require that the system has two monomer types
62 UTIL_CHECK(2 == system().mixture().nMonomer());
63
64 readInterval(in);
65 readOutputFileName(in);
66 readOptional(in,"nSamplePerBlock", nSamplePerBlock_);
67 }
68
69 /*
70 * BinaryStructureFactorGrid setup
71 */
72 template <int D>
74 {
75 // Check if the system has two monomer types
76 const int nMonomer = system().mixture().nMonomer();
77 if (nMonomer != 2) {
78 UTIL_THROW("nMonomer != 2 in BinaryStructureFactorGrid");
79 }
80
81 //Allocate variables
82 IntVec<D> const & dimensions = system().domain().mesh().dimensions();
83
84 // Compute Fourier space kMeshDimensions_ and kSize_
85 kSize_ = 1;
86 for (int i = 0; i < D; ++i) {
87 if (i < D - 1) {
88 kMeshDimensions_[i] = dimensions[i];
89 } else {
90 kMeshDimensions_[i] = dimensions[i]/2 + 1;
91 }
92 kSize_ *= kMeshDimensions_[i];
93 }
94
95 if (!isInitialized_){
96 wKGrid_.allocate(nMonomer);
97 wm_.allocate(dimensions);
98 wk_.allocate(dimensions);
99 for (int i = 0; i < nMonomer; ++i) {
100 wKGrid_[i].allocate(dimensions);
101 }
102
103 // Compute qList
104 qList_.resize(kSize_);
105 IntVec<D> G;
106 IntVec<D> Gmin;
107 UnitCell<D> const & unitCell = system().domain().unitCell();
108 MeshIterator<D> itr(kMeshDimensions_);
109 double Gsq;
110 for (itr.begin(); !itr.atEnd(); ++itr){
111 // Obtain square magnitude of reciprocal basis vector
112 G = itr.position();
113 Gmin = shiftToMinimum(G, dimensions, unitCell);
114 Gsq = unitCell.ksq(Gmin);
115 qList_[itr.rank()] = sqrt(Gsq);
116 }
117
118 }
119
120 nWave_ = kSize_;
121 structureFactors_.allocate(nWave_);
122 accumulators_.allocate(nWave_);
123 isInitialized_ = true;
124
125 // Clear accumulators
126 for (int i = 0; i < nWave_; ++i) {
127 structureFactors_[i] = 0.0;
128 accumulators_[i].setNSamplePerBlock(nSamplePerBlock_);
129 accumulators_[i].clear();
130 }
131 if (!isInitialized_) {
132 UTIL_THROW("Error: object is not initialized");
133 }
134 }
135
136 /*
137 * Increment structure factors for all wavevectors and modes.
138 */
139 template <int D>
141 {
142 UTIL_CHECK(system().w().hasData());
143
144 if (isAtInterval(iStep)) {
145
146 // Compute W-
147 for (int i = 0; i < wm_.capacity(); ++i) {
148 wm_[i] = (system().w().rgrid(0)[i] - system().w().rgrid(1)[i])/2;
149 }
150
151 // Convert real grid to KGrid format
152 system().domain().fft().forwardTransform(wm_, wk_);
153
154 // Pass square magnitudes of Fourier components to accumulators
155 for (int k=0; k< wk_.capacity(); k++) {
156 std::complex<double> wmKGrid(wk_[k][0], wk_[k][1]);
157 double squared_magnitude = std::norm(wmKGrid);
158 accumulators_[k].sample(squared_magnitude);
159 }
160
161 }
162 }
163
164 template <int D>
166 {
167 const double vSystem = system().domain().unitCell().volume();
168 const double vMonomer = system().mixture().vMonomer();
169 double n = vSystem / vMonomer;
170 double chi= system().interaction().chi(0,1);
171 MeshIterator<D> itr;
172 itr.setDimensions(wKGrid_[0].dftDimensions());
173 for (itr.begin(); !itr.atEnd(); ++itr) {
174 // Compute vS(q)
175 structureFactors_[itr.rank()] = n / (chi * chi) * accumulators_[itr.rank()].average() - 1.0/(2.0*chi);
176
177 // Compute S(q)
178 structureFactors_[itr.rank()] /= vMonomer;
179 }
180 }
181
182 // Average S(k) over k of equal magnitude
183 template <int D>
185 {
186 // Convert real grid to KGrid format
187 const int nMonomer = system().mixture().nMonomer();
188 for (int i = 0; i < nMonomer; ++i) {
189 system().domain().fft().forwardTransform(system().w().rgrid()[i],
190 wKGrid_[i]);
191 }
192
193 std::map<double, double> SMap;
194 {
195 int qListcapacity = (int)qList_.capacity();
196 double q, s;
197 UTIL_CHECK(qListcapacity == structureFactors_.capacity());
198 for (int i = 0; i < qListcapacity; ++i) {
199 q = qList_[i];
200 s = structureFactors_[i];
201 SMap[q] += s;
202 }
203 }
204
205 // Average structure factor with same magnitude value of q
206 for (auto& i : SMap) {
207 double q = i.first;
208 double sum = i.second;
209 int count = std::count(qList_.begin(), qList_.end(), q);
210 i.second = sum / count;
211 }
212
213 // Average structure factor for q in range of +-tolerance
214 double tolerance = 1e-5;
215 auto it = SMap.begin();
216 double currentq = it->first;
217 double sumS = it->second;
218 int count = 1;
219 for (++it; it != SMap.end(); ++it) {
220 double key = it->first;
221 double s = it->second;
222 if (std::abs(key - currentq) <= tolerance) {
223 sumS += s;
224 count++;
225 } else {
226 averageSMap_[currentq] = sumS / count;
227 currentq = key;
228 sumS = s;
229 count = 1;
230 }
231 }
232
233 // Add last batch of data (last number within tolerance)
234 averageSMap_[currentq] = sumS / count;
235 }
236
237 /*
238 * Output final results to output file.
239 */
240 template <int D>
242 {
243 computeStructureFactor();
244 averageStructureFactor();
245
246 // Output structure factors to one file
247 system().fileMaster().openOutputFile(outputFileName(), outputFile_);
248 outputFile_ << "\t" << "q";
249 outputFile_ << "\t" <<"\t" <<"S(q)/(\u03C1 N)";
250 outputFile_<< std::endl;
251 for (const auto& i : averageSMap_) {
252 double q = i.first;
253 double averageS = i.second;
254 outputFile_ << Dbl(q, 18, 8);
255 outputFile_ << Dbl(averageS, 18, 8);
256 outputFile_ << std::endl;
257 }
258 outputFile_.close();
259 }
260
261}
262}
263#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
Abstract base for periodic output and/or analysis actions.
void setClassName(const char *className)
Set class name string.
void readParameters(std::istream &in)
Read parameters from file.
BinaryStructureFactorGrid(Simulator< D > &simulator, System< D > &system)
Constructor.
void sample(long iStep)
Add particles to BinaryStructureFactor accumulators.
void averageStructureFactor()
Compute average S(k) over k of equal magnitude.
void output()
Output results to predefined output file.
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
Wrapper for a double precision number, for formatted ostream output.
Definition Dbl.h:40
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.