14#include <prdc/crystal/UnitCell.h>
15#include <prdc/crystal/SpaceGroup.h>
16#include <prdc/crystal/shiftToMinimum.h>
17#include <pscf/mesh/Mesh.h>
18#include <pscf/mesh/MeshIterator.h>
58 std::string groupName)
61 readGroup(groupName, group);
62 makeBasis(mesh, unitCell, group);
78 unitCellPtr_ = &unitCell;
82 waves_.allocate(nWave_);
83 waveIds_.allocate(nWave_);
92 bool valid = isValid();
97 isInitialized_ =
true;
111 IntVec<D> meshDimensions = mesh().dimensions();
112 std::vector< TWave<D> > twaves;
113 twaves.reserve(nWave_);
119 for (itr.begin(); !itr.atEnd(); ++itr) {
120 w.indicesDft = itr.position();
121 v = shiftToMinimum(w.indicesDft, meshDimensions, *unitCellPtr_);
123 w.sqNorm = unitCell().ksq(v);
128 TWaveNormComp<D> comp;
129 std::sort(twaves.begin(), twaves.end(), comp);
132 for (
int i = 0; i < nWave_; ++i) {
133 waves_[i].sqNorm = twaves[i].sqNorm;
134 waves_[i].indicesDft = twaves[i].indicesDft;
135 waves_[i].indicesBz = twaves[i].indicesBz;
145 void Basis<D>::makeStars(SpaceGroup<D>
const & group)
172 std::set< TWave<D>, TWaveDftComp<D> > list;
173 std::set< TWave<D>, TWaveDftComp<D> > star;
174 std::vector< TWave<D> > tempStar;
176 typename std::set< TWave<D>, TWaveDftComp<D> >::iterator rootItr;
177 typename std::set< TWave<D>, TWaveDftComp<D> >::iterator setItr;
181 Basis<D>::Star newStar;
182 std::complex<double> coeff;
187 const double epsilon = 1.0E-8;
188 IntVec<D> meshDimensions = mesh().dimensions();
190 IntVec<D> rootVecDft;
347 Gsq_max = waves_[0].sqNorm;
348 for (i = 1; i <= nWave_; ++i) {
351 bool newList =
false;
354 listSize = listEnd - listBegin;
357 Gsq = waves_[i].sqNorm;
358 if (Gsq > Gsq_max + epsilon) {
361 listSize = listEnd - listBegin;
372 for (j = listBegin; j < listEnd; ++j) {
373 wave.indicesDft = waves_[j].indicesDft;
374 wave.indicesBz = waves_[j].indicesBz;
375 wave.sqNorm = waves_[j].sqNorm;
377 UTIL_CHECK( std::abs(wave.sqNorm-waves_[j].sqNorm)
392 rootItr = list.begin();
400 while (list.size() > 0) {
402 rootVecBz = rootItr->indicesBz;
403 rootVecDft = rootItr->indicesDft;
404 Gsq = rootItr->sqNorm;
410 for (j = 0; j < group.size(); ++j) {
414 vec = rootVecBz*group[j];
417 UTIL_CHECK(std::abs(Gsq - unitCell().ksq(vec)) < epsilon);
421 wave.indicesBz = shiftToMinimum(vec, meshDimensions,
423 wave.indicesDft = vec;
424 mesh().shift(wave.indicesDft);
429 for (k = 0; k < D; ++k) {
430 wave.phase += rootVecBz[k]*(group[j].t(k));
432 while (wave.phase > 0.5) {
435 while (wave.phase <= -0.5) {
446 if (wave.indicesDft == rootVecDft) {
447 if (std::abs(wave.phase) > 1.0E-6) {
454 setItr = star.find(wave);
456 if (setItr == star.end()) {
468 phase_diff = setItr->phase - wave.phase;
469 while (phase_diff > 0.5) {
472 while (phase_diff <= -0.5) {
475 if (std::abs(phase_diff) > 1.0E-6) {
485 setItr = star.begin();
486 for ( ; setItr != star.end(); ++setItr) {
487 tempStar.push_back(*setItr);
491 TWaveBzComp<D> waveBzComp;
492 std::sort(tempStar.begin(), tempStar.end(), waveBzComp);
495 int tempStarSize = tempStar.size();
496 for (j = 0; j < tempStarSize; ++j) {
497 list.erase(tempStar[j]);
498 tempList.
append(tempStar[j]);
506 nBasisWave_ += star.size();
511 newStar.beginId = starBegin;
512 newStar.endId = newStar.beginId + star.size();
513 newStar.size = star.size();
514 newStar.cancel = cancel;
518 if (nextInvert == -1) {
523 newStar.invertFlag = -1;
524 rootItr = list.begin();
533 nVec.negate(rootVecBz);
536 (*meshPtr_).shift(nVec);
539 bool inverseFound =
false;
540 setItr = star.begin();
541 for ( ; setItr != star.end(); ++setItr) {
542 if (nVec == setItr->indicesDft) {
553 newStar.invertFlag = 0;
554 rootItr = list.begin();
562 newStar.invertFlag = 1;
569 setItr = list.begin();
570 for ( ; setItr != list.end(); ++setItr) {
571 if (nVec == setItr->indicesDft) {
583 std::cout <<
"Inverse not found for: " <<
"\n";
584 std::cout <<
" vec (ft):"
585 << rootVecDft <<
"\n";
586 std::cout <<
" vec (bz):"
588 std::cout <<
"-vec (dft):" << nVec <<
"\n";
596 stars_.append(newStar);
597 starBegin = newStar.endId;
608 for (j = 0; j < tempList.
size(); ++j) {
610 waves_[k].indicesDft = tempList[j].indicesDft;
611 waves_[k].indicesBz = tempList[j].indicesBz;
612 waves_[k].sqNorm = tempList[j].sqNorm;
613 coeff = std::complex<double>(0.0, tempList[j].phase);
615 if (std::abs(
imag(coeff)) < 1.0E-6) {
616 coeff = std::complex<double>(
real(coeff), 0.0);
618 if (std::abs(
real(coeff)) < 1.0E-6) {
619 coeff = std::complex<double>(0.0,
imag(coeff));
621 waves_[k].coeff = coeff;
635 nStar_ = stars_.size();
652 std::complex<double> rootCoeff;
653 std::complex<double> partCoeff;
654 std::complex<double> d;
656 for (i = 0; i < nStar_; ++i) {
660 if (stars_[i].invertFlag == 0) {
663 for (j = stars_[i].beginId; j < stars_[i].endId; ++j) {
664 if (waves_[j].inverseId < 0) {
667 nVec.negate(waves_[j].indicesBz);
668 (*meshPtr_).shift(nVec);
674 k = stars_[i].endId - 1 - (j - stars_[i].beginId);
675 if (nVec == waves_[k].indicesDft) {
676 waves_[j].inverseId = k;
677 waves_[k].inverseId = j;
682 for (k = j; k < stars_[i].endId; ++k) {
683 if (nVec == waves_[k].indicesDft) {
684 waves_[j].inverseId = k;
685 waves_[k].inverseId = j;
693 if (waves_[j].inverseId < 0) {
695 std::cout <<
"Inverse not found in closed star"
697 std::cout <<
"G = " << waves_[j].indicesBz
698 <<
", coeff = " << waves_[j].coeff
700 std::cout <<
"All waves in star " << i
702 for (k=stars_[i].beginId; k < stars_[i].endId; ++k) {
703 std::cout << waves_[k].indicesBz <<
" "
704 << waves_[k].coeff << std::endl;
715 rootId = stars_[i].beginId;
716 stars_[i].waveBz = waves_[rootId].indicesBz;
718 if (stars_[i].cancel) {
721 std::complex<double> czero(0.0, 0.0);
722 for (j = stars_[i].beginId; j < stars_[i].endId; ++j) {
723 waves_[j].coeff = czero;
729 partId = waves_[rootId].inverseId;
732 rootCoeff = waves_[rootId].coeff;
733 for (j = stars_[i].beginId; j < stars_[i].endId; ++j) {
734 waves_[j].coeff /= rootCoeff;
736 rootCoeff = waves_[rootId].coeff;
741 if (partId != rootId) {
743 partCoeff = waves_[partId].coeff;
744 UTIL_CHECK(std::abs(std::abs(partCoeff) - 1.0) < 1.0E-9);
745 if (std::abs(partCoeff - rootCoeff) > 1.0E-6) {
747 if (
real(d) < -1.0E-4) {
750 if (std::abs(
real(d)) <= 1.0E-4) {
755 for (j=stars_[i].beginId; j < stars_[i].endId; ++j){
756 waves_[j].coeff /= d;
766 if (stars_[i].invertFlag == 1) {
771 UTIL_CHECK(stars_[i].size == stars_[i+1].size);
772 UTIL_CHECK(stars_[i].cancel == stars_[i+1].cancel);
776 for (j = stars_[i].beginId; j < stars_[i].endId; ++j) {
777 if (waves_[j].inverseId < 0) {
780 nVec.negate(waves_[j].indicesBz);
781 (*meshPtr_).shift(nVec);
787 k = stars_[i+1].endId - 1 - (j - stars_[i].beginId);
788 if (nVec == waves_[k].indicesDft) {
789 waves_[j].inverseId = k;
790 waves_[k].inverseId = j;
795 k = stars_[i+1].beginId;
796 for ( ; k < stars_[i+1].endId; ++k) {
797 if (nVec == waves_[k].indicesDft) {
798 waves_[j].inverseId = k;
799 waves_[k].inverseId = j;
814 for (j = stars_[i+1].beginId; j < stars_[i+1].endId; ++j) {
820 rootId = stars_[i].beginId;
821 stars_[i].waveBz = waves_[rootId].indicesBz;
825 partId = waves_[rootId].inverseId;
826 stars_[i+1].waveBz = waves_[partId].indicesBz;
828 if (stars_[i].cancel) {
830 std::complex<double> czero(0.0, 0.0);
831 for (j = stars_[i].beginId; j < stars_[i].endId; ++j) {
832 waves_[j].coeff = czero;
834 for (j = stars_[i+1].beginId; j < stars_[i+1].endId; ++j) {
835 waves_[j].coeff = czero;
841 rootCoeff = waves_[rootId].coeff;
842 for (j = stars_[i].beginId; j < stars_[i].endId; ++j) {
843 waves_[j].coeff /= rootCoeff;
847 partCoeff = waves_[partId].coeff;
848 for (j = stars_[i+1].beginId; j < stars_[i+1].endId; ++j) {
849 waves_[j].coeff /= partCoeff;
862 for (i = 0; i < nStar_; ++i) {
863 double snorm = 1.0/sqrt(
double(stars_[i].size));
864 for (j = stars_[i].beginId; j < stars_[i].endId; ++j) {
865 waves_[j].coeff *= snorm;
866 waves_[j].starId = i;
871 for (i = 0; i < nWave_; ++i) {
872 if (std::abs(
real(waves_[i].coeff)) < 1.0E-8) {
874 = std::complex<double>(0.0,
imag(waves_[i].coeff));
876 if (std::abs(
imag(waves_[i].coeff)) < 1.0E-8) {
878 = std::complex<double>(
real(waves_[i].coeff), 0.0);
883 for (i = 0; i < nWave_; ++i) {
884 vec = waves_[i].indicesDft;
887 for (j = 0; j < D; ++j) {
893 if ((vec[D-1] + 1) > (meshDimensions[D-1]/2 + 1)) {
894 waves_[i].implicit =
true;
896 waves_[i].implicit =
false;
900 waveIds_[mesh().rank(vec)] = i;
904 starIds_.allocate(nBasis_);
906 for (i = 0; i < nStar_; ++i) {
907 stars_[i].starId = i;
908 if (stars_[i].cancel) {
909 stars_[i].basisId = -1;
911 stars_[i].basisId = j;
930 out <<
"N_wave" << std::endl;
932 out <<
" " << nWave_ << std::endl;
934 out <<
" " << nBasisWave_ << std::endl;
938 for (i = 0; i < nWave_; ++i) {
939 starId = waves_[i].starId;
940 if (outputAll || (!stars_[starId].cancel)) {
943 for (j = 0; j < D; ++j) {
944 out <<
Int(waves_[i].indicesBz[j], 5);
946 out <<
Int(waves_[i].starId, 6);
947 out <<
" " <<
Dbl(waves_[i].coeff.real(), 15);
948 out <<
" " <<
Dbl(waves_[i].coeff.imag(), 15);
961 out <<
"N_star" << std::endl
962 <<
" " << nStar_ << std::endl;
964 out <<
"N_basis" << std::endl
965 <<
" " << nBasis_ << std::endl;
970 for (i = 0; i < nStar_; ++i) {
971 if (outputAll || (!stars_[i].cancel)) {
972 out <<
Int(stars_[i].basisId, 6);
974 out <<
Int(stars_[i].size, 5)
975 <<
Int(stars_[i].beginId, 8)
976 <<
Int(stars_[i].endId, 8)
977 <<
Int(stars_[i].invertFlag, 4);
979 out <<
Int(stars_[i].cancel, 4);
981 for (j = 0; j < D; ++j) {
982 out <<
Int(stars_[i].waveBz[j], 6);
994 int is, ib, iw, iwp, j;
997 if (nWave_ != mesh().size()) {
998 std::cout <<
"nWave != size of mesh" << std::endl;
1007 if (wave(iw).indicesDft != v) {
1008 std::cout <<
"Inconsistent waveId and Wave::indicesDft"
1015 for (iw = 0; iw < nWave_; ++iw) {
1018 v = waves_[iw].indicesBz;
1019 Gsq = unitCell().ksq(v);
1020 if (std::abs(Gsq - waves_[iw].sqNorm) > 1.0E-8) {
1022 std::cout <<
"Incorrect sqNorm:" <<
"\n"
1023 <<
"wave.indicesBz = " <<
"\n"
1024 <<
"wave.sqNorm = " << waves_[iw].sqNorm <<
"\n"
1025 <<
"|v|^{2} = " << Gsq <<
"\n";
1031 if (v != waves_[iw].indicesDft) {
1033 std::cout <<
"shift(indicesBz) != indicesDft" << std::endl;
1038 is = waves_[iw].starId;
1039 if (iw < stars_[is].beginId) {
1041 std::cout <<
"Wave::starId < Star::beginId" << std::endl;
1044 if (iw >= stars_[is].endId) {
1046 std::cout <<
"Wave::starId >= Star::endId" << std::endl;
1051 if (waves_[iw].inverseId < 0) {
1053 std::cout <<
"Wave::inverseId not assigned\n";
1054 std::cout <<
"G = " << waves_[iw].indicesBz << std::endl;
1059 v.
negate(waves_[iw].indicesBz);
1061 iwp = waves_[iw].inverseId;
1062 if (waves_[iwp].indicesDft != v) {
1064 std::cout <<
"Wave::inverseId is not inverse" << std::endl;
1065 std::cout <<
"G = " << waves_[iw].indicesBz << std::endl;
1066 std::cout <<
"-G (from inverseId) = "
1067 << waves_[iwp].indicesBz << std::endl;
1072 if (waves_[iwp].inverseId != iw) {
1074 std::cout <<
"Wave::inverseId values do not agree\n";
1075 std::cout <<
"+G = " << waves_[iw].indicesBz << std::endl;
1076 std::cout <<
"-G = " << waves_[iwp].indicesBz << std::endl;
1081 if (waves_[iw].implicit ==
true && waves_[iwp].implicit ==
true)
1084 std::cout <<
"Wave and its inverse are both implicit";
1085 std::cout <<
"+G = " << waves_[iw].indicesBz << std::endl;
1086 std::cout <<
"-G = " << waves_[iwp].indicesBz << std::endl;
1093 for (is = 0; is < nStar_; ++is) {
1096 nWave += stars_[is].size;
1097 if (stars_[is].size != stars_[is].endId - stars_[is].beginId) {
1099 std::cout <<
"Inconsistent Star::size:" << std::endl;
1100 std::cout <<
"Star id " << is << std::endl;
1101 std::cout <<
"star size " << stars_[is].size << std::endl;
1102 std::cout <<
"Star begin " << stars_[is].beginId << std::endl;
1103 std::cout <<
"Star end " << stars_[is].endId << std::endl;
1107 if (stars_[is].beginId != stars_[is-1].endId) {
1109 std::cout <<
"Star ranges not consecutive:" << std::endl;
1110 std::cout <<
"Star id " << is << std::endl;
1111 std::cout <<
"stars_[" << is <<
"]" <<
".beginId = "
1112 << stars_[is].beginId << std::endl;
1113 std::cout <<
"stars_[" << is-1 <<
"]" <<
".endId = "
1114 << stars_[is-1].endId << std::endl;
1120 if (stars_[is].invertFlag == -1) {
1121 v.
negate(stars_[is-1].waveBz);
1122 v = shiftToMinimum(v, mesh().dimensions(), *unitCellPtr_);
1123 if (stars_[is].waveBz != v) {
1125 std::cout <<
"waveBz of star is not inverse of waveBz "
1126 <<
"of previous star" << std::endl;
1127 std::cout <<
"star id " << is << std::endl;
1128 std::cout <<
"waveBz " << stars_[is].waveBz << std::endl;
1129 std::cout <<
"waveBz (previous star) "
1130 << stars_[is-1].waveBz << std::endl;
1134 v = waves_[stars_[is].beginId].indicesBz;
1135 if (stars_[is].waveBz != v) {
1137 std::cout <<
"waveBz of star != first wave of star"
1139 std::cout <<
"star id " << is << std::endl;
1140 std::cout <<
"waveBz " << stars_[is].waveBz
1142 std::cout <<
"first wave " << v << std::endl;
1148 for (iw = stars_[is].beginId; iw < stars_[is].endId; ++iw) {
1149 if (waves_[iw].starId != is) {
1151 std::cout <<
"Inconsistent Wave::starId :" << std::endl;
1152 std::cout <<
"star id " << is << std::endl;
1153 std::cout <<
"star beginId " << stars_[is].beginId <<
"\n";
1154 std::cout <<
"star endId " << stars_[is].endId <<
"\n";
1155 std::cout <<
"wave id " << iw <<
"\n";
1156 std::cout <<
"wave starId " << waves_[iw].starId <<
"\n";
1162 if (stars_[is].starId != is) {
1164 std::cout <<
"stars_[is].starId != is for "
1165 <<
"is = " << is <<
"\n";
1170 ib = stars_[is].basisId;
1171 if (stars_[is].cancel) {
1174 std::cout <<
"basisId != -1 for cancelled star\n";
1175 std::cout <<
"star id = " << is <<
"\n";
1179 if (starIds_[ib] != is) {
1181 std::cout <<
"starIds_[stars_[is].basisId] != is for: \n";
1182 std::cout <<
"is = " << is <<
"\n";
1183 std::cout <<
"ib = stars_[is].basisId = " << ib <<
"\n";
1184 std::cout <<
"starIds_[ib] = " << starIds_[ib]
1191 for (iw = stars_[is].beginId + 1; iw < stars_[is].endId; ++iw) {
1192 if (waves_[iw].indicesBz > waves_[iw-1].indicesBz) {
1194 std::cout <<
"Failure of ordering by indicesB within star"
1198 if (waves_[iw].indicesBz == waves_[iw-1].indicesBz) {
1200 std::cout <<
"Equal values of indicesBz within star"
1207 if (stars_[is].cancel) {
1208 for (iw = stars_[is].beginId + 1; iw < stars_[is].endId; ++iw)
1210 if (std::abs(waves_[iw].coeff) > 1.0E-8) {
1212 std::cout <<
"Nonzero coefficient in a cancelled star"
1214 std::cout <<
"G = " << waves_[iw].indicesBz
1215 <<
" coeff = " << waves_[iw].coeff
1225 if (stars_[nStar_-1].endId != mesh().size()) {
1227 std::cout <<
"Star endId of last star != mesh size" << std::endl;
1230 if (nWave != mesh().size()) {
1232 std::cout <<
"Sum of star sizes != mesh size" << std::endl;
1238 std::complex<double> cdel;
1241 while (is < nStar_) {
1242 cancel = stars_[is].cancel;
1244 if (stars_[is].invertFlag == 0) {
1247 int begin = stars_[is].beginId;
1248 int end = stars_[is].endId;
1249 for (iw = begin; iw < end; ++iw) {
1250 iwp = waves_[iw].inverseId;
1251 if (waves_[iwp].starId != is) {
1253 std::cout <<
"Inverse not found in closed star"
1255 std::cout <<
"G = " << waves_[iw].indicesBz
1256 <<
"coeff = " << waves_[iw].coeff
1258 std::cout <<
"All waves in star " << is <<
"\n";
1259 for (j=begin; j < end; ++j) {
1260 std::cout << waves_[j].indicesBz <<
" "
1261 << waves_[j].coeff <<
"\n";
1266 cdel = std::conj(waves_[iwp].coeff);
1267 cdel -= waves_[iw].coeff;
1268 if (std::abs(cdel) > 1.0E-8) {
1270 std::cout <<
"Function for closed star is not real:"
1272 std::cout <<
"+G = " << waves_[iw].indicesBz
1273 <<
" coeff = " << waves_[iw].coeff
1275 std::cout <<
"-G = " << waves_[iwp].indicesBz
1276 <<
" coeff = " << waves_[iwp].coeff
1278 std::cout <<
"Coefficients are not conjugates."
1280 std::cout <<
"All waves in star " << is <<
"\n";
1281 for (j=begin; j < end; ++j) {
1282 std::cout << waves_[j].indicesBz <<
" "
1283 << waves_[j].coeff <<
"\n";
1297 if (stars_[is].invertFlag != 1) {
1299 std::cout <<
"Expected invertFlag == 1" << std::endl;
1302 if (stars_[is+1].invertFlag != -1) {
1304 std::cout <<
"Expected invertFlag == -1" << std::endl;
1307 if (stars_[is+1].size != stars_[is].size) {
1309 std::cout <<
"Partner stars of different size" << std::endl;
1312 if (stars_[is+1].cancel != stars_[is].cancel) {
1314 std::cout <<
"Partners stars with different cancel flags"
1320 int begin1 = stars_[is].beginId;
1321 int end1 = stars_[is].endId;
1322 int begin2 = stars_[is+1].beginId;
1323 int end2 = stars_[is+1].endId;
1329 for (iw = begin1; iw < end1; ++iw) {
1330 iwp = waves_[iw].inverseId;
1331 if (waves_[iwp].starId != is + 1) {
1333 std::cout <<
"Inverse not found for G in open star"
1335 std::cout <<
"First star id = " << is << std::endl;
1336 std::cout <<
"+G = " << waves_[iw].indicesBz
1337 <<
"coeff = " << waves_[iw].coeff
1339 std::cout <<
"Waves in star " << is
1340 <<
" (starInvert ==1):" <<
"\n";
1341 for (j = begin1; j < end1; ++j) {
1342 std::cout << waves_[j].indicesBz <<
" "
1343 << waves_[j].coeff <<
"\n";
1345 std::cout <<
"Waves in star " << is+1
1346 <<
" (starInvert == -1):" <<
"\n";
1347 for (j=begin2; j < end2; ++j) {
1348 std::cout << waves_[j].indicesBz <<
" "
1349 << waves_[j].coeff <<
"\n";
1354 cdel = std::conj(waves_[iwp].coeff);
1355 cdel -= waves_[iw].coeff;
1356 if (std::abs(cdel) > 1.0E-8) {
1358 std::cout <<
"Error of coefficients in open stars:"
1360 std::cout <<
"First star id = " << is << std::endl;
1361 std::cout <<
"+G = " << waves_[iw].indicesBz
1362 <<
" coeff = " << waves_[iw].coeff
1364 std::cout <<
"-G = " << waves_[iwp].indicesBz
1365 <<
" coeff = " << waves_[iwp].coeff
1367 std::cout <<
"Coefficients are not conjugates."
1369 std::cout <<
"Waves in star " << is
1370 <<
" (starInvert ==1):" <<
"\n";
1371 for (j = begin1; j < end1; ++j) {
1372 std::cout << waves_[j].indicesBz <<
" "
1373 << waves_[j].coeff <<
"\n";
1375 std::cout <<
"Waves in star " << is+1
1376 <<
" (starInvert == -1):" <<
"\n";
1377 for (j=begin2; j < end2; ++j) {
1378 std::cout << waves_[j].indicesBz <<
" "
1379 << waves_[j].coeff <<
"\n";
1394 for (ib = 0; ib < nBasis_; ++ib) {
1396 if (stars_[is].cancel) {
1398 std::cout <<
"Star referred to by starIds_ is cancelled\n";
1401 if (stars_[is].basisId != ib) {
1403 std::cout <<
"Error: stars_[starIds_[ib]].basisId != ib\n";
1404 std::cout <<
"Basis function index ib = " << ib <<
"\n";
1405 std::cout <<
"is = starIds_[ib] = " << is <<
"\n";
1406 std::cout <<
"stars_[is].basisId = "
1407 << stars_[is].basisId <<
"\n";
An IntVec<D, T> is a D-component vector of elements of integer type T.
Iterator over points in a Mesh<D>.
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)?
IntVec< D > position() const
Get current position in the grid, as integer vector.
Description of a regular grid of points in a periodic domain.
IntVec< D > dimensions() const
Get an IntVec<D> of the grid dimensions.
int size() const
Get total number of grid points.
Symmetry-adapted Fourier basis for pseudo-spectral scft.
int nBasis() const
Total number of nonzero symmetry-adapted basis functions.
Basis()
Default constructor.
void makeBasis(Mesh< D > const &mesh, UnitCell< D > const &unitCell, SpaceGroup< D > const &group)
Construct basis for a specific mesh and space group.
void outputWaves(std::ostream &out, bool outputAll=false) const
Print a list of all waves to an output stream.
bool isValid() const
Returns true if valid, false otherwise.
void outputStars(std::ostream &out, bool outputAll=false) const
Print a list of all stars to an output stream.
Crystallographic space group.
void checkMeshDimensions(IntVec< D > const &dimensions) const
Check if input mesh dimensions are compatible with space group.
Base template for UnitCell<D> classes, D=1, 2 or 3.
Vec< D, T > & negate(const Vec< D, T > &v)
Return negative of vector v.
static const double Pi
Trigonometric constant Pi.
Wrapper for a double precision number, for formatted ostream output.
An automatically growable array, analogous to a std::vector.
int size() const
Return logical size of this array (i.e., current number of elements).
void clear()
Reset to empty state.
void append(Data const &data)
Append an element to the end of the sequence.
Wrapper for an int, 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.
RT imag(CT const &a)
Return the imaginary part of a complex number.
RT real(CT const &a)
Return the real part of a complex number.
PSCF package top-level namespace.
Simple wave struct for use within Basis construction.