13#include <rpg/fts/simulator/Simulator.h>
14#include <rpg/fts/simulator/SimulatorFactory.h>
15#include <rpg/fts/compressor/Compressor.h>
16#include <rpg/scft/sweep/Sweep.h>
17#include <rpg/scft/sweep/SweepFactory.h>
18#include <rpg/scft/iterator/Iterator.h>
19#include <rpg/scft/iterator/IteratorFactory.h>
20#include <rpg/solvers/Polymer.h>
21#include <rpg/solvers/Solvent.h>
23#include <prdc/cuda/resources.h>
24#include <prdc/cuda/RField.h>
25#include <prdc/cuda/RFieldComparison.h>
26#include <prdc/crystal/BFieldComparison.h>
28#include <pscf/inter/Interaction.h>
29#include <pscf/math/IntVec.h>
30#include <pscf/homogeneous/Clump.h>
32#include <util/containers/FSArray.h>
33#include <util/param/BracketPolicy.h>
34#include <util/param/ParamComponent.h>
35#include <util/format/Str.h>
36#include <util/format/Int.h>
37#include <util/format/Dbl.h>
38#include <util/misc/ioUtil.h>
59 interactionPtr_(nullptr),
60 iteratorPtr_(nullptr),
61 iteratorFactoryPtr_(nullptr),
63 sweepFactoryPtr_(nullptr),
64 simulatorPtr_(nullptr),
65 simulatorFactoryPtr_(nullptr),
75 isAllocatedGrid_(false),
76 isAllocatedBasis_(false),
82 domain_.setFileMaster(fileMaster_);
83 w_.setFieldIo(domain_.fieldIo());
84 h_.setFieldIo(domain_.fieldIo());
85 mask_.setFieldIo(domain_.fieldIo());
101 if (interactionPtr_) {
102 delete interactionPtr_;
107 if (iteratorFactoryPtr_) {
108 delete iteratorFactoryPtr_;
113 if (sweepFactoryPtr_) {
114 delete sweepFactoryPtr_;
117 delete simulatorPtr_;
119 if (simulatorFactoryPtr_) {
120 delete simulatorFactoryPtr_;
147 while ((c = getopt(argc, argv,
"ed:p:c:i:o:t:")) != -1) {
177 Log::file() <<
"Unknown option -" << optopt << std::endl;
196 fileMaster_.setParamFileName(std::string(pArg));
198 UTIL_THROW(
"Missing required -p option - no parameter file");
203 fileMaster_.setCommandFileName(std::string(cArg));
205 UTIL_THROW(
"Missing required -c option - no command file");
210 fileMaster_.setInputPrefix(std::string(iArg));
215 fileMaster_.setOutputPrefix(std::string(oArg));
233 readParamComposite(in, mixture_);
236 int nm = mixture_.nMonomer();
237 int np = mixture_.nPolymer();
238 int ns = mixture_.nSolvent();
245 interaction().setNMonomer(nm);
246 readParamComposite(in, interaction());
249 readParamComposite(in, domain_);
251 UTIL_CHECK(domain_.unitCell().nParameter() > 0);
255 mixture_.associate(domain_.mesh(), fft(),
256 domain_.unitCell(), domain_.waveList());
261 allocateFieldsGrid();
264 std::string className;
267 iteratorFactoryPtr_->readObjectOptional(in, *
this, className,
270 Log::file() << indent() <<
" Iterator{ [absent] }\n";
274 if (iteratorPtr_ && !isEnd) {
276 sweepFactoryPtr_->readObjectOptional(in, *
this, className,
279 Log::file() << indent() <<
" Sweep{ [absent] }\n";
286 simulatorFactoryPtr_->readObjectOptional(in, *
this,
289 Log::file() << indent() <<
" Simulator{ [absent] }\n";
295 homogeneous_.setNMolecule(np+ns);
296 homogeneous_.setNMonomer(nm);
307 readBegin(in, className().c_str());
317 { readParam(fileMaster_.paramFile()); }
326 std::string command, filename, inFileName, outFileName;
328 bool readNext =
true;
339 if (command ==
"FINISH") {
343 if (command ==
"READ_W_BASIS") {
344 readEcho(in, filename);
345 readWBasis(filename);
347 if (command ==
"READ_W_RGRID") {
348 readEcho(in, filename);
349 readWRGrid(filename);
351 if (command ==
"ESTIMATE_W_FROM_C") {
352 readEcho(in, inFileName);
353 estimateWfromC(inFileName);
355 if (command ==
"SET_UNIT_CELL") {
358 Log::file() <<
" " << unitCell << std::endl;
359 setUnitCell(unitCell);
361 if (command ==
"COMPUTE") {
365 if (command ==
"ITERATE") {
367 bool isContinuation =
false;
368 int fail = iterate(isContinuation);
373 if (command ==
"SWEEP") {
378 if (command ==
"COMPRESS") {
381 simulator().compressor().compress();
383 if (command ==
"SIMULATE") {
390 if (command ==
"ANALYZE" || command ==
"ANALYZE_TRAJECTORY") {
397 std::string classname;
398 readEcho(in, classname);
399 readEcho(in, filename);
400 simulator().analyze(min, max, classname, filename);
402 if (command ==
"WRITE_TIMERS") {
403 readEcho(in, filename);
405 fileMaster_.openOutputFile(filename, file);
409 if (command ==
"CLEAR_TIMERS") {
412 if (command ==
"WRITE_PARAM") {
413 readEcho(in, filename);
415 fileMaster_.openOutputFile(filename, file);
416 writeParamNoSweep(file);
419 if (command ==
"WRITE_THERMO") {
420 readEcho(in, filename);
422 fileMaster_.openOutputFile(filename, file,
427 if (command ==
"WRITE_STRESS") {
428 readEcho(in, filename);
430 fileMaster_.openOutputFile(filename, file,
435 if (command ==
"WRITE_W_BASIS") {
436 readEcho(in, filename);
437 writeWBasis(filename);
439 if (command ==
"WRITE_W_RGRID") {
440 readEcho(in, filename);
441 writeWRGrid(filename);
443 if (command ==
"WRITE_C_BASIS") {
444 readEcho(in, filename);
445 writeCBasis(filename);
447 if (command ==
"WRITE_C_RGRID") {
448 readEcho(in, filename);
449 writeCRGrid(filename);
451 if (command ==
"WRITE_BLOCK_C_RGRID") {
452 readEcho(in, filename);
453 writeBlockCRGrid(filename);
455 if (command ==
"WRITE_Q_SLICE") {
456 int polymerId, blockId, directionId, segmentId;
457 readEcho(in, filename);
462 Log::file() <<
Str(
"polymer ID ", 21) << polymerId <<
"\n"
463 <<
Str(
"block ID ", 21) << blockId <<
"\n"
464 <<
Str(
"direction ID ", 21) << directionId
466 <<
Str(
"segment ID ", 21) << segmentId
468 writeQSlice(filename, polymerId, blockId, directionId,
471 if (command ==
"WRITE_Q_TAIL") {
472 int polymerId, blockId, directionId;
473 readEcho(in, filename);
477 Log::file() <<
Str(
"polymer ID ", 21) << polymerId <<
"\n"
478 <<
Str(
"block ID ", 21) << blockId <<
"\n"
479 <<
Str(
"direction ID ", 21) << directionId
481 writeQTail(filename, polymerId, blockId, directionId);
483 if (command ==
"WRITE_Q") {
484 int polymerId, blockId, directionId;
485 readEcho(in, filename);
489 Log::file() <<
Str(
"polymer ID ", 21) << polymerId <<
"\n"
490 <<
Str(
"block ID ", 21) << blockId <<
"\n"
491 <<
Str(
"direction ID ", 21) << directionId <<
"\n";
492 writeQ(filename, polymerId, blockId, directionId);
494 if (command ==
"WRITE_Q_ALL") {
495 readEcho(in, filename);
498 if (command ==
"WRITE_STARS") {
499 readEcho(in, filename);
500 writeStars(filename);
502 if (command ==
"WRITE_WAVES") {
503 readEcho(in, filename);
504 writeWaves(filename);
506 if (command ==
"WRITE_GROUP") {
507 readEcho(in, filename);
508 writeGroup(filename);
511 if (command ==
"BASIS_TO_RGRID") {
512 readEcho(in, inFileName);
513 readEcho(in, outFileName);
514 basisToRGrid(inFileName, outFileName);
516 if (command ==
"RGRID_TO_BASIS") {
517 readEcho(in, inFileName);
518 readEcho(in, outFileName);
519 rGridToBasis(inFileName, outFileName);
521 if (command ==
"KGRID_TO_RGRID") {
522 readEcho(in, inFileName);
523 readEcho(in, outFileName);
524 kGridToRGrid(inFileName, outFileName);
526 if (command ==
"RGRID_TO_KGRID") {
527 readEcho(in, inFileName);
528 readEcho(in, outFileName);
529 rGridToKGrid(inFileName, outFileName);
531 if (command ==
"KGRID_TO_BASIS") {
532 readEcho(in, inFileName);
533 readEcho(in, outFileName);
534 kGridToBasis(inFileName, outFileName);
536 if (command ==
"CHECK_RGRID_SYMMETRY") {
538 readEcho(in, inFileName);
539 readEcho(in, epsilon);
541 hasSymmetry = checkRGridFieldSymmetry(inFileName, epsilon);
544 <<
"Symmetry of r-grid file matches this space group."
545 << std::endl << std::endl;
548 <<
"Symmetry of r-grid file does not match this\n"
549 <<
"space group to within error threshold of "
550 <<
Dbl(epsilon) <<
" ." << std::endl << std::endl;
553 if (command ==
"COMPARE_BASIS") {
556 std::string filecompare1, filecompare2;
557 readEcho(in, filecompare1);
558 readEcho(in, filecompare2);
562 domain().fieldIo().readFieldsBasis(filecompare1, Bfield1,
564 domain().fieldIo().readFieldsBasis(filecompare2, Bfield2,
569 compare(Bfield1, Bfield2);
572 if (command ==
"COMPARE_RGRID") {
574 std::string filecompare1, filecompare2;
575 readEcho(in, filecompare1);
576 readEcho(in, filecompare2);
580 domain().fieldIo().readFieldsRGrid(filecompare1, Rfield1,
582 domain().fieldIo().readFieldsRGrid(filecompare2, Rfield2,
587 compare(Rfield1, Rfield2);
590 if (command ==
"SCALE_BASIS") {
592 readEcho(in, inFileName);
593 readEcho(in, outFileName);
594 readEcho(in, factor);
595 scaleFieldsBasis(inFileName, outFileName, factor);
597 if (command ==
"SCALE_RGRID") {
599 readEcho(in, inFileName);
600 readEcho(in, outFileName);
601 readEcho(in, factor);
602 scaleFieldsRGrid(inFileName, outFileName, factor);
604 if (command ==
"EXPAND_RGRID_DIMENSION") {
605 readEcho(in, inFileName);
606 readEcho(in, outFileName);
612 Log::file() <<
Str(
"Expand dimension to: ") << d <<
"\n";
617 for (
int i = 0; i < d-D; i++) {
618 in >> newGridDimensions[i];
621 <<
Str(
"Numbers of grid points in added dimensions : ");
622 for (
int i = 0; i < d-D; i++) {
623 Log::file() << newGridDimensions[i] <<
" ";
628 d, newGridDimensions);
631 if (command ==
"REPLICATE_UNIT_CELL") {
632 readEcho(in, inFileName);
633 readEcho(in, outFileName);
637 for (
int i = 0; i < D; i++){
640 for (
int i = 0; i < D; i++){
641 Log::file() <<
"Replicate unit cell in direction "
649 if (command ==
"BASIS_TO_KGRID") {
650 readEcho(in, inFileName);
651 readEcho(in, outFileName);
652 basisToKGrid(inFileName, outFileName);
654 if (command ==
"READ_H_BASIS") {
655 readEcho(in, filename);
656 if (!h_.isAllocatedBasis()) {
657 h_.allocateBasis(domain().basis().nBasis());
659 if (!h_.isAllocatedRGrid()) {
660 h_.allocateRGrid(domain().mesh().dimensions());
663 h_.readBasis(filename, tmpUnitCell);
665 if (command ==
"READ_H_RGRID") {
666 readEcho(in, filename);
667 if (!h_.isAllocatedRGrid()) {
668 h_.allocateRGrid(domain().mesh().dimensions());
670 if (iterator().isSymmetric() && !h_.isAllocatedBasis()) {
671 mask_.allocateBasis(domain().basis().nBasis());
674 h_.readRGrid(filename, tmpUnitCell);
676 if (command ==
"WRITE_H_BASIS") {
677 readEcho(in, filename);
680 domain().fieldIo().writeFieldsBasis(filename, h_.basis(),
681 domain().unitCell());
683 if (command ==
"WRITE_H_RGRID") {
684 readEcho(in, filename);
686 domain().fieldIo().writeFieldsRGrid(filename, h_.rgrid(),
687 domain().unitCell());
689 if (command ==
"READ_MASK_BASIS") {
690 readEcho(in, filename);
692 if (!mask_.isAllocatedBasis()) {
693 mask_.allocateBasis(domain().basis().nBasis());
695 if (!mask_.isAllocatedRGrid()) {
696 mask_.allocateRGrid(domain().mesh().dimensions());
699 mask_.readBasis(filename, tmpUnitCell);
701 if (command ==
"READ_MASK_RGRID") {
702 readEcho(in, filename);
703 if (!mask_.isAllocatedRGrid()) {
704 mask_.allocateRGrid(domain().mesh().dimensions());
706 if (iterator().isSymmetric() && !mask_.isAllocatedBasis()) {
708 mask_.allocateBasis(domain().basis().nBasis());
711 mask_.readRGrid(filename, tmpUnitCell);
713 if (command ==
"WRITE_MASK_BASIS") {
714 readEcho(in, filename);
717 domain().fieldIo().writeFieldBasis(filename, mask_.basis(),
718 domain().unitCell());
720 if (command ==
"WRITE_MASK_RGRID") {
721 readEcho(in, filename);
723 domain().fieldIo().writeFieldRGrid(filename, mask_.rgrid(),
725 mask_.isSymmetric());
728 << command << std::endl;
740 if (fileMaster_.commandFileName().empty()) {
743 readCommands(fileMaster_.commandFile());
757 if (!domain_.unitCell().isInitialized()) {
760 UTIL_CHECK(domain_.unitCell().isInitialized());
763 if (!isAllocatedBasis_) {
764 allocateFieldsBasis();
768 w_.readBasis(filename, domain_.unitCell());
770 hasFreeEnergy_ =
false;
773 domain_.waveList().clearUnitCellData();
774 mixture_.clearUnitCellData();
783 if (!domain_.unitCell().isInitialized()) {
786 UTIL_CHECK(domain_.unitCell().isInitialized());
787 if (domain_.hasGroup() && !isAllocatedBasis_) {
790 allocateFieldsBasis();
794 w_.readRGrid(filename, domain_.unitCell());
796 hasFreeEnergy_ =
false;
799 domain_.waveList().clearUnitCellData();
800 mixture_.clearUnitCellData();
813 const int nm = mixture_.nMonomer();
819 if (!domain_.unitCell().isInitialized()) {
822 UTIL_CHECK(domain_.unitCell().isInitialized());
824 if (!isAllocatedBasis_) {
825 allocateFieldsBasis();
829 const int nb = domain_.basis().nBasis();
833 for (
int i = 0; i < nm; ++i) {
838 fieldIo().readFieldsBasis(filename, tmpCFieldsBasis,
842 for (
int i = 0; i < nb; ++i) {
843 for (
int j = 0; j < nm; ++j) {
844 tmpFieldsBasis_[j][i] = 0.0;
845 for (
int k = 0; k < nm; ++k) {
846 tmpFieldsBasis_[j][i] += interaction().chi(j,k)
847 * tmpCFieldsBasis[k][i];
853 w_.setBasis(tmpFieldsBasis_);
855 hasFreeEnergy_ =
false;
858 domain_.waveList().clearUnitCellData();
859 mixture_.clearUnitCellData();
874 hasFreeEnergy_ =
false;
886 hasFreeEnergy_ =
false;
898 hasFreeEnergy_ =
false;
916 domain_.setUnitCell(unitCell);
918 mixture_.clearUnitCellData();
919 if (domain_.hasGroup() && !isAllocatedBasis_) {
921 allocateFieldsBasis();
933 domain_.setUnitCell(lattice, parameters);
935 mixture_.clearUnitCellData();
936 if (domain_.hasGroup() && !isAllocatedBasis_) {
938 allocateFieldsBasis();
948 domain_.setUnitCell(parameters);
950 mixture_.clearUnitCellData();
951 if (domain_.hasGroup() && !isAllocatedBasis_) {
953 allocateFieldsBasis();
970 mixture_.compute(w_.rgrid(), c_.rgrid(), mask_.phiTot());
972 hasFreeEnergy_ =
false;
975 if (w_.isSymmetric()) {
977 fieldIo().convertRGridToBasis(c_.rgrid(), c_.basis());
982 mixture_.computeStress(mask().phiTot());
994 if (iterator().isSymmetric()) {
999 hasFreeEnergy_ =
false;
1005 int error = iterator().solve(isContinuation);
1009 if (w_.isSymmetric()) {
1010 fieldIo().convertRGridToBasis(c_.rgrid(), c_.basis());
1015 if (!iterator().isFlexible()) {
1016 mixture_.computeStress(mask().phiTot());
1018 computeFreeEnergy();
1020 if (!iterator().isFlexible()) {
1035 if (iterator().isSymmetric()) {
1040 hasCFields_ =
false;
1041 hasFreeEnergy_ =
false;
1058 hasCFields_ =
false;
1059 hasFreeEnergy_ =
false;
1061 simulator().simulate(nStep);
1070 std::string
const & outFileName,
1078 domain().fieldIo().readFieldsRGrid(inFileName,
1083 domain().fieldIo().expandRGridDimension(outFileName,
1086 d, newGridDimensions);
1094 std::string
const & outFileName,
1099 domain().fieldIo().readFieldsRGrid(inFileName, tmpFieldsRGrid_,
1103 domain().fieldIo().replicateUnitCell(outFileName,
1117 if (hasFreeEnergy_)
return;
1122 int nm = mixture().nMonomer();
1123 int np = mixture().nPolymer();
1124 int ns = mixture().nSolvent();
1138 for (
int i = 0; i < np; ++i) {
1139 polymerPtr = &mixture_.polymer(i);
1140 phi = polymerPtr->
phi();
1141 mu = polymerPtr->
mu();
1143 length = polymerPtr->
length();
1144 if (phi > 1.0E-08) {
1145 fIdeal_ += phi*( mu - 1.0 )/length;
1154 for (
int i = 0; i < ns; ++i) {
1155 solventPtr = &mixture_.solvent(i);
1156 phi = solventPtr->
phi();
1157 mu = solventPtr->
mu();
1158 size = solventPtr->
size();
1159 if (phi > 1.0E-08) {
1160 fIdeal_ += phi*( mu - 1.0 )/size;
1175 const double phiTot = mask().phiTot();
1179 if (w_.isSymmetric()) {
1181 const int nBasis = domain_.basis().nBasis();
1182 for (
int i = 0; i < nm; ++i) {
1183 for (
int k = 0; k < nBasis; ++k) {
1184 temp -= w_.basis(i)[k] * c_.basis(i)[k];
1189 const int meshSize = domain().mesh().size();
1190 for (
int i = 0; i < nm; ++i) {
1193 temp /= double(meshSize);
1197 fHelmholtz_ += fIdeal_;
1200 if (hasExternalFields()) {
1201 if (iterator().isSymmetric()) {
1203 const int nBasis = domain_.basis().nBasis();
1204 for (
int i = 0; i < nm; ++i) {
1205 for (
int k = 0; k < nBasis; ++k) {
1206 fExt_ += h_.basis(i)[k] * c_.basis(i)[k];
1211 const int meshSize = domain().mesh().size();
1212 for (
int i = 0; i < nm; ++i) {
1215 fExt_ /= double(meshSize);
1218 fHelmholtz_ += fExt_;
1222 if (w_.isSymmetric()) {
1223 const int nBasis = domain_.basis().nBasis();
1224 for (
int i = 0; i < nm; ++i) {
1225 for (
int j = i; j < nm; ++j) {
1226 const double chi = interaction().chi(i,j);
1227 if (std::abs(chi) > 1.0E-9) {
1229 for (
int k = 0; k < nBasis; ++k) {
1230 temp += c_.basis(i)[k] * c_.basis(j)[k];
1233 fInter_ += 0.5*chi*temp;
1235 fInter_ += chi*temp;
1241 const int meshSize = domain().mesh().size();
1242 for (
int i = 0; i < nm; ++i) {
1243 for (
int j = i; j < nm; ++j) {
1244 const double chi = interaction().chi(i,j);
1245 if (std::abs(chi) > 1.0E-9) {
1248 fInter_ += 0.5*chi*temp;
1250 fInter_ += chi*temp;
1255 fInter_ /= double(meshSize);
1258 fHelmholtz_ += fInter_;
1261 pressure_ = -fHelmholtz_;
1267 for (
int i = 0; i < np; ++i) {
1268 polymerPtr = &mixture_.polymer(i);
1269 phi = polymerPtr->
phi();
1270 mu = polymerPtr->
mu();
1271 length = polymerPtr->
length();
1272 if (phi > 1.0E-08) {
1273 pressure_ += mu * phi / length;
1282 for (
int i = 0; i < ns; ++i) {
1283 solventPtr = &mixture_.solvent(i);
1284 phi = solventPtr->
phi();
1285 mu = solventPtr->
mu();
1286 size = solventPtr->
size();
1287 if (phi > 1.0E-08) {
1288 pressure_ += mu * phi / size;
1293 hasFreeEnergy_ =
true;
1303 if (hasIterator()) {
1305 iterator().outputTimers(out);
1307 if (hasSimulator()){
1309 simulator().outputTimers(out);
1319 if (hasIterator()) {
1320 iterator().clearTimers();
1322 if (hasSimulator()){
1323 simulator().clearTimers();
1332 out <<
"System{" << std::endl;
1333 mixture_.writeParam(out);
1334 interaction().writeParam(out);
1335 domain_.writeParam(out);
1336 if (hasIterator()) {
1337 iteratorPtr_->writeParam(out);
1339 out <<
"}" << std::endl;
1348 if (!hasFreeEnergy_) {
1349 computeFreeEnergy();
1353 out <<
"fHelmholtz " <<
Dbl(fHelmholtz(), 18, 11) << std::endl;
1354 out <<
"pressure " <<
Dbl(pressure(), 18, 11) << std::endl;
1356 out <<
"fIdeal " <<
Dbl(fIdeal_, 18, 11) << std::endl;
1357 out <<
"fInter " <<
Dbl(fInter_, 18, 11) << std::endl;
1358 if (hasExternalFields()) {
1359 out <<
"fExt " <<
Dbl(fExt_, 18, 11) << std::endl;
1363 int np = mixture_.nPolymer();
1364 int ns = mixture_.nSolvent();
1367 out <<
"polymers:" << std::endl;
1372 for (
int i = 0; i < np; ++i) {
1374 <<
" " <<
Dbl(mixture_.polymer(i).phi(),18, 11)
1375 <<
" " <<
Dbl(mixture_.polymer(i).mu(), 18, 11)
1382 out <<
"solvents:" << std::endl;
1387 for (
int i = 0; i < ns; ++i) {
1389 <<
" " <<
Dbl(mixture_.solvent(i).phi(),18, 11)
1390 <<
" " <<
Dbl(mixture_.solvent(i).mu(), 18, 11)
1396 out <<
"cellParams:" << std::endl;
1397 for (
int i = 0; i < domain_.unitCell().nParameter(); ++i) {
1400 <<
Dbl(domain_.unitCell().parameter(i), 18, 11)
1412 out <<
"stress:" << std::endl;
1413 for (
int i = 0; i < domain().unitCell().nParameter(); ++i) {
1416 <<
Dbl(mixture_.stress(i), 18, 11)
1430 fieldIo().writeFieldsBasis(filename, w_.basis(), domain_.unitCell());
1440 fieldIo().writeFieldsRGrid(filename, w_.rgrid(),
1453 fieldIo().writeFieldsBasis(filename, c_.basis(), domain_.unitCell());
1463 fieldIo().writeFieldsRGrid(filename, c_.rgrid(),
1479 blockCFields.
allocate(mixture_.nSolvent() + mixture_.nBlock());
1481 for (
int i = 0; i < n; i++) {
1482 blockCFields[i].
allocate(domain_.mesh().dimensions());
1486 mixture_.createBlockCRGrid(blockCFields);
1487 fieldIo().writeFieldsRGrid(filename, blockCFields,
1497 int polymerId,
int blockId,
1498 int directionId,
int segmentId)
1503 Polymer<D> const& polymer = mixture_.polymer(polymerId);
1509 propagator = polymer.
propagator(blockId, directionId);
1510 fieldIo().writeFieldRGrid(filename, propagator.
q(segmentId),
1511 domain_.unitCell(), w_.isSymmetric());
1519 int polymerId,
int blockId,
int directionId)
1524 Polymer<D> const& polymer = mixture_.polymer(polymerId);
1530 propagator = polymer.
propagator(blockId, directionId);
1531 fieldIo().writeFieldRGrid(filename, propagator.
tail(),
1532 domain_.unitCell(), w_.isSymmetric());
1540 int polymerId,
int blockId,
int directionId)
1545 Polymer<D> const& polymer = mixture_.polymer(polymerId);
1551 propagator = polymer.
propagator(blockId, directionId);
1552 int ns = propagator.
ns();
1556 fileMaster_.openOutputFile(filename, file);
1559 fieldIo().writeFieldHeader(file, 1, domain_.unitCell(),
1561 file <<
"ngrid" << std::endl
1562 <<
" " << domain_.mesh().dimensions() << std::endl
1563 <<
"nslice" << std::endl
1564 <<
" " << ns << std::endl;
1567 bool hasHeader =
false;
1568 for (
int i = 0; i < ns; ++i) {
1569 file <<
"slice " << i << std::endl;
1570 fieldIo().writeFieldRGrid(file, propagator.
q(i),
1571 domain_.unitCell(), hasHeader);
1581 std::string filename;
1582 int np, nb, ip, ib, id;
1583 np = mixture_.nPolymer();
1584 for (ip = 0; ip < np; ++ip) {
1587 nb = mixture_.polymer(ip).nBlock();
1588 for (ib = 0; ib < nb; ++ib) {
1589 for (
id = 0;
id < 2; ++id) {
1590 filename = basename;
1598 writeQ(filename, ip, ib,
id);
1612 std::ofstream outFile;
1613 fileMaster_.openOutputFile(filename, outFile);
1614 bool isSymmetric =
true;
1615 fieldIo().writeFieldHeader(outFile, mixture_.nMonomer(),
1616 domain_.unitCell(), isSymmetric);
1617 basis().outputStars(outFile);
1628 std::ofstream outFile;
1629 fileMaster_.openOutputFile(filename, outFile);
1630 bool isSymmetric =
true;
1631 fieldIo().writeFieldHeader(outFile, mixture_.nMonomer(),
1632 domain_.unitCell(), isSymmetric);
1633 basis().outputWaves(outFile);
1643 Pscf::Prdc::writeGroup(filename, domain_.group());
1653 const std::string & outFileName)
1659 if (!domain_.unitCell().isInitialized()) {
1662 UTIL_CHECK(domain_.unitCell().isInitialized());
1664 if (!isAllocatedBasis_) {
1665 allocateFieldsBasis();
1670 fieldIo().readFieldsBasis(inFileName, tmpFieldsBasis_, tmpUnitCell);
1671 fieldIo().convertBasisToRGrid(tmpFieldsBasis_, tmpFieldsRGrid_);
1672 fieldIo().writeFieldsRGrid(outFileName, tmpFieldsRGrid_,
1681 const std::string & outFileName)
1687 if (!domain_.unitCell().isInitialized()) {
1690 UTIL_CHECK(domain_.unitCell().isInitialized());
1692 if (!isAllocatedBasis_) {
1693 allocateFieldsBasis();
1698 fieldIo().readFieldsRGrid(inFileName, tmpFieldsRGrid_, tmpUnitCell);
1699 fieldIo().convertRGridToBasis(tmpFieldsRGrid_, tmpFieldsBasis_);
1700 fieldIo().writeFieldsBasis(outFileName, tmpFieldsBasis_,
1709 const std::string& outFileName)
1712 if (!domain_.unitCell().isInitialized()) {
1715 UTIL_CHECK(domain_.unitCell().isInitialized());
1719 fieldIo().readFieldsKGrid(inFileName, tmpFieldsKGrid_, tmpUnitCell);
1720 for (
int i = 0; i < mixture_.nMonomer(); ++i) {
1721 fft().inverseTransformUnsafe(tmpFieldsKGrid_[i], tmpFieldsRGrid_[i]);
1723 fieldIo().writeFieldsRGrid(outFileName, tmpFieldsRGrid_,
1732 const std::string & outFileName)
1735 if (!domain_.unitCell().isInitialized()) {
1738 UTIL_CHECK(domain_.unitCell().isInitialized());
1742 fieldIo().readFieldsRGrid(inFileName, tmpFieldsRGrid_,
1744 for (
int i = 0; i < mixture_.nMonomer(); ++i) {
1745 fft().forwardTransform(tmpFieldsRGrid_[i], tmpFieldsKGrid_[i]);
1747 fieldIo().writeFieldsKGrid(outFileName, tmpFieldsKGrid_,
1756 const std::string& outFileName)
1762 if (!domain_.unitCell().isInitialized()) {
1765 UTIL_CHECK(domain_.unitCell().isInitialized());
1767 if (!isAllocatedBasis_) {
1768 allocateFieldsBasis();
1773 fieldIo().readFieldsKGrid(inFileName, tmpFieldsKGrid_, tmpUnitCell);
1774 fieldIo().convertKGridToBasis(tmpFieldsKGrid_, tmpFieldsBasis_);
1775 fieldIo().writeFieldsBasis(outFileName, tmpFieldsBasis_,
1784 const std::string & outFileName)
1790 if (!domain_.unitCell().isInitialized()) {
1793 UTIL_CHECK(domain_.unitCell().isInitialized());
1795 if (!isAllocatedBasis_) {
1796 allocateFieldsBasis();
1801 fieldIo().readFieldsBasis(inFileName, tmpFieldsBasis_, tmpUnitCell);
1802 fieldIo().convertBasisToKGrid(tmpFieldsBasis_, tmpFieldsKGrid_);
1803 fieldIo().writeFieldsKGrid(outFileName, tmpFieldsKGrid_,
1819 if (!isAllocatedBasis_) {
1821 allocateFieldsBasis();
1826 domain_.fieldIo().readFieldsRGrid(inFileName,
1827 tmpFieldsRGrid_, tmpUnitCell);
1830 for (
int i = 0; i < mixture().nMonomer(); ++i) {
1832 symmetric = domain_.fieldIo().hasSymmetry(tmpFieldsRGrid_[i],
1847 std::string
const & outFileName,
1852 if (!isAllocatedBasis_) {
1854 allocateFieldsBasis();
1858 FieldIo<D> const & fieldIo = domain().fieldIo();
1870 std::string
const & outFileName,
1871 double factor)
const
1874 FieldIo<D> const & fieldIo = domain().fieldIo();
1892 comparison.
compare(field1,field2);
1894 Log::file() <<
"\n Basis expansion field comparison results"
1896 Log::file() <<
" Maximum Absolute Difference: "
1897 << comparison.
maxDiff() << std::endl;
1898 Log::file() <<
" Root-Mean-Square Difference: "
1899 << comparison.
rmsDiff() <<
"\n" << std::endl;
1910 comparison.
compare(field1, field2);
1912 Log::file() <<
"\n Real-space field comparison results"
1914 Log::file() <<
" Maximum Absolute Difference: "
1915 << comparison.
maxDiff() << std::endl;
1916 Log::file() <<
" Root-Mean-Square Difference: "
1917 << comparison.
rmsDiff() <<
"\n" << std::endl;
1930 int nMonomer = mixture_.nMonomer();
1936 IntVec<D> const & dimensions = domain_.mesh().dimensions();
1939 w_.setNMonomer(nMonomer);
1940 w_.allocateRGrid(dimensions);
1943 c_.setNMonomer(nMonomer);
1944 c_.allocateRGrid(dimensions);
1946 h_.setNMonomer(nMonomer);
1949 tmpFieldsRGrid_.allocate(nMonomer);
1950 tmpFieldsKGrid_.allocate(nMonomer);
1951 for (
int i = 0; i < nMonomer; ++i) {
1952 tmpFieldsRGrid_[i].allocate(dimensions);
1953 tmpFieldsKGrid_[i].allocate(dimensions);
1955 workArray_.allocate(dimensions);
1957 isAllocatedGrid_ =
true;
1964 void System<D>::allocateFieldsBasis()
1968 const int nMonomer = mixture_.nMonomer();
1970 UTIL_CHECK(domain_.unitCell().isInitialized());
1971 UTIL_CHECK(domain_.unitCell().nParameter() > 0);
1976 const int nBasis = basis().nBasis();
1980 w_.allocateBasis(nBasis);
1981 c_.allocateBasis(nBasis);
1984 tmpFieldsBasis_.allocate(nMonomer);
1985 for (
int i = 0; i < nMonomer; ++i) {
1986 tmpFieldsBasis_[i].allocate(nBasis);
1989 isAllocatedBasis_ =
true;
1996 void System<D>::readFieldHeader(std::string filename)
2004 fileMaster_.openInputFile(filename, file);
2009 domain_.fieldIo().readFieldHeader(file, nMonomer,
2010 domain_.unitCell(), isSymmetric);
2017 UTIL_CHECK(domain_.unitCell().nParameter() > 0);
2019 UTIL_CHECK(domain_.unitCell().isInitialized());
2020 if (domain_.hasGroup()) {
2030 void System<D>::readEcho(std::istream& in, std::string&
string)
const
2040 void System<D>::readEcho(std::istream& in,
double& value)
const
2044 UTIL_THROW(
"Unable to read floating point parameter.");
2053 void System<D>::initHomogeneous()
2057 int nm = mixture_.nMonomer();
2058 int np = mixture_.nPolymer();
2059 int ns = mixture_.nSolvent();
2061 UTIL_CHECK(homogeneous_.nMolecule() == np + ns);
2076 for (i = 0; i < np; ++i) {
2079 for (j = 0; j < nm; ++j) {
2084 nb = mixture_.polymer(i).nBlock();
2085 for (k = 0; k < nb; ++k) {
2086 Block<D>& block = mixture_.polymer(i).block(k);
2087 j = block.monomerId();
2088 cTmp[j] += block.length();
2093 for (j = 0; j < nm; ++j) {
2094 if (cTmp[j] > 1.0E-8) {
2098 homogeneous_.molecule(i).setNClump(nc);
2102 for (j = 0; j < nm; ++j) {
2103 if (cTmp[j] > 1.0E-8) {
2104 homogeneous_.molecule(i).clump(k).setMonomerId(j);
2105 homogeneous_.molecule(i).clump(k).setSize(cTmp[j]);
2109 homogeneous_.molecule(i).computeSize();
2118 for (
int is = 0; is < ns; ++is) {
2120 monomerId = mixture_.solvent(is).monomerId();
2121 size = mixture_.solvent(is).size();
2122 homogeneous_.molecule(i).setNClump(1);
2123 homogeneous_.molecule(i).clump(0).setMonomerId(monomerId);
2124 homogeneous_.molecule(i).clump(0).setSize(size);
2125 homogeneous_.molecule(i).computeSize();
Dynamic array on the GPU device with aligned data.
double rmsDiff() const
Return the precomputed root-mean-squared difference.
double compare(FT const &a, FT const &b)
Compare individual fields.
double maxDiff() const
Return the precomputed maximum element-by-element difference.
An IntVec<D, T> is a D-component vector of elements of integer type T.
Flory-Huggins excess free energy model.
int nBlock() const
Number of blocks.
Propagator & propagator(int blockId, int directionId)
Get propagator for a specific block and direction.
double length() const
Sum of the lengths of all blocks in the polymer.
Comparator for fields in symmetry-adapted basis format.
Comparator for fields in real-space (r-grid) format.
Field of real double precision values on an FFT mesh.
void scaleFieldsRGrid(DArray< RFRT > &fields, double factor) const
Scale an array of r-grid fields by a real scalar.
void readFieldsBasis(std::istream &in, DArray< DArray< double > > &fields, UnitCell< D > &unitCell) const
Read concentration or chemical potential fields from file.
void scaleFieldsBasis(DArray< DArray< double > > &fields, double factor) const
Scale an array of fields in basis format by a real scalar.
void writeFieldsBasis(std::ostream &out, DArray< DArray< double > > const &fields, UnitCell< D > const &unitCell) const
Write concentration or chemical potential field components to file.
Base template for UnitCell<D> classes, D=1, 2 or 3.
File input/output operations and format conversions for fields.
void writeFieldsRGrid(std::ostream &out, DArray< RField< D > > const &fields, UnitCell< D > const &unitCell, bool writeHeader=true, bool isSymmetric=true, bool writeMeshSize=true) const
Write array of RField objects (fields on r-space grid) to a stream.
void readFieldsRGrid(std::istream &in, DArray< RField< D > > &fields, UnitCell< D > &unitCell) const
Read array of RField objects (r-grid fields) from a stream.
Factory for subclasses of Iterator.
Descriptor and solver for a branched polymer species.
MDE solver for one-direction of one block.
int ns() const
Get the number of chain contour points.
RField< D > const & tail() const
Return q-field at end of block (after propagator is solved).
RField< D > const & q(int i) const
Return const q-field at specified step by reference (after solving).
Factory for subclasses of Simulator.
Solver and descriptor for a solvent species.
Default Factory for subclasses of Sweep.
Main class for calculations that represent one system.
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.
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.
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.
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.
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.
void writeWRGrid(const std::string &filename) const
Write chemical potential fields in real space grid (r-grid) format.
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.
virtual void readParameters(std::istream &in)
Read body of parameter file (without opening, closing lines).
void sweep()
Sweep in parameter space, solving an SCF problem at each point.
void basisToKGrid(const std::string &inFileName, const std::string &outFileName)
Convert fields from symmetrized basis to Fourier (k-grid) format.
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.
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.
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.
void writeCRGrid(const std::string &filename) const
Write concentration fields in real space grid (r-grid) format.
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.
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.
void writeThermo(std::ostream &out)
Write thermodynamic properties to a file.
void simulate(int nStep)
Perform a field theoretic simulation.
void scaleFieldsBasis(const std::string &inFileName, const std::string &outFileName, double factor)
Multiply all components of an array of basis fields by a scalar.
double size() const
Get the size (number of monomers) in this solvent.
double phi() const
Get the overall volume fraction for this species.
double mu() const
Get the chemical potential for this species (units kT=1).
int capacity() const
Return allocated size.
Dynamically allocatable contiguous array template.
void allocate(int capacity)
Allocate the underlying C array.
Wrapper for a double precision number, for formatted ostream output.
A fixed capacity (static) contiguous array with a variable logical size.
Wrapper for an int, for formatted ostream output.
static std::ostream & file()
Get log ostream by reference.
static bool echo()
Get echo parameter.
static void setEcho(bool echo=true)
Enable or disable echoing for all subclasses of ParamComponent.
void setClassName(const char *className)
Set class name string.
Wrapper for a std::string, 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.
std::string toString(int n)
Return string representation of an integer.
void readFieldHeader(std::istream &in, int &ver1, int &ver2, UnitCell< D > &cell, std::string &groupName, int &nMonomer)
Read common part of field header (fortran PSCF format).
void replicateUnitCell(std::ostream &out, DArray< AT > const &fields, IntVec< D > const &meshDimensions, UnitCell< D > const &unitCell, IntVec< D > const &replicas)
Write r-grid fields in a replicated unit cell to std::ostream.
bool hasSymmetry(AT const &in, Basis< D > const &basis, IntVec< D > const &dftDimensions, double epsilon=1.0e-8, bool verbose=true)
Check if a k-grid field has the declared space group symmetry.
void expandRGridDimension(std::ostream &out, DArray< AT > const &fields, IntVec< D > const &meshDimensions, UnitCell< D > const &unitCell, int d, DArray< int > newGridDimensions)
Expand the dimensionality of space from D to d.
void init()
Initialize static variables in Pscf::ThreadArray namespace.
void setThreadsPerBlock()
Set the number of threads per block to a default value.
void setThreadsPerBlock(int blockSize)
Manually set the block size that should be used by default.
cudaReal innerProduct(DeviceArray< cudaReal > const &a, DeviceArray< cudaReal > const &b)
Compute inner product of two real arrays (GPU kernel wrapper).
Fields, FFTs, and utilities for periodic boundary conditions (CUDA)
Periodic fields and crystallography.
PSCF package top-level namespace.
void set(BracketPolicy::Type policy)
Set policy regarding use of bracket delimiters on arrays.
Utility classes for scientific computation.