PSCF v1.2
rpg/System.h
1#ifndef RPG_SYSTEM_H
2#define RPG_SYSTEM_H
3
4/*
5* PSCF - Polymer Self-Consistent Field Theory
6*
7* Copyright 2016 - 2022, The Regents of the University of Minnesota
8* Distributed under the terms of the GNU General Public License.
9*/
10
11// Header file includes
12#include <util/param/ParamComposite.h> // base class
13#include <rpg/solvers/Mixture.h> // member
14#include <rpg/field/Domain.h> // member
15#include <rpg/field/FieldIo.h> // member
16#include <rpg/field/WFieldContainer.h> // member
17#include <rpg/field/CFieldContainer.h> // member
18#include <rpg/field/Mask.h> // member
19#include <prdc/cuda/RField.h> // member (tmpFieldsRGrid_)
20#include <prdc/cuda/RFieldDft.h> // member (tmpFieldsKGrid_)
21#include <pscf/homogeneous/Mixture.h> // member
22#include <util/misc/FileMaster.h> // member
23#include <util/containers/DArray.h> // member (tmpFields ...)
24//#include <util/containers/FSArray.h> // ????
25
26// Forward references
27namespace Util {
28 template <typename T, int N> class FSArray;
29}
30namespace Pscf {
31 class Interaction;
32 template <typename Data> class DeviceArray;
33 namespace Prdc {
34 template <int D> class UnitCell;
35 }
36 namespace Rpg {
37 template <int D> class Iterator;
38 template <int D> class IteratorFactory;
39 template <int D> class Sweep;
40 template <int D> class SweepFactory;
41 template <int D> class Simulator;
42 template <int D> class SimulatorFactory;
43 }
44}
45
46
47namespace Pscf {
48namespace Rpg {
49
50 // Namespaces that are implicitly available, without qualification
51 using namespace Util;
52 using namespace Pscf::Prdc;
53 using namespace Pscf::Prdc::Cuda;
54
105 template <int D>
106 class System : public ParamComposite
107 {
108
109 public:
110
113
117 System();
118
122 ~System();
123
127
139 void setOptions(int argc, char **argv);
140
146 virtual void readParam(std::istream& in);
147
154 void readParam();
155
161 virtual void readParameters(std::istream& in);
162
168 void readCommands(std::istream& in);
169
176 void readCommands();
177
181
204 void readWBasis(const std::string & filename);
205
226 void readWRGrid(const std::string & filename);
227
242 void setWBasis(DArray< DArray<double> > const & fields);
243
255 void setWRGrid(DArray< RField<D> > const & fields);
256
269 void setWRGrid(DeviceArray<cudaReal> & fields);
270
277 void symmetrizeWFields();
278
292 void estimateWfromC(const std::string& filename);
293
297
313 void setUnitCell(UnitCell<D> const & unitCell);
314
325 void setUnitCell(typename UnitCell<D>::LatticeSystem lattice,
326 FSArray<double, 6> const & parameters);
327
336 void setUnitCell(FSArray<double, 6> const & parameters);
337
341
366 void compute(bool needStress = false);
367
391 int iterate(bool isContinuation = false);
392
405 void sweep();
406
422 void simulate(int nStep);
423
427
438 void computeFreeEnergy();
439
446 double fHelmholtz() const;
447
454 double pressure() const;
455
459
468 void writeParamNoSweep(std::ostream& out) const;
469
486 void writeThermo(std::ostream& out);
487
501 void writeStress(std::ostream& out);
502
506
512 void writeWBasis(const std::string & filename);
513
519 void writeWRGrid(const std::string & filename) const;
520
526 void writeCBasis(const std::string & filename);
527
533 void writeCRGrid(const std::string & filename) const;
534
545 void writeBlockCRGrid(const std::string & filename) const;
546
550
560 void writeQSlice(std::string const & filename,
561 int polymerId, int blockId,
562 int directionId, int segmentId) const;
563
572 void writeQTail(std::string const & filename, int polymerId,
573 int blockId, int directionId) const;
574
583 void writeQ(std::string const & filename, int polymerId,
584 int blockId, int directionId) const;
585
604 void writeQAll(std::string const & basename);
605
609
618 void writeStars(const std::string & filename) const;
619
628 void writeWaves(const std::string & filename) const;
629
635 void writeGroup(std::string const & filename) const;
636
640
651 void basisToRGrid(const std::string & inFileName,
652 const std::string & outFileName);
653
667 void rGridToBasis(const std::string & inFileName,
668 const std::string & outFileName);
669
676 void kGridToRGrid(const std::string& inFileName,
677 const std::string& outFileName);
678
685 void rGridToKGrid(const std::string & inFileName,
686 const std::string & outFileName);
687
701 void kGridToBasis(const std::string& inFileName,
702 const std::string& outFileName);
703
710 void basisToKGrid(const std::string & inFileName,
711 const std::string & outFileName);
712
722 void compare(const DArray< DArray<double> > field1,
723 const DArray< DArray<double> > field2);
724
734 void compare(const DArray< RField<D> > field1,
735 const DArray< RField<D> > field2);
736
744 bool checkRGridFieldSymmetry(const std::string & inFileName,
745 double epsilon = 1.0E-8);
746
754 void scaleFieldsBasis(const std::string & inFileName,
755 const std::string & outFileName,
756 double factor);
757
765 void scaleFieldsRGrid(const std::string & inFileName,
766 const std::string & outFileName,
767 double factor) const;
768
788 void expandRGridDimension(const std::string & inFileName,
789 const std::string & outFileName,
790 int d,
791 DArray<int> newGridDimensions);
805 void replicateUnitCell(const std::string & inFileName,
806 const std::string & outFileName,
807 IntVec<D> const & replicas);
808
812
818 void writeTimers(std::ostream& out);
819
823 void clearTimers();
824
828
832 WFieldContainer<D> const & w() const;
833
837 CFieldContainer<D> const & c() const;
838
843
847 WFieldContainer<D> const & h() const;
848
852 Mask<D>& mask();
853
857 Mask<D> const & mask() const;
858
862
867
871 Mixture<D> const & mixture() const;
872
877
881 Interaction const & interaction() const;
882
886 Domain<D> & domain();
887
891 Domain<D> const & domain() const;
892
897
901 Iterator<D> const & iterator() const;
902
907
911 UnitCell<D> const & unitCell() const;
912
916 Mesh<D> const & mesh() const;
917
921 Basis<D> const & basis() const;
922
926 FieldIo<D> const & fieldIo() const;
927
931 FFT<D> const & fft() const ;
932
937
942
946
950 bool hasIterator() const;
951
955 bool hasSweep() const;
956
960 bool hasSimulator() const;
961
965 bool hasExternalFields() const;
966
970 bool hasMask() const;
971
975 bool hasCFields() const;
976
980 bool hasFreeEnergy() const;
981
983
984 private:
985
989 Mixture<D> mixture_;
990
994 Domain<D> domain_;
995
999 FileMaster fileMaster_;
1000
1004 Homogeneous::Mixture homogeneous_;
1005
1009 Interaction* interactionPtr_;
1010
1014 Iterator<D>* iteratorPtr_;
1015
1019 IteratorFactory<D>* iteratorFactoryPtr_;
1020
1024 Sweep<D>* sweepPtr_;
1025
1029 SweepFactory<D>* sweepFactoryPtr_;
1030
1034 Simulator<D>* simulatorPtr_;
1035
1039 SimulatorFactory<D>* simulatorFactoryPtr_;
1040
1045
1050
1055
1059 Mask<D> mask_;
1060
1066 mutable DArray< DArray<double> > tmpFieldsBasis_;
1067
1073 mutable DArray< RField<D> > tmpFieldsRGrid_;
1074
1080 mutable DArray<RFieldDft<D> > tmpFieldsKGrid_;
1081
1085 double fHelmholtz_;
1086
1094 double fIdeal_;
1095
1099 double fInter_;
1100
1104 double fExt_;
1105
1112 double pressure_;
1113
1117 bool isAllocatedGrid_;
1118
1122 bool isAllocatedBasis_;
1123
1127 bool hasMixture_;
1128
1132 bool hasCFields_;
1133
1137 bool hasFreeEnergy_;
1138
1142 IntVec<D> kMeshDimensions_;
1143
1147 RField<D> workArray_;
1148
1149 // Private member functions
1150
1156 void allocateFieldsGrid();
1157
1163 void allocateFieldsBasis();
1164
1170 void readFieldHeader(std::string filename);
1171
1180 void readEcho(std::istream& in, std::string& string) const;
1181
1190 void readEcho(std::istream& in, double& value) const;
1191
1195 void initHomogeneous();
1196
1197 };
1198
1199 // Inline member functions
1200
1201 // Get the associated Mixture<D> object.
1202 template <int D>
1204 { return mixture_; }
1205
1206 // Get the associated Mixture<D> object by const reference.
1207 template <int D>
1208 inline Mixture<D> const & System<D>::mixture() const
1209 { return mixture_; }
1210
1211 // Get the Interaction (excess free energy) by non-const reference.
1212 template <int D>
1214 {
1215 UTIL_ASSERT(interactionPtr_);
1216 return *interactionPtr_;
1217 }
1218
1219 // Get the Interaction (excess free energy ) by const reference.
1220 template <int D>
1222 {
1223 UTIL_ASSERT(interactionPtr_);
1224 return *interactionPtr_;
1225 }
1226
1227 // Get the Domain by non-const reference.
1228 template <int D>
1230 { return domain_; }
1231
1232 // Get the Domain by const reference.
1233 template <int D>
1234 inline Domain<D> const & System<D>::domain() const
1235 { return domain_; }
1236
1237 template <int D>
1238 inline UnitCell<D> const & System<D>::unitCell() const
1239 { return domain_.unitCell(); }
1240
1241 // Get the Mesh<D> by const reference.
1242 template <int D>
1243 inline Mesh<D> const & System<D>::mesh() const
1244 { return domain_.mesh(); }
1245
1246 // Get the Basis<D> by const reference.
1247 template <int D>
1248 inline Basis<D> const & System<D>::basis() const
1249 { return domain_.basis(); }
1250
1251 // Get the FFT<D> object by const reference.
1252 template <int D>
1253 inline FFT<D> const & System<D>::fft() const
1254 { return domain_.fft(); }
1255
1256 // Get the const FieldIo<D> object by const reference.
1257 template <int D>
1258 inline FieldIo<D> const & System<D>::fieldIo() const
1259 { return domain_.fieldIo(); }
1260
1261 // Get the Iterator by non-const reference.
1262 template <int D>
1264 {
1265 UTIL_ASSERT(iteratorPtr_);
1266 return *iteratorPtr_;
1267 }
1268
1269 // Get the Iterator by const reference.
1270 template <int D>
1271 inline Iterator<D> const & System<D>::iterator() const
1272 {
1273 UTIL_ASSERT(iteratorPtr_);
1274 return *iteratorPtr_;
1275 }
1276
1277 // Get the Simulator by non-const reference.
1278 template <int D>
1280 {
1281 UTIL_ASSERT(simulatorPtr_);
1282 return *simulatorPtr_;
1283 }
1284
1285 // Get the FileMaster.
1286 template <int D>
1288 { return fileMaster_; }
1289
1290 // Get the Homogeneous::Mixture object.
1291 template <int D>
1292 inline
1294 { return homogeneous_; }
1295
1296 // Get container of w fields (const reference)
1297 template <int D>
1298 inline
1300 { return w_; }
1301
1302 // Get container of c fields (const reference)
1303 template <int D>
1304 inline
1306 { return c_; }
1307
1308 // Get container of external potential fields (reference)
1309 template <int D>
1311 { return h_; }
1312
1313 // Get container of external potential fields (const reference)
1314 template <int D>
1315 inline WFieldContainer<D> const & System<D>::h() const
1316 { return h_; }
1317
1318 // Get mask field (reference)
1319 template <int D>
1321 { return mask_; }
1322
1323 // Get mask field (const reference)
1324 template <int D>
1325 inline Mask<D> const & System<D>::mask() const
1326 { return mask_; }
1327
1328 // Get precomputed Helmoltz free energy per monomer / kT.
1329 template <int D>
1330 inline double System<D>::fHelmholtz() const
1331 {
1332 UTIL_CHECK(hasFreeEnergy_);
1333 return fHelmholtz_;
1334 }
1335
1336 // Get precomputed pressure (units of kT / monomer volume).
1337 template <int D>
1338 inline double System<D>::pressure() const
1339 {
1340 UTIL_CHECK(hasFreeEnergy_);
1341 return pressure_;
1342 }
1343
1344 // Has the free energy been computed for the current w fields?
1345 template <int D>
1346 inline bool System<D>::hasFreeEnergy() const
1347 { return hasFreeEnergy_; }
1348
1349 // Have the c fields been computed for the current w fields?
1350 template <int D>
1351 inline bool System<D>::hasCFields() const
1352 { return hasCFields_; }
1353
1354 // Does the system have an Iterator object?
1355 template <int D>
1356 inline bool System<D>::hasIterator() const
1357 { return (iteratorPtr_); }
1358
1359 // Does this system have an associated Sweep object?
1360 template <int D>
1361 inline bool System<D>::hasSweep() const
1362 { return (sweepPtr_); }
1363
1364 // Does the system have an associated Simulator ?
1365 template <int D>
1366 inline bool System<D>::hasSimulator() const
1367 { return (simulatorPtr_); }
1368
1369 // Does this system have external potential fields?
1370 template <int D>
1372 { return h_.hasData(); }
1373
1374 // Does this system have a mask?
1375 template <int D>
1376 inline bool System<D>::hasMask() const
1377 { return mask_.hasData(); }
1378
1379 #ifndef RPG_SYSTEM_TPP
1380 // Suppress implicit instantiation
1381 extern template class System<1>;
1382 extern template class System<2>;
1383 extern template class System<3>;
1384 #endif
1385
1386} // namespace Rpg
1387} // namespace Pscf
1388#endif
Dynamic array on the GPU device with aligned data.
Definition rpg/System.h:32
A spatially homogeneous mixture.
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Flory-Huggins excess free energy model.
Definition Interaction.h:26
Description of a regular grid of points in a periodic domain.
Symmetry-adapted Fourier basis for pseudo-spectral scft.
Definition fieldIoUtil.h:21
Fourier transform wrapper.
Field of real double precision values on an FFT mesh.
Base template for UnitCell<D> classes, D=1, 2 or 3.
Definition rpg/System.h:34
A list of c fields stored in both basis and r-grid format.
Spatial domain and spatial discretization for a periodic structure.
File input/output operations and format conversions for fields.
Factory for subclasses of Iterator.
Definition rpg/System.h:38
Base class for iterative solvers for SCF equations.
Definition rpg/System.h:37
A field to which the total monomer concentration is constrained.
Solver for a mixture of polymers and solvents.
Factory for subclasses of Simulator.
Definition rpg/System.h:42
Field theoretic simulator (base class).
Definition rpg/System.h:41
Default Factory for subclasses of Sweep.
Definition rpg/System.h:40
Solve a sequence of problems along a line in parameter space.
Definition rpg/System.h:39
Main class for calculations that represent one system.
Definition rpg/System.h:107
void expandRGridDimension(const std::string &inFileName, const std::string &outFileName, int d, DArray< int > newGridDimensions)
Expand the number of spatial dimensions of an r-grid field.
bool hasExternalFields() const
Does this system have external potential fields?
void kGridToBasis(const std::string &inFileName, const std::string &outFileName)
Convert fields from Fourier (k-grid) to symmetrized basis format.
void writeCBasis(const std::string &filename)
Write concentrations in symmetry-adapted basis format.
void estimateWfromC(const std::string &filename)
Construct trial w-fields from c-fields.
void writeQTail(std::string const &filename, int polymerId, int blockId, int directionId) const
Write the final slice of a propagator in r-grid format.
Simulator< D > & simulator()
Get Simulator for field theoretic simulation.
bool hasMask() const
Does this system have a mask (inhomogeneous density constraint) ?
void readParam()
Read input parameters from default param file.
void computeFreeEnergy()
Compute free energy density and pressure for current fields.
void writeBlockCRGrid(const std::string &filename) const
Write c fields for all blocks and solvents in r-grid format.
void compute(bool needStress=false)
Solve the modified diffusion equation once, without iteration.
void setWRGrid(DArray< RField< D > > const &fields)
Set new w fields, in real-space (r-grid) format.
void writeQ(std::string const &filename, int polymerId, int blockId, int directionId) const
Write one propagator for one block, in r-grid format.
void writeTimers(std::ostream &out)
Write timer file to an ostream.
Mixture< D > & mixture()
Get Mixture by reference.
void rGridToKGrid(const std::string &inFileName, const std::string &outFileName)
Convert fields from real-space (r-grid) to Fourier (k-grid) format.
void rGridToBasis(const std::string &inFileName, const std::string &outFileName)
Convert a field from real-space grid to symmetrized basis format.
void setWBasis(DArray< DArray< double > > const &fields)
Set chemical potential fields, in symmetry-adapted basis format.
void kGridToRGrid(const std::string &inFileName, const std::string &outFileName)
Convert fields from Fourier (k-grid) to real-space (r-grid) format.
FieldIo< D > const & fieldIo() const
Get the FieldIo by const reference.
void writeParamNoSweep(std::ostream &out) const
Write parameter file to an ostream, omitting the sweep block.
void readCommands()
Read and process commands from the default command file.
void scaleFieldsRGrid(const std::string &inFileName, const std::string &outFileName, double factor) const
Multiply all elements of an array of r-grid fields by a scalar.
void writeWBasis(const std::string &filename)
Write chemical potential fields in symmetry adapted basis format.
void writeGroup(std::string const &filename) const
Output all elements of the space group.
Homogeneous::Mixture & homogeneous()
Get homogeneous mixture (for reference calculations).
Mesh< D > const & mesh() const
Get spatial discretization Mesh by const reference.
void writeWRGrid(const std::string &filename) const
Write chemical potential fields in real space grid (r-grid) format.
Domain< D > & domain()
Get Domain by non const reference.
void symmetrizeWFields()
Symmetrize r-grid w-fields, compute basis components.
void readWBasis(const std::string &filename)
Read chemical potential fields in symmetry adapted basis format.
void writeStars(const std::string &filename) const
Output information about stars and symmetrized basis functions.
Iterator< D > & iterator()
Get the iterator by non-const reference.
virtual void readParameters(std::istream &in)
Read body of parameter file (without opening, closing lines).
Mask< D > & mask()
Get the mask (field to which total density is constrained).
void sweep()
Sweep in parameter space, solving an SCF problem at each point.
bool hasFreeEnergy() const
Has the free energy been computed from the current w fields?
void basisToKGrid(const std::string &inFileName, const std::string &outFileName)
Convert fields from symmetrized basis to Fourier (k-grid) format.
double fHelmholtz() const
Get precomputed Helmoltz free energy per monomer / kT.
~System()
Destructor.
bool hasIterator() const
Does this system have an Iterator object?
int iterate(bool isContinuation=false)
Iteratively solve a SCFT problem.
void basisToRGrid(const std::string &inFileName, const std::string &outFileName)
Convert a field from symmetry-adapted basis to r-grid format.
double pressure() const
Get precomputed pressure times monomer volume / kT.
CFieldContainer< D > const & c() const
Get container of monomer concentration fields (c fields).
System()
Constructor.
void writeStress(std::ostream &out)
Write stress properties to a file.
void readWRGrid(const std::string &filename)
Read chemical potential fields in real space grid (r-grid) format.
void setOptions(int argc, char **argv)
Process command line options.
bool hasCFields() const
Have c fields been computed from the current w fields ?
bool hasSimulator() const
Does this system have an initialized Simulator?
void clearTimers()
Clear timers.
void writeWaves(const std::string &filename) const
Output information about waves.
void replicateUnitCell(const std::string &inFileName, const std::string &outFileName, IntVec< D > const &replicas)
Replicate the crystal unit cell to create a larger cell.
void writeQSlice(std::string const &filename, int polymerId, int blockId, int directionId, int segmentId) const
Write specified slice of a propagator at fixed s in r-grid format.
UnitCell< D > const & unitCell() const
Get crystal UnitCell by const reference.
void writeCRGrid(const std::string &filename) const
Write concentration fields in real space grid (r-grid) format.
FFT< D > const & fft() const
Get the FFT object by const reference.
void writeQAll(std::string const &basename)
Write all propagators of all blocks, each to a separate file.
void compare(const DArray< DArray< double > > field1, const DArray< DArray< double > > field2)
Compare arrays of fields in basis format, output a report.
Interaction & interaction()
Get interaction (i.e., excess free energy) by reference.
bool hasSweep() const
Does this system have an associated Sweep object?
bool checkRGridFieldSymmetry(const std::string &inFileName, double epsilon=1.0E-8)
Check if r-grid fields have the declared space group symmetry.
void setUnitCell(UnitCell< D > const &unitCell)
Set parameters of the associated unit cell.
FileMaster & fileMaster()
Get FileMaster by reference.
void writeThermo(std::ostream &out)
Write thermodynamic properties to a file.
WFieldContainer< D > & h()
Get container of external potential fields (non-const reference).
void simulate(int nStep)
Perform a field theoretic simulation.
Basis< D > const & basis() const
Get the Basis by reference.
WFieldContainer< D > const & w() const
Get container of chemical potential fields (w fields).
void scaleFieldsBasis(const std::string &inFileName, const std::string &outFileName, double factor)
Multiply all components of an array of basis fields by a scalar.
A list of fields stored in both basis and r-grid format.
Dynamically allocatable contiguous array template.
A fixed capacity (static) contiguous array with a variable logical size.
Definition rpg/System.h:28
A FileMaster manages input and output files for a simulation.
Definition FileMaster.h:143
An object that can read multiple parameters from file.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
#define UTIL_ASSERT(condition)
Assertion macro suitable for debugging serial or parallel code.
Definition global.h:75
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.