Program Listing for File bz_trellis.hpp¶
↰ Return to documentation for file (src/bz_trellis.hpp)
/* This file is part of brille.
Copyright © 2019,2020 Greg Tucker <greg.tucker@stfc.ac.uk>
brille is free software: you can redistribute it and/or modify it under the
terms of the GNU Affero General Public License as published by the Free
Software Foundation, either version 3 of the License, or (at your option)
any later version.
brille is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with brille. If not, see <https://www.gnu.org/licenses/>. */
#ifndef BRILLE_BZ_TRELLIS_
#define BRILLE_BZ_TRELLIS_
#include "bz.hpp"
#include "trellis.hpp"
#include "phonon.hpp"
namespace brille {
template<class T, class R>
class BrillouinZoneTrellis3: public PolyhedronTrellis<T,R>{
using SuperClass = PolyhedronTrellis<T,R>;
BrillouinZone brillouinzone;
public:
template<typename... A>
BrillouinZoneTrellis3(const BrillouinZone& bz, A... args):
SuperClass(bz.get_ir_polyhedron(), args...),
brillouinzone(bz) {}
BrillouinZone get_brillouinzone(void) const {return this->brillouinzone;}
bArray<double> get_xyz(void) const {return this->vertices();}
bArray<double> get_inner_xyz(void) const {return this->cube_vertices(); }
bArray<double> get_outer_xyz(void) const {return this->poly_vertices(); }
bArray<double> get_hkl(void) const { return xyz_to_hkl(brillouinzone.get_lattice(),this->vertices());}
bArray<double> get_inner_hkl(void) const {return xyz_to_hkl(brillouinzone.get_lattice(),this->cube_vertices()); }
bArray<double> get_outer_hkl(void) const {return xyz_to_hkl(brillouinzone.get_lattice(),this->poly_vertices()); }
std::vector<std::array<brille::ind_t,4>> get_vertices_per_tetrahedron(void) const {return this->vertices_per_tetrahedron();}
template<class S>
std::tuple<brille::Array<T>,brille::Array<R>>
interpolate_at(const LQVec<S>& x, const int nth, const bool no_move=false) const {
profile_update("Start BrillouinZoneTrellis3::interpolate_at");
LQVec<S> q(x.get_lattice(), x.size(0));
LQVec<int> tau(x.get_lattice(), x.size(0));
if (no_move){
// Special mode for testing where no specified points are moved
// IT IS IMPERITIVE THAT THE PROVIDED POINTS ARE *INSIDE* THE IRREDUCIBLE
// POLYHEDRON otherwise the interpolation will fail or give garbage back.
q = x;
} else if (!brillouinzone.moveinto(x, q, tau, nth)){
std::string msg;
msg = "Moving all points into the first Brillouin zone failed.";
throw std::runtime_error(msg);
}
if (nth < 2)
return this->SuperClass::interpolate_at(q.get_xyz());
return this->SuperClass::interpolate_at(q.get_xyz(), nth);
}
template<class S>
std::tuple<brille::Array<T>,brille::Array<R>>
ir_interpolate_at(const LQVec<S>& x, const int nth, const bool no_move=false) const {
verbose_update("BZTrellisQ::ir_interpoalte_at called with ",nth," threads");
profile_update("Start BrillouinZoneTrellis3::ir_interpolate_at");
LQVec<S> ir_q(x.get_lattice(), x.size(0));
LQVec<int> tau(x.get_lattice(), x.size(0));
std::vector<size_t> rot(x.size(0),0u), invrot(x.size(0),0u);
if (no_move){
// Special mode for testing where no specified points are moved
// IT IS IMPERITIVE THAT THE PROVIDED POINTS ARE *INSIDE* THE IRREDUCIBLE
// POLYHEDRON otherwise the interpolation will fail or give garbage back.
ir_q = x;
} else if (!brillouinzone.ir_moveinto(x, ir_q, tau, rot, invrot, nth)){
std::string msg;
msg = "Moving all points into the irreducible Brillouin zone failed.";
throw std::runtime_error(msg);
}
auto [vals, vecs] = (nth>1)
? this->SuperClass::interpolate_at(ir_q.get_xyz(), nth)
: this->SuperClass::interpolate_at(ir_q.get_xyz());
profile_update("Apply rotations/permutations to interpolated results");
// we always need the pointgroup operations to 'rotate'
PointSymmetry psym = brillouinzone.get_pointgroup_symmetry();
// and might need the Phonon Gamma table
GammaTable pgt{GammaTable()};
if (RotatesLike::Gamma == this->data().vectors().rotateslike()){
pgt.construct(brillouinzone.get_lattice().star(), brillouinzone.add_time_reversal());
}
// make data-sharing Array2 objects for the rotation functions
brille::Array2<T> vals2(vals);
brille::Array2<R> vecs2(vecs);
// actually perform the rotation to Q
this->data().values() .rotate_in_place(vals2, ir_q, pgt, psym, rot, invrot, nth);
this->data().vectors().rotate_in_place(vecs2, ir_q, pgt, psym, rot, invrot, nth);
// we're done so bundle the output
profile_update(" End BrillouinZoneTrellis3::ir_interpolate_at");
return std::make_tuple(vals, vecs);
}
};
} // end namespace brille
#endif // _BZ_TRELLIS_