PSCF v1.2
FieldIoReal.tpp
1#ifndef RPC_FIELD_IO_REAL_TPP
2#define RPC_FIELD_IO_REAL_TPP
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#include "FieldIoReal.h"
12
13#include <prdc/field/fieldIoUtil.h>
14#include <prdc/crystal/fieldHeader.h>
15
16#include <prdc/crystal/shiftToMinimum.h>
17#include <prdc/crystal/UnitCell.h>
18#include <prdc/crystal/Basis.h>
19#include <prdc/crystal/SpaceGroup.h>
20#include <prdc/crystal/UnitCell.h>
21
22#include <pscf/mesh/Mesh.h>
23#include <pscf/mesh/MeshIterator.h>
24#include <pscf/mesh/MeshIteratorFortran.h>
25#include <pscf/math/IntVec.h>
26
27#include <util/containers/DArray.h>
28#include <util/misc/FileMaster.h>
29#include <util/misc/Log.h>
30#include <util/format/Str.h>
31#include <util/format/Int.h>
32#include <util/format/Dbl.h>
33
34#include <iomanip>
35#include <string>
36
37namespace Pscf {
38namespace Prdc {
39
40 using namespace Util;
41
42 /*
43 * Constructor.
44 */
45 template <int D, class RFRT, class RFKT, class FFTT>
47 : meshPtr_(0),
48 fftPtr_(0),
49 hasGroupPtr_(0),
50 groupNamePtr_(0),
51 groupPtr_(0),
52 basisPtr_(0),
53 fileMasterPtr_()
54 {}
55
56 /*
57 * Destructor.
58 */
59 template <int D, class RFRT, class RFKT, class FFTT>
62
63
64 // %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
65 // %%%%%%% Shared Functions : Implemented in this template %%%%
66 // %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
67
68 /*
69 * Create associations with other members of parent Domain.
70 */
71 template <int D, class RFRT, class RFKT, class FFTT>
72 void
74 Mesh<D> const & mesh,
75 FFTT const & fft,
76 typename UnitCell<D>::LatticeSystem const & lattice,
77 bool const & hasGroup,
78 std::string const & groupName,
79 SpaceGroup<D> const & group,
80 Basis<D> & basis)
81 {
82 meshPtr_ = &mesh;
83 fftPtr_ = &fft;
84 latticePtr_ = &lattice;
85 hasGroupPtr_ = &hasGroup;
86 groupNamePtr_ = &groupName;
87 groupPtr_ = &group;
88 basisPtr_ = &basis;
89 }
90
91 /*
92 * Create an association with a FileMaster.
93 */
94 template <int D, class RFRT, class RFKT, class FFTT>
96 FileMaster const & fileMaster)
97 { fileMasterPtr_ = &fileMaster; }
98
99 // Field File IO - Symmetry-Adapted Basis Format
100
101 /*
102 * Read an array of fields in basis format from an input stream.
103 */
104 template <int D, class RFRT, class RFKT, class FFTT>
106 std::istream& in,
107 DArray< DArray<double> >& fields,
108 UnitCell<D>& unitCell) const
110 // Precondition
111 UTIL_CHECK(hasGroup());
112
113 // Read header (checks compatibility with space group)
114 int nMonomer;
115 bool isSymmetric;
116 readFieldHeader(in, nMonomer, unitCell, isSymmetric);
117 UTIL_CHECK(isSymmetric);
118 UTIL_CHECK(basis().isInitialized());
119 // Note: readFieldHeader can initialize basis if not done previously
120 int nBasisIn = readNBasis(in);
121
122 // Check allocation of fields container, allocate if necessary
123 if (fields.isAllocated()) {
124 int nMonomerFields, fieldCapacity;
125 inspectArrays(fields, nMonomerFields, fieldCapacity);
126 UTIL_CHECK(nMonomerFields == nMonomer);
127 } else {
128 fields.allocate(nMonomer);
129 for (int i = 0; i < nMonomer; ++i) {
130 fields[i].allocate(nBasisIn);
131 }
132 }
133
134 // Read field data (components in basis)
135 Prdc::readBasisData(in, fields, unitCell, mesh(), basis(), nBasisIn);
136 }
137
138 /*
139 * Read a single field in basis format from an input stream
140 */
141 template <int D, class RFRT, class RFKT, class FFTT>
143 std::istream& in,
144 DArray<double>& field,
145 UnitCell<D>& unitCell) const
146 {
147 // Local array container, of type required by readFieldsBasis
148 DArray< DArray<double> > fields;
149
150 // If single field is allocated, allocate local array fields
151 if (field.isAllocated()) {
152 fields.allocate(1);
153 fields[0].allocate(field.capacity());
154 }
155 // Otherwise, pass unallocated fields array to readFieldsBasis
156
157 // Read file containing a single field, allocate fields if needed.
158 readFieldsBasis(in, fields, unitCell);
159
160 // Check that it only read 1 field
161 UTIL_CHECK(fields.capacity() == 1);
162
163 // Copy data from local array fields to function parameter field
164 field = fields[0];
166
167 /*
168 * Write an array of fields in basis format to an output stream.
169 */
170 template <int D, class RFRT, class RFKT, class FFTT>
172 std::ostream &out,
173 DArray< DArray<double> > const & fields,
174 UnitCell<D> const & unitCell) const
175 {
176 // Inspect fields to obtain nMonomer and fieldCapacity
177 int nMonomer;
178 int fieldCapacity;
179 inspectArrays(fields, nMonomer, fieldCapacity);
181 // Preconditions
182 UTIL_CHECK(basis().isInitialized());
183 UTIL_CHECK(fieldCapacity <= basis().nBasis());
184 int nBasis = fieldCapacity;
185
186 // Write header
187 bool isSymmetric = true;
188 writeFieldHeader(out, nMonomer, unitCell, isSymmetric);
189 writeNBasis(out, nBasis);
190
191 // Write data (field components)
192 Prdc::writeBasisData(out, fields, basis());
193 }
195 /*
196 * Write a single field in basis format to an output stream.
197 */
198 template <int D, class RFRT, class RFKT, class FFTT>
200 std::ostream& out,
201 DArray<double> const & field,
202 UnitCell<D> const & unitCell) const
203 {
204 // Create local array of type required by writeFieldsBasis
205 DArray<DArray<double> > fields;
206 fields.allocate(1);
207 fields[0].allocate(field.capacity());
209 // Copy data from input parameter to local array
210 fields[0] = field;
211
212 writeFieldsBasis(out, fields, unitCell);
213 }
214
215 // Field File IO - R-Grid Format
216 // K-Grid Field Format
217
218 // Field Format Conversion Functions - Symmetry Adapted Basis
219
220 template <int D, class RFRT, class RFKT, class FFTT>
222 DArray< DArray <double> > const & in,
223 DArray< RFKT >& out) const
224 {
225 // Inspect input and output field containers
226 int nMonomer, nMonomerOut, capacity;
227 inspectArrays(in, nMonomer, capacity);
228 IntVec<D> dimensions;
229 inspectFields(out, nMonomerOut, dimensions);
230 UTIL_CHECK(nMonomer == nMonomerOut);
231
232 // Convert fields for all monomer types
233 for (int i = 0; i < nMonomer; ++i) {
234 convertBasisToKGrid(in[i], out[i]);
235 }
237
238 template <int D, class RFRT, class RFKT, class FFTT>
240 DArray< RFKT > const & in,
241 DArray< DArray <double> > & out,
242 bool checkSymmetry,
243 double epsilon) const
244 {
245 // Inspect input and output containers
246 int nMonomer, nMonomerOut, capacity;
247 IntVec<D> dimensions;
248 inspectFields(in, nMonomer, dimensions);
249 inspectArrays(out, nMonomerOut, capacity);
250 UTIL_CHECK(nMonomer == nMonomerOut);
251
252 // Convert fields for all monomer types
253 for (int i = 0; i < nMonomer; ++i) {
254 convertKGridToBasis(in[i], out[i], checkSymmetry, epsilon);
255 }
256 }
257
258 /*
259 * Test if an RFRT has declared space group symmetry.
260 * Return true if symmetric, false otherwise. Print error values
261 * if verbose == true and hasSymmetry == false.
262 */
263 template <int D, class RFRT, class RFKT, class FFTT>
265 RFRT const & in,
266 double epsilon,
267 bool verbose) const
268 {
269 checkAllocateField(workDft_, mesh().dimensions());
270 fft().forwardTransform(in, workDft_);
271 return hasSymmetry(workDft_, epsilon, verbose);
272 }
273
274 template <int D, class RFRT, class RFKT, class FFTT>
276 DArray<double> const & in,
277 RFRT& out) const
278 {
279 checkAllocateField(workDft_, mesh().dimensions());
280 convertBasisToKGrid(in, workDft_);
281 fft().inverseTransformUnsafe(workDft_, out);
282 }
283
284 template <int D, class RFRT, class RFKT, class FFTT>
286 DArray< DArray <double> > const & in,
287 DArray< RFRT >& out) const
288 {
289 UTIL_ASSERT(in.capacity() == out.capacity());
290 checkAllocateField(workDft_, mesh().dimensions());
291
292 int n = in.capacity();
293 for (int i = 0; i < n; ++i) {
294 convertBasisToKGrid(in[i], workDft_);
295 fft().inverseTransformUnsafe(workDft_, out[i]);
296 }
297 }
298
299 template <int D, class RFRT, class RFKT, class FFTT>
301 RFRT const & in,
302 DArray<double> & out,
303 bool checkSymmetry,
304 double epsilon) const
305 {
306 checkAllocateField(workDft_, mesh().dimensions());
307 fft().forwardTransform(in, workDft_);
308 convertKGridToBasis(workDft_, out, checkSymmetry, epsilon);
310
311 template <int D, class RFRT, class RFKT, class FFTT>
313 DArray< RFRT > const & in,
314 DArray< DArray <double> > & out,
315 bool checkSymmetry,
316 double epsilon) const
317 {
318 // Inspect input and output containers
319 int nMonomer, nMonomerOut, capacity;
320 IntVec<D> dimensions;
321 inspectFields(in, nMonomer, dimensions);
322 UTIL_CHECK(mesh().dimensions() == dimensions);
323 inspectArrays(out, nMonomerOut, capacity);
324 UTIL_CHECK(nMonomer == nMonomerOut);
325 checkAllocateField(workDft_, dimensions);
326
327 // Convert RGrid -> KGrid -> Basis for each field
328 for (int i = 0; i < nMonomer; ++i) {
329 fft().forwardTransform(in[i], workDft_);
330 convertKGridToBasis(workDft_, out[i], checkSymmetry, epsilon);
331 }
332 }
333
334 // FFT conversion functions (KGrid <-> RGrid) [Same in Rpc and Rpg]
335
336 /*
337 * Apply inverse FFT to an array of k-grid fields.
338 */
339 template <int D, class RFRT, class RFKT, class FFTT>
341 DArray< RFKT > const & in,
342 DArray< RFRT >& out) const
343 {
344 UTIL_ASSERT(in.capacity() == out.capacity());
345 int n = in.capacity();
346 for (int i = 0; i < n; ++i) {
347 fft().inverseTransformSafe(in[i], out[i]);
348 }
349 }
350
351 /*
352 * Apply inverse FFT to a single k-grid field.
353 */
354 template <int D, class RFRT, class RFKT, class FFTT>
356 RFKT const & in, RFRT& out) const
357 {
358 fft().inverseTransformSafe(in, out);
359 }
360
361 /*
362 * Apply forward FFT to an array of r-grid fields.
363 */
364 template <int D, class RFRT, class RFKT, class FFTT>
366 DArray< RFRT > const & in,
367 DArray< RFKT >& out) const
368 {
369 UTIL_ASSERT(in.capacity() == out.capacity());
370 int n = in.capacity();
371 for (int i = 0; i < n; ++i) {
372 fft().forwardTransform(in[i], out[i]);
373 }
374 }
375
376 /*
377 * Apply forward FFT to a single r-grid field.
378 */
379 template <int D, class RFRT, class RFKT, class FFTT>
381 RFRT const & in,
382 RFKT& out) const
383 {
384 fft().forwardTransform(in, out);
385 }
386
387 // Grid Manipulation Utilities
388
389 /*
390 * File IO wrapper functions:
391 *
392 * These functions take a file name as an argument, and simply wrap
393 * file open and close operations around a function of the same name
394 * that takes an io stream argument. These functions can use the same
395 * implementation in Rpc and Rpg.
396 */
398 /*
399 * Open-close a file and read a set of fields in basis format.
400 */
401 template <int D, class RFRT, class RFKT, class FFTT>
403 std::string filename,
404 DArray<DArray<double> >& fields,
405 UnitCell<D>& unitCell) const
406 {
407
408 std::ifstream file;
409 fileMaster().openInputFile(filename, file);
410 readFieldsBasis(file, fields, unitCell);
411 file.close();
412 }
413
414 /*
415 * Open-close a file and read single fields in basis format.
416 */
417 template <int D, class RFRT, class RFKT, class FFTT>
419 std::string filename,
420 DArray<double>& field,
421 UnitCell<D>& unitCell) const
422 {
423 std::ifstream file;
424 fileMaster().openInputFile(filename, file);
425 readFieldBasis(file, field, unitCell);
426 file.close();
427 }
428
429 /*
430 * Open-close a file, and write an array of fields in basis format.
431 */
432 template <int D, class RFRT, class RFKT, class FFTT>
434 std::string filename,
435 DArray<DArray<double> > const & fields,
436 UnitCell<D> const & unitCell) const
437 {
438 std::ofstream file;
439 fileMaster().openOutputFile(filename, file);
440 writeFieldsBasis(file, fields, unitCell);
441 file.close();
442 }
443
444 /*
445 * Write a single field in basis format, open and close the file.
446 */
447 template <int D, class RFRT, class RFKT, class FFTT>
449 std::string filename,
450 DArray<double> const & field,
451 UnitCell<D> const & unitCell) const
452 {
453 std::ofstream file;
454 fileMaster().openOutputFile(filename, file);
455 writeFieldBasis(file, field, unitCell);
456 file.close();
457 }
458
459 template <int D, class RFRT, class RFKT, class FFTT>
461 std::string filename,
462 DArray< RFRT >& fields,
463 UnitCell<D>& unitCell) const
464 {
465 std::ifstream file;
466 fileMaster().openInputFile(filename, file);
467 readFieldsRGrid(file, fields, unitCell);
468 file.close();
469 }
470
471 template <int D, class RFRT, class RFKT, class FFTT>
473 std::string filename,
474 RFRT & field,
475 UnitCell<D>& unitCell) const
476 {
477 std::ifstream file;
478 fileMaster().openInputFile(filename, file);
479 readFieldRGrid(file, field, unitCell);
480 file.close();
481 }
482
483 template <int D, class RFRT, class RFKT, class FFTT>
485 std::string filename,
486 DArray< RFRT > const & fields,
487 UnitCell<D> const & unitCell,
488 bool isSymmetric) const
489 {
490 std::ofstream file;
491 fileMaster().openOutputFile(filename, file);
492 bool writeHeader = true;
493 bool writeMeshSize = true;
494 writeFieldsRGrid(file, fields, unitCell,
495 writeHeader, isSymmetric, writeMeshSize);
496 file.close();
497 }
498
499 template <int D, class RFRT, class RFKT, class FFTT>
501 std::string filename,
502 RFRT const & field,
503 UnitCell<D> const & unitCell,
504 bool isSymmetric) const
505 {
506 std::ofstream file;
507 fileMaster().openOutputFile(filename, file);
508 writeFieldRGrid(file, field, unitCell, isSymmetric);
509 file.close();
510 }
511
512 template <int D, class RFRT, class RFKT, class FFTT>
514 std::string filename,
515 DArray< RFKT >& fields,
516 UnitCell<D>& unitCell) const
517 {
518 std::ifstream file;
519 fileMaster().openInputFile(filename, file);
520 readFieldsKGrid(file, fields, unitCell);
521 file.close();
522 }
523
524 template <int D, class RFRT, class RFKT, class FFTT>
526 std::string filename,
527 DArray< RFKT > const & fields,
528 UnitCell<D> const & unitCell,
529 bool isSymmetric) const
530 {
531 std::ofstream file;
532 fileMaster().openOutputFile(filename, file);
533 writeFieldsKGrid(file, fields, unitCell, isSymmetric);
534 file.close();
535 }
536
537 template <int D, class RFRT, class RFKT, class FFTT>
539 DArray< DArray<double> >& fields,
540 double factor) const
541 {
542 int n = fields.capacity();
543 for (int i = 0; i < n; ++i) {
544 scaleFieldBasis(fields[i], factor);
545 }
546 }
547
548 template <int D, class RFRT, class RFKT, class FFTT>
550 DArray< RFRT > & fields,
551 double factor) const
552 {
553 int n = fields.capacity();
554 for (int i = 0; i < n; ++i) {
555 scaleFieldRGrid(fields[i], factor);
556 }
557 }
558
559 template <int D, class RFRT, class RFKT, class FFTT>
561 std::string filename,
562 DArray<RFRT> const & fields,
563 UnitCell<D> const & unitCell,
564 IntVec<D> const & replicas) const
565 {
566 std::ofstream file;
567 fileMaster().openOutputFile(filename, file);
568 replicateUnitCell(file, fields, unitCell, replicas);
569 file.close();
570 }
571
572 template <int D, class RFRT, class RFKT, class FFTT>
574 std::string filename,
575 DArray<RFRT> const & fields,
576 UnitCell<D> const & unitCell, int d,
577 DArray<int> newGridDimensions) const
578 {
579 std::ofstream file;
580 fileMaster().openOutputFile(filename, file);
581 expandRGridDimension(file, fields, unitCell, d, newGridDimensions);
582 file.close();
583 }
584
585 // File Header IO Utilities
587 /*
588 * Read common part of field header and extract the number of monomers
589 * (i.e., number of fields) and unitCell from the file.
590 */
591 template <int D, class RFRT, class RFKT, class FFTT>
593 std::istream& in,
594 int& nMonomer,
595 UnitCell<D>& unitCell,
596 bool & isSymmetric) const
597 {
598 // Preconditions
599 UTIL_CHECK(latticePtr_);
600 if (unitCell.lattice() == UnitCell<D>::Null) {
601 UTIL_CHECK(unitCell.nParameter() == 0);
602 } else {
603 UTIL_CHECK(unitCell.nParameter() > 0);
604 UTIL_CHECK(unitCell.lattice() == lattice());
606
607 // Read field header to set unitCell, groupNameIn, nMonomer
608 int ver1, ver2;
609 std::string groupNameIn;
610
611 Pscf::Prdc::readFieldHeader(in, ver1, ver2, unitCell,
612 groupNameIn, nMonomer);
613 // Note: Function definition in prdc/crystal/fieldHeader.tpp
614
615 // Checks of data from header
616 UTIL_CHECK(ver1 == 1);
617 //UTIL_CHECK(ver2 == 0);
618 UTIL_CHECK(unitCell.isInitialized());
619 UTIL_CHECK(unitCell.lattice() != UnitCell<D>::Null);
620 UTIL_CHECK(unitCell.nParameter() > 0);
622 // Validate or initialize lattice type
623 if (lattice() != unitCell.lattice()) {
624 Log::file() << std::endl
625 << "Error - Mismatched lattice types "
626 << "in FieldIo function readFieldHeader:\n"
627 << " FieldIo::lattice :" << lattice() << "\n"
628 << " Unit cell lattice :" << unitCell.lattice()
629 << "\n";
630 UTIL_THROW("Mismatched lattice types");
631 }
632
633 // Check for presence of group name
634 isSymmetric = false;
635 if (groupNameIn != "") {
636 isSymmetric = true;
637 }
638
639 // Process group and basis (if any)
640 if (hasGroup()) {
641
642 // Check consistency of groupName values
643 if (isSymmetric) {
644 UTIL_CHECK(groupNamePtr_);
645 if (groupNameIn != groupName()) {
646 Log::file() << std::endl
647 << "Error - Mismatched group names in "
648 << "FieldIo member function readFieldHeader:\n"
649 << " FieldIo::groupName :" << groupName() << "\n"
650 << " Field file header :" << groupNameIn << "\n";
651 UTIL_THROW("Mismatched group names");
653 }
654
655 // If there is a group but no basis, construct a basis
656 UTIL_CHECK(basisPtr_);
657 if (!basis().isInitialized()) {
658 basisPtr_->makeBasis(mesh(), unitCell, group());
659 }
660 UTIL_CHECK(basis().isInitialized());
661
662 }
663
664 }
665
666 template <int D, class RFRT, class RFKT, class FFTT>
668 std::ostream &out,
669 int nMonomer,
670 UnitCell<D> const & unitCell,
671 bool isSymmetric) const
672 {
673 int v1 = 1;
674 int v2 = 0;
675 std::string gName = "";
676 if (isSymmetric) {
677 UTIL_CHECK(hasGroup());
678 gName = groupName();
679 }
680 Pscf::Prdc::writeFieldHeader(out, v1, v2, unitCell,
681 gName, nMonomer);
682 // Note: This function is defined in prdc/crystal/fieldHeader.tpp
683 }
684
685 /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
686 * %%%%%%% Dummy Functions : Not implemented in this template %%%%%
687 * %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
688 *
689 * These virtual functions all have a default implementation that
690 * should never be called, and that throws an Exception. Such unusable
691 * implementations are provided, rather than declaring all of these
692 * functions to be pure virtual (= 0 suffix), because the absence of
693 * any pure virtual functions allows this to be a concrete class. This
694 * allows explicit instances of this template can be compiled for the
695 * 6 actual use cases, corresponding to D = 1, 2 or 3, with fields
696 * defined on the Cpu or Gpu).
697 */
698
699 /*
700 * Read an array of fields in r-grid format.
701 */
702 template <int D, class RFRT, class RFKT, class FFTT>
704 std::istream &in,
705 DArray<RFRT >& fields,
706 UnitCell<D>& unitCell) const
707 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
708
709 /*
710 * Read the data section of an array of fields in r-grid format.
711 */
712 template <int D, class RFRT, class RFKT, class FFTT>
714 std::istream& in,
715 DArray< RFRT >& fields,
716 int nMonomer) const
717 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
718
719 /*
720 * Read a single fields in r-grid format.
721 */
722 template <int D, class RFRT, class RFKT, class FFTT>
724 std::istream &in,
725 RFRT & field,
726 UnitCell<D>& unitCell) const
727 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
728
729 /*
730 * Write an array of fields in r-grid format
731 */
732 template <int D, class RFRT, class RFKT, class FFTT>
734 std::ostream &out,
735 DArray<RFRT > const & fields,
736 UnitCell<D> const & unitCell,
737 bool writeHeader,
738 bool isSymmetric,
739 bool writeMeshSize) const
740 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
741
742 /*
743 * Read a single fields in r-grid format
744 */
745 template <int D, class RFRT, class RFKT, class FFTT>
747 std::ostream &out,
748 RFRT const & field,
749 UnitCell<D> const & unitCell,
750 bool writeHeader,
751 bool isSymmetric) const
752 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
753
754 /*
755 * Read an array of fields in k-grid format
756 */
757 template <int D, class RFRT, class RFKT, class FFTT>
759 std::istream &in,
760 DArray<RFKT >& fields,
761 UnitCell<D>& unitCell) const
762 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
763
764 template <int D, class RFRT, class RFKT, class FFTT>
765 void
767 std::ostream &out,
768 DArray<RFKT > const & fields,
769 UnitCell<D> const & unitCell,
770 bool isSymmetric) const
771 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
772
773 template <int D, class RFRT, class RFKT, class FFTT>
775 DArray<double> const & in,
776 RFKT& out) const
777 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
778
779 template <int D, class RFRT, class RFKT, class FFTT>
781 RFKT const & in,
782 DArray<double>& out,
783 bool checkSymmetry,
784 double epsilon) const
785 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
786
787 /*
788 * Test if an real field DFT has the declared space group symmetry.
789 */
790 template <int D, class RFRT, class RFKT, class FFTT>
792 RFKT const & in,
793 double epsilon,
794 bool verbose) const
795 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
796
797 template <int D, class RFRT, class RFKT, class FFTT>
799 DArray<double>& field,
800 double factor) const
801 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
802
803 template <int D, class RFRT, class RFKT, class FFTT>
805 RFRT & field,
806 double factor) const
807 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
808
809 template <int D, class RFRT, class RFKT, class FFTT>
811 std::ostream &out,
812 DArray< RFRT > const & fields,
813 UnitCell<D> const & unitCell,
814 IntVec<D> const & replicas) const
815 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
816
817 /*
818 * Expand dimension of an array of r-grid fields, write to ostream.
819 */
820 template <int D, class RFRT, class RFKT, class FFTT>
822 std::ostream &out,
823 DArray<RFRT> const & fields,
824 UnitCell<D> const & unitCell,
825 int d,
826 DArray<int> const& newGridDimensions) const
827 { UTIL_THROW("Unimplemented function in FieldIoReal base class"); }
828
829} // namespace Prdc
830} // namespace Pscf
831#endif
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Description of a regular grid of points in a periodic domain.
Symmetry-adapted Fourier basis for pseudo-spectral scft.
Definition fieldIoUtil.h:21
FieldIoReal()
Constructor.
virtual void readFieldsRGridData(std::istream &in, DArray< RFRT > &fields, int nMonomer) const
Read data for array of r-grid fields, with no header section.
virtual void scaleFieldBasis(DArray< double > &field, double factor) const
Multiply a single field in basis format by a real scalar.
void scaleFieldsRGrid(DArray< RFRT > &fields, double factor) const
Scale an array of r-grid fields by a real scalar.
void writeFieldHeader(std::ostream &out, int nMonomer, UnitCell< D > const &unitCell, bool isSymmetric=true) const
Write header for field file (fortran pscf format)
virtual bool hasSymmetry(RFKT const &in, double epsilon=1.0e-8, bool verbose=true) const
Check if a k-grid field has the declared space group symmetry.
virtual void expandRGridDimension(std::ostream &out, DArray< RFRT > const &fields, UnitCell< D > const &unitCell, int d, DArray< int > const &newGridDimensions) const
Expand dimension of an array of r-grid fields, write to ostream.
virtual void writeFieldsRGrid(std::ostream &out, DArray< RFRT > 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 ostream.
virtual void replicateUnitCell(std::ostream &out, DArray< RFRT > const &fields, UnitCell< D > const &unitCell, IntVec< D > const &replicas) const
Write r-grid fields in a replicated unit cell to std::ostream.
virtual void convertBasisToKGrid(DArray< double > const &components, RFKT &dft) const
Convert a field from symmetrized basis to Fourier grid (k-grid).
virtual void convertKGridToBasis(RFKT const &in, DArray< double > &out, bool checkSymmetry=true, double epsilon=1.0e-8) const
Convert a field from Fourier (k-grid) to symmetrized basis form.
virtual void scaleFieldRGrid(RFRT &field, double factor) const
Multiply a single field in r-grid format by a real scalar.
void convertBasisToRGrid(DArray< double > const &in, RFRT &out) const
Convert a field from symmetrized basis to spatial grid (r-grid).
virtual void readFieldRGrid(std::istream &in, RFRT &field, UnitCell< D > &unitCell) const
Read single RField (field on an r-space grid) from an istream.
virtual ~FieldIoReal()
Destructor.
virtual void writeFieldsKGrid(std::ostream &out, DArray< RFKT > const &fields, UnitCell< D > const &unitCell, bool isSymmetric=true) const
Write array of RFieldDft objects (k-space fields) to file.
void convertRGridToKGrid(DArray< RFRT > const &in, DArray< RFKT > &out) const
Convert fields from spatial grid (r-grid) to k-grid format.
virtual void writeFieldRGrid(std::ostream &out, RFRT const &field, UnitCell< D > const &unitCell, bool writeHeader=true, bool isSymmetric=true) const
Write a single RField (field on an r-space grid) to ostream.
void readFieldHeader(std::istream &in, int &nMonomer, UnitCell< D > &unitCell, bool &isSymmetric) const
Reader header of field file (fortran pscf format)
void setFileMaster(FileMaster const &fileMaster)
Create an association with a FileMaster.
void convertRGridToBasis(RFRT const &in, DArray< double > &out, bool checkSymmetry=true, double epsilon=1.0e-8) const
Convert a field from spatial grid (r-grid) to symmetrized basis.
void readFieldBasis(std::istream &in, DArray< double > &field, UnitCell< D > &unitCell) const
Read single concentration or chemical potential field from file.
void readFieldsBasis(std::istream &in, DArray< DArray< double > > &fields, UnitCell< D > &unitCell) const
Read concentration or chemical potential fields from file.
virtual void readFieldsRGrid(std::istream &in, DArray< RFRT > &fields, UnitCell< D > &unitCell) const
Read array of RField objects (r-grid fields) from an istream.
void scaleFieldsBasis(DArray< DArray< double > > &fields, double factor) const
Scale an array of fields in basis format by a real scalar.
void convertKGridToRGrid(DArray< RFKT > const &in, DArray< RFRT > &out) const
Convert fields from k-grid (DFT) to real space (r-grid) format.
void associate(Mesh< D > const &mesh, FFTT const &fft, typename UnitCell< D >::LatticeSystem const &lattice, bool const &hasGroup, std::string const &groupName, SpaceGroup< D > const &group, Basis< D > &basis)
Create association with other objects in parent Domain.
virtual void readFieldsKGrid(std::istream &in, DArray< RFKT > &fields, UnitCell< D > &unitCell) const
Read array of RFieldDft objects (k-space fields) from istream.
void writeFieldsBasis(std::ostream &out, DArray< DArray< double > > const &fields, UnitCell< D > const &unitCell) const
Write concentration or chemical potential field components to file.
void writeFieldBasis(std::string filename, DArray< double > const &field, UnitCell< D > const &unitCell) const
Write single concentration or chemical potential field to file.
Crystallographic space group.
Definition FieldIoReal.h:23
bool isInitialized() const
Has this unit cell been initialized?
int nParameter() const
Get the number of parameters in the unit cell.
Base template for UnitCell<D> classes, D=1, 2 or 3.
Definition rpg/System.h:34
int capacity() const
Return allocated size.
Definition Array.h:159
Dynamically allocatable contiguous array template.
void allocate(int capacity)
Allocate the underlying C array.
Definition DArray.h:199
bool isAllocated() const
Return true if this DArray has been allocated, false otherwise.
Definition DArray.h:247
A FileMaster manages input and output files for a simulation.
Definition FileMaster.h:143
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
#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
#define UTIL_ASSERT(condition)
Assertion macro suitable for debugging serial or parallel code.
Definition global.h:75
void writeFieldHeader(std::ostream &out, int ver1, int ver2, UnitCell< D > const &cell, std::string const &groupName, int nMonomer)
Write common part of field header (fortran PSCF format).
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.
void writeNBasis(std::ostream &out, int nBasis)
Write the number of basis functions to a basis field file header.
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 inspectArrays(DArray< AT > const &arrays, int &nMonomer, int &capacity)
Inspect dimensions of a DArray of 1D arrays, each of type AT.
void inspectFields(DArray< FT > const &fields, int &nMonomer, IntVec< D > &dimensions)
Inspect dimensions of a DArray of fields, each of type FT.
void convertBasisToKGrid(DArray< double > const &components, AT &dft, Basis< D > const &basis, IntVec< D > const &dftDimensions)
Convert a real field from symmetrized basis to Fourier grid.
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 convertKGridToBasis(AT const &in, DArray< double > &out, Basis< D > const &basis, IntVec< D > const &dftDimensions, bool checkSymmetry=true, double epsilon=1.0e-8)
Convert a real field from Fourier grid to symmetrized basis.
int readNBasis(std::istream &in)
Read the number of basis functions from a basis field file header.
void writeBasisData(std::ostream &out, DArray< DArray< double > > const &fields, Basis< D > const &basis)
Write array of fields in basis format, without a header.
void readBasisData(std::istream &in, DArray< DArray< double > > &fields, UnitCell< D > const &unitCell, Mesh< D > const &mesh, Basis< D > const &basis, int nStarIn)
Read a set of fields in basis format.
void checkAllocateField(FT &field, IntVec< D > const &dimensions)
Check allocation of a single field, allocate if necessary.
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.