-
Notifications
You must be signed in to change notification settings - Fork 139
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #36 from PhilipFackler/refactor-test_RotatedSPOs
Refactor test_RotatedSPOs
- Loading branch information
Showing
59 changed files
with
13,914 additions
and
3,774 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,314 @@ | ||
////////////////////////////////////////////////////////////////////////////////////// | ||
// This file is distributed under the University of Illinois/NCSA Open Source | ||
// License. See LICENSE file in top directory for details. | ||
// | ||
// Copyright (c) 2020 QMCPACK developers. | ||
// | ||
// File developed by: Jordan E. Vincent, University of Illinois at | ||
// Urbana-Champaign | ||
// Luke Shulenburger, [email protected], Sandia National | ||
// Laboratories Jeremy McMinnis, [email protected], | ||
// University of Illinois at Urbana-Champaign Jeongnim Kim, | ||
// [email protected], University of Illinois at | ||
// Urbana-Champaign Miguel Morales, [email protected], | ||
// Lawrence Livermore National Laboratory Mark Dewing, | ||
// [email protected], University of Illinois at | ||
// Urbana-Champaign Mark A. Berrill, [email protected], Oak | ||
// Ridge National Laboratory | ||
// | ||
// File created by: Jeongnim Kim, [email protected], University of Illinois | ||
// at Urbana-Champaign | ||
////////////////////////////////////////////////////////////////////////////////////// | ||
|
||
#include "InitMolecularSystemT.h" | ||
|
||
#include "OhmmsData/AttributeSet.h" | ||
#include "Particle/DistanceTableT.h" | ||
#include "Particle/ParticleSetPoolT.h" | ||
#include "ParticleBase/RandomSeqGeneratorGlobal.h" | ||
|
||
namespace qmcplusplus | ||
{ | ||
template <typename T> | ||
InitMolecularSystemT<T>::InitMolecularSystemT( | ||
ParticleSetPoolT<T>& pset, const char* aname) : | ||
OhmmsElementBase(aname), | ||
ptclPool(pset) | ||
{ | ||
} | ||
|
||
template <typename T> | ||
bool | ||
InitMolecularSystemT<T>::put(xmlNodePtr cur) | ||
{ | ||
std::string target("e"), source("i"), volume("no"); | ||
OhmmsAttributeSet hAttrib; | ||
hAttrib.add(target, "target"); | ||
hAttrib.add(source, "source"); | ||
hAttrib.add(volume, "use_volume"); | ||
hAttrib.put(cur); | ||
ParticleSetT<T>* els = ptclPool.getParticleSet(target); | ||
if (els == 0) { | ||
ERRORMSG("No target particle " << target << " exists.") | ||
return false; | ||
} | ||
ParticleSetT<T>* ions = ptclPool.getParticleSet(source); | ||
if (ions == 0) { | ||
ERRORMSG("No source particle " << source << " exists.") | ||
return false; | ||
} | ||
|
||
app_log() << "<init source=\"" << source << "\" target=\"" << target | ||
<< "\">" << std::endl; | ||
|
||
if (volume == "yes") | ||
initWithVolume(ions, els); | ||
else | ||
initMolecule(ions, els); | ||
|
||
makeUniformRandom(els->spins); | ||
els->spins *= 2 * M_PI; | ||
|
||
app_log() << "</init>" << std::endl; | ||
app_log().flush(); | ||
|
||
return true; | ||
} | ||
|
||
template <typename T> | ||
void | ||
InitMolecularSystemT<T>::initAtom(ParticleSetT<T>* ions, ParticleSetT<T>* els) | ||
{ | ||
// 3N-dimensional Gaussian | ||
typename ParticleSetT<T>::ParticlePos chi(els->getTotalNum()); | ||
makeGaussRandom(chi); | ||
RealType q = std::sqrt(static_cast<RealType>(els->getTotalNum())) * 0.5; | ||
int nel(els->getTotalNum()), items(0); | ||
while (nel) { | ||
els->R[items] = ions->R[0] + q * chi[items]; | ||
--nel; | ||
++items; | ||
} | ||
} | ||
|
||
template <typename TReal> | ||
struct LoneElectronT | ||
{ | ||
using RealType = TReal; | ||
int ID; | ||
RealType BondLength; | ||
inline LoneElectronT(int id, RealType bl) : ID(id), BondLength(bl) | ||
{ | ||
} | ||
}; | ||
|
||
template <typename T> | ||
void | ||
InitMolecularSystemT<T>::initMolecule( | ||
ParticleSetT<T>* ions, ParticleSetT<T>* els) | ||
{ | ||
if (ions->getTotalNum() == 1) | ||
return initAtom(ions, els); | ||
|
||
const int d_ii_ID = ions->addTable(*ions); | ||
ions->update(); | ||
const typename ParticleSetT<T>::ParticleIndex& grID(ions->GroupID); | ||
SpeciesSet& Species(ions->getSpeciesSet()); | ||
int Centers = ions->getTotalNum(); | ||
std::vector<int> Qtot(Centers), Qcore(Centers), Qval(Centers, 0); | ||
// use charge as the core electrons first | ||
int icharge = Species.addAttribute("charge"); | ||
// Assign default core charge | ||
for (int iat = 0; iat < Centers; iat++) | ||
Qtot[iat] = static_cast<int>(Species(icharge, grID[iat])); | ||
// cutoff radius (Bohr) this a random choice | ||
RealType cutoff = 4.0; | ||
typename ParticleSetT<T>::ParticlePos chi(els->getTotalNum()); | ||
// makeGaussRandom(chi); | ||
makeSphereRandom(chi); | ||
// the upper limit of the electron index with spin up | ||
const int numUp = els->last(0); | ||
// the upper limit of the electron index with spin down. Pay attention to | ||
// the no spin down electron case. | ||
const int numDown = els->last(els->groups() > 1 ? 1 : 0) - els->first(0); | ||
// consumer counter of random numbers chi | ||
int random_number_counter = 0; | ||
int nup_tot = 0, ndown_tot = numUp; | ||
std::vector<LoneElectronT<RealType>> loneQ; | ||
RealType rmin = cutoff; | ||
typename ParticleSetT<T>::SingleParticlePos cm; | ||
|
||
const auto& dist = ions->getDistTableAA(d_ii_ID).getDistances(); | ||
// Step 1. Distribute even Q[iat] of atomic center iat. If Q[iat] is odd, | ||
// put Q[iat]-1 and save the lone electron. | ||
for (size_t iat = 0; iat < Centers; iat++) { | ||
cm += ions->R[iat]; | ||
for (size_t jat = iat + 1; jat < Centers; ++jat) { | ||
rmin = std::min(rmin, dist[jat][iat]); | ||
} | ||
// use 40% of the minimum bond | ||
RealType sep = rmin * 0.4; | ||
int v2 = Qtot[iat] / 2; | ||
if (Qtot[iat] > v2 * 2) { | ||
loneQ.push_back(LoneElectronT<RealType>(iat, sep)); | ||
} | ||
for (int k = 0; k < v2; k++) { | ||
// initialize electron positions in pairs | ||
if (nup_tot < numUp) | ||
els->R[nup_tot++] = | ||
ions->R[iat] + sep * chi[random_number_counter++]; | ||
if (ndown_tot < numDown) | ||
els->R[ndown_tot++] = | ||
ions->R[iat] + sep * chi[random_number_counter++]; | ||
} | ||
} | ||
|
||
// Step 2. Distribute the electrons left alone | ||
// mmorales: changed order of spin assignment to help with spin | ||
// imbalances in molecules at large distances. | ||
// Not guaranteed to work, but should help in most cases | ||
// as long as atoms in molecules are defined sequencially | ||
typename std::vector<LoneElectronT<RealType>>::iterator it(loneQ.begin()); | ||
typename std::vector<LoneElectronT<RealType>>::iterator it_end(loneQ.end()); | ||
while (it != it_end && nup_tot != numUp && ndown_tot != numDown) { | ||
if (nup_tot < numUp) { | ||
els->R[nup_tot++] = ions->R[(*it).ID] + | ||
(*it).BondLength * chi[random_number_counter++]; | ||
++it; | ||
} | ||
if (ndown_tot < numDown && it != it_end) { | ||
els->R[ndown_tot++] = ions->R[(*it).ID] + | ||
(*it).BondLength * chi[random_number_counter++]; | ||
++it; | ||
} | ||
} | ||
|
||
// Step 3. Handle more than neutral electrons | ||
// extra electrons around the geometric center | ||
RealType cnorm = 1.0 / static_cast<RealType>(Centers); | ||
RealType sep = rmin * 2; | ||
cm = cnorm * cm; | ||
if (nup_tot < numUp) | ||
while (nup_tot < numUp) | ||
els->R[nup_tot++] = cm + sep * chi[random_number_counter++]; | ||
if (ndown_tot < numDown) | ||
while (ndown_tot < numDown) | ||
els->R[ndown_tot++] = cm + sep * chi[random_number_counter++]; | ||
|
||
// safety check. all the random numbers should have been consumed once and | ||
// only once. | ||
if (random_number_counter != chi.size()) | ||
throw std::runtime_error("initMolecule unexpected random number " | ||
"consumption. Please report a bug!"); | ||
|
||
// put all the electrons in a unit box | ||
if (els->getLattice().SuperCellEnum != SUPERCELL_OPEN) { | ||
els->R.setUnit(PosUnit::Cartesian); | ||
els->applyBC(els->R); | ||
els->update(false); | ||
} | ||
} | ||
|
||
/// helper function to determine the lower bound of a domain (need to move up) | ||
template <typename T> | ||
inline TinyVector<T, 3> | ||
lower_bound(const TinyVector<T, 3>& a, const TinyVector<T, 3>& b) | ||
{ | ||
return TinyVector<T, 3>( | ||
std::min(a[0], b[0]), std::min(a[1], b[1]), std::min(a[2], b[2])); | ||
} | ||
|
||
/// helper function to determine the upper bound of a domain (need to move up) | ||
template <typename T> | ||
inline TinyVector<T, 3> | ||
upper_bound(const TinyVector<T, 3>& a, const TinyVector<T, 3>& b) | ||
{ | ||
return TinyVector<T, 3>( | ||
std::max(a[0], b[0]), std::max(a[1], b[1]), std::max(a[2], b[2])); | ||
} | ||
|
||
template <typename T> | ||
void | ||
InitMolecularSystemT<T>::initWithVolume( | ||
ParticleSetT<T>* ions, ParticleSetT<T>* els) | ||
{ | ||
TinyVector<RealType, OHMMS_DIM> start(1.0); | ||
TinyVector<RealType, OHMMS_DIM> end(0.0); | ||
|
||
typename ParticleSetT<T>::ParticlePos Ru(ions->getTotalNum()); | ||
Ru.setUnit(PosUnit::Lattice); | ||
ions->applyBC(ions->R, Ru); | ||
|
||
for (int iat = 0; iat < Ru.size(); iat++) { | ||
start = lower_bound(Ru[iat], start); | ||
end = upper_bound(Ru[iat], end); | ||
} | ||
|
||
TinyVector<RealType, OHMMS_DIM> shift; | ||
Tensor<RealType, OHMMS_DIM> newbox(ions->getLattice().R); | ||
|
||
RealType buffer = 2.0; // buffer 2 bohr | ||
for (int idim = 0; idim < OHMMS_DIM; ++idim) { | ||
// if(ions->getLattice().BoxBConds[idim]) | ||
//{ | ||
// start[idim]=0.0; | ||
// end[idim]=1.0; | ||
// shift[idim]=0.0; | ||
// } | ||
// else | ||
{ | ||
RealType buffer_r = buffer * ions->getLattice().OneOverLength[idim]; | ||
start[idim] = std::max((RealType)0.0, (start[idim] - buffer_r)); | ||
end[idim] = std::min((RealType)1.0, (end[idim] + buffer_r)); | ||
shift[idim] = start[idim] * ions->getLattice().Length[idim]; | ||
if (std::abs(end[idim] = start[idim]) < | ||
buffer) { // handle singular case | ||
start[idim] = std::max(0.0, start[idim] - buffer_r / 2.0); | ||
end[idim] = std::min(1.0, end[idim] + buffer_r / 2.0); | ||
} | ||
|
||
newbox(idim, idim) = | ||
(end[idim] - start[idim]) * ions->getLattice().Length[idim]; | ||
} | ||
} | ||
|
||
typename ParticleSetT<T>::ParticleLayout slattice(ions->getLattice()); | ||
slattice.set(newbox); | ||
|
||
app_log() << " InitMolecularSystem::initWithVolume " << std::endl; | ||
app_log() << " Effective Lattice shifted by " << shift << std::endl; | ||
app_log() << newbox << std::endl; | ||
|
||
Ru.resize(els->getTotalNum()); | ||
makeUniformRandom(Ru); | ||
for (int iat = 0; iat < Ru.size(); ++iat) | ||
els->R[iat] = slattice.toCart(Ru[iat]) + shift; | ||
els->R.setUnit(PosUnit::Cartesian); | ||
} | ||
|
||
template <typename T> | ||
bool | ||
InitMolecularSystemT<T>::put(std::istream& is) | ||
{ | ||
return true; | ||
} | ||
|
||
template <typename T> | ||
bool | ||
InitMolecularSystemT<T>::get(std::ostream& os) const | ||
{ | ||
return true; | ||
} | ||
|
||
template <typename T> | ||
void | ||
InitMolecularSystemT<T>::reset() | ||
{ | ||
} | ||
|
||
template class InitMolecularSystemT<double>; | ||
template class InitMolecularSystemT<float>; | ||
template class InitMolecularSystemT<std::complex<double>>; | ||
template class InitMolecularSystemT<std::complex<float>>; | ||
|
||
} // namespace qmcplusplus |
Oops, something went wrong.