3
0
mirror of https://github.com/triqs/dft_tools synced 2025-01-10 21:18:22 +01:00

Revision of Wannier90Converter:

1) write the Hamiltonian in Bloch space for charge self-consistent calculations, and 2) spin-orbit coupling

if bloch_basis = True:
* if "seedname_u.dat" (and in case of disentanglement "seedname_u_dis.dat")
  present, write hopping in Bloch basis
* "proj_mat" transforming from Bloch to orbital space
* diagonal hoppings are directly read from "seedname.eig"
* fermi weights and band_window of Wannier Hamiltonian are read from DFT
  output and "seedname.nnkp", written into new subgroup "dft_misc_input"
* automatic calculation of "density_required"
* implemented for Quantum Espresso (read from "seedname.nscf.out" if
  verbosity = 'high') and VASP (read from "OUTCAR"/"LOCPROJ")

* spin-orbit coupling SO = 1 implemented
* substitute k_mesh and bz_weights with kpts and kpt_weights,
  respectively (previous names kept for compatibility)
* updated tests
This commit is contained in:
Alexander Hampel 2021-02-05 18:08:46 -05:00 committed by GitHub
commit 92e24a9ada
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 427 additions and 108 deletions

View File

@ -24,14 +24,15 @@
# Wannier90 to HDF5 converter for the SumkDFT class of dfttools/TRIQS; # Wannier90 to HDF5 converter for the SumkDFT class of dfttools/TRIQS;
# #
# written by Gabriele Sclauzero (Materials Theory, ETH Zurich), Dec 2015 -- Jan 2016, # written by Gabriele Sclauzero (Materials Theory, ETH Zurich), Dec 2015 -- Jan 2016,
# and updated by Maximilian Merkel (Materials Theory, ETH Zurich), Aug 2020, # updated by Maximilian Merkel (Materials Theory, ETH Zurich), Aug 2020 -- Jan 2021,
# and by Sophie Beck (Materials Theory, ETH Zurich), Sep 2020 -- Jan 2021,
# under the supervision of Claude Ederer (Materials Theory). # under the supervision of Claude Ederer (Materials Theory).
# Partially based on previous work by K. Dymkovski and the DFT_tools/TRIQS team. # Partially based on previous work by K. Dymkovski and the DFT_tools/TRIQS team.
# #
# Limitations of the current implementation: # Limitations of the current implementation:
# - the case with SO=1 is not considered at the moment
# - the T rotation matrices are not used in this implementation # - the T rotation matrices are not used in this implementation
# - projectors for uncorrelated shells (proj_mat_all) cannot be set # - in bloch_basis mode, the number of wannier functions per shell has to be equal
# to the dim of the shell
# #
# Things to be improved/checked: # Things to be improved/checked:
# - the case with SP=1 might work, but was never tested (do we need to define # - the case with SP=1 might work, but was never tested (do we need to define
@ -46,11 +47,11 @@
import numpy import numpy
import math
from h5 import *
from .converter_tools import *
from itertools import product
import os.path import os.path
from itertools import product
from h5 import HDFArchive
from .converter_tools import ConverterTools
import triqs.utility.mpi as mpi import triqs.utility.mpi as mpi
class Wannier90Converter(ConverterTools): class Wannier90Converter(ConverterTools):
@ -59,8 +60,8 @@ class Wannier90Converter(ConverterTools):
""" """
def __init__(self, seedname, hdf_filename=None, dft_subgrp='dft_input', def __init__(self, seedname, hdf_filename=None, dft_subgrp='dft_input',
symmcorr_subgrp='dft_symmcorr_input', repacking=False, symmcorr_subgrp='dft_symmcorr_input', misc_subgrp='dft_misc_input',
rot_mat_type='hloc_diag'): repacking=False, rot_mat_type='hloc_diag', bloch_basis=False):
""" """
Initialise the class. Initialise the class.
@ -74,11 +75,15 @@ class Wannier90Converter(ConverterTools):
Name of subgroup storing necessary DFT data Name of subgroup storing necessary DFT data
symmcorr_subgrp : string, optional symmcorr_subgrp : string, optional
Name of subgroup storing correlated-shell symmetry data Name of subgroup storing correlated-shell symmetry data
misc_subgrp : string, optional
Name of subgroup storing miscellaneous DFT data.
repacking : boolean, optional repacking : boolean, optional
Does the hdf5 archive need to be repacked to save space? Does the hdf5 archive need to be repacked to save space?
rot_mat_type : string, optional rot_mat_type : string, optional
Type of rot_mat used Type of rot_mat used
Can be 'hloc_diag', 'wannier', 'none' Can be 'hloc_diag', 'wannier', 'none'
bloch_basis : boolean, optional
Should the Hamiltonian be written in Bloch rather than Wannier basis?
""" """
self._name = "Wannier90Converter" self._name = "Wannier90Converter"
@ -93,11 +98,13 @@ class Wannier90Converter(ConverterTools):
self.w90_seed = seedname self.w90_seed = seedname
self.dft_subgrp = dft_subgrp self.dft_subgrp = dft_subgrp
self.symmcorr_subgrp = symmcorr_subgrp self.symmcorr_subgrp = symmcorr_subgrp
self.misc_subgrp = misc_subgrp
self.fortran_to_replace = {'D': 'E'} self.fortran_to_replace = {'D': 'E'}
# threshold below which matrix elements from wannier90 should be # threshold below which matrix elements from wannier90 should be
# considered equal # considered equal
self._w90zero = 2.e-6 self._w90zero = 2.e-6
self.rot_mat_type = rot_mat_type self.rot_mat_type = rot_mat_type
self.bloch_basis = bloch_basis
if self.rot_mat_type not in ('hloc_diag', 'wannier', 'none'): if self.rot_mat_type not in ('hloc_diag', 'wannier', 'none'):
raise ValueError('Parameter rot_mat_type invalid, should be one of' raise ValueError('Parameter rot_mat_type invalid, should be one of'
+ '"hloc_diag", "wannier", "none"') + '"hloc_diag", "wannier", "none"')
@ -113,14 +120,14 @@ class Wannier90Converter(ConverterTools):
- dft_subgrp - dft_subgrp
- symmcorr_subgrp - symmcorr_subgrp
in the hdf5 archive. in the hdf5 archive.
""" """
# Read and write only on the master node # Read and write only on the master node
if not (mpi.is_master_node()): if not (mpi.is_master_node()):
return return
mpi.report("Reading input from %s..." % self.inp_file) mpi.report("\nReading input from %s..." % self.inp_file)
# R is a generator : each R.Next() will return the next number in the # R is a generator : each R.Next() will return the next number in the
# file # file
@ -139,8 +146,10 @@ class Wannier90Converter(ConverterTools):
else: else:
# some default grid, if everything else fails... # some default grid, if everything else fails...
nki = [8, 8, 8] nki = [8, 8, 8]
# read the total number of electrons per cell # read the total number of electrons per cell if not in bloch basis
# in bloch basis, this is later calculated from the partial occupations
density_required = float(next(R)) density_required = float(next(R))
# we do not read shells, because we have no additional shells beyond correlated ones, # we do not read shells, because we have no additional shells beyond correlated ones,
# and the data will be copied from corr_shells into shells (see below) # and the data will be copied from corr_shells into shells (see below)
# number of corr. shells (e.g. Fe d, Ce f) in the unit cell, # number of corr. shells (e.g. Fe d, Ce f) in the unit cell,
@ -149,6 +158,7 @@ class Wannier90Converter(ConverterTools):
# l, dim, SO flag, irep): # l, dim, SO flag, irep):
corr_shells = [{name: int(val) for name, val in zip( corr_shells = [{name: int(val) for name, val in zip(
corr_shell_entries, R)} for icrsh in range(n_corr_shells)] corr_shell_entries, R)} for icrsh in range(n_corr_shells)]
try: try:
self.fermi_energy = float(next(R)) self.fermi_energy = float(next(R))
except: except:
@ -169,17 +179,40 @@ class Wannier90Converter(ConverterTools):
for ish in range(n_shells): for ish in range(n_shells):
shells.append({key: corr_shells[ish].get( shells.append({key: corr_shells[ish].get(
key, None) for key in shell_entries}) key, None) for key in shell_entries})
###
SP = 0 # NO spin-polarised calculations for now # Determine if any shell requires SO
SO = 0 # NO spin-orbit calculation for now if any([corr_shell['SO']==1 for corr_shell in corr_shells]):
SO = 1
SP = 1
print('Spin-orbit interaction turned on')
else:
SO = 0
SP = 0
# Only one block supported - either non-spin-polarized or spin-orbit coupled
assert SP == SO, 'Spin-polarized calculations not implemented'
charge_below = 0 # total charge below energy window NOT used for now charge_below = 0 # total charge below energy window NOT used for now
energy_unit = 1.0 # should be understood as eV units energy_unit = 1.0 # should be understood as eV units
###
# this is more general # this is more general
n_spin = SP + 1 - SO n_spin_blocs = SP + 1 - SO
assert n_spin_blocs > 0, 'Input error, if SO=1, SP must be 1.'
if SO == 1:
for shell_list in [shells, corr_shells]:
for entry in shell_list:
entry['dim'] *= 2
dim_corr_shells = sum([sh['dim'] for sh in corr_shells]) dim_corr_shells = sum([sh['dim'] for sh in corr_shells])
mpi.report( mpi.report('Total number of WFs expected in the correlated shells: {0:d}'.format(dim_corr_shells))
"Total number of WFs expected in the correlated shells: %d" % dim_corr_shells)
# build the k-point mesh, if its size was given on input (kmesh_mode >= 0),
# otherwise it is built according to the data in the hr file (see below)
# If output is in bloch_basis, we use k mesh from seedname_u.mat for consistency
if kmesh_mode >= 0 and not self.bloch_basis:
n_k, kpts, kpt_weights = self.kmesh_build(nki, kmesh_mode)
self.n_k = n_k
self.kpts = kpts
# determine the number of inequivalent correlated shells and maps, # determine the number of inequivalent correlated shells and maps,
# needed for further processing # needed for further processing
@ -190,15 +223,6 @@ class Wannier90Converter(ConverterTools):
shells_map = [inequiv_to_corr[corr_to_inequiv[ish]] shells_map = [inequiv_to_corr[corr_to_inequiv[ish]]
for ish in range(n_corr_shells)] for ish in range(n_corr_shells)]
mpi.report("Mapping: " + format(shells_map)) mpi.report("Mapping: " + format(shells_map))
mpi.report("Subtracting %f eV from the Fermi level." % self.fermi_energy)
# build the k-point mesh, if its size was given on input (kmesh_mode >= 0),
# otherwise it is built according to the data in the hr file (see
# below)
if kmesh_mode >= 0:
n_k, k_mesh, bz_weights = self.kmesh_build(nki, kmesh_mode)
self.n_k = n_k
self.k_mesh = k_mesh
# not used in this version: reset to dummy values? # not used in this version: reset to dummy values?
n_reps = [1 for i in range(n_inequiv_shells)] n_reps = [1 for i in range(n_inequiv_shells)]
@ -207,33 +231,37 @@ class Wannier90Converter(ConverterTools):
for ish in range(n_inequiv_shells): for ish in range(n_inequiv_shells):
ll = 2 * corr_shells[inequiv_to_corr[ish]]['l'] + 1 ll = 2 * corr_shells[inequiv_to_corr[ish]]['l'] + 1
lmax = ll * (corr_shells[inequiv_to_corr[ish]]['SO'] + 1) lmax = ll * (corr_shells[inequiv_to_corr[ish]]['SO'] + 1)
T.append(numpy.zeros([lmax, lmax], numpy.complex_)) T.append(numpy.zeros([lmax, lmax], dtype=complex))
spin_w90name = ['_up', '_down'] spin_w90name = ['_up', '_down']
hamr_full = [] hamr_full = []
umat_full = []
udismat_full = []
bandmat_full = []
# TODO: generalise to SP=1 (only partially done) # TODO: generalise to SP=1 (only partially done)
rot_mat_time_inv = [0 for i in range(n_corr_shells)] rot_mat_time_inv = [0 for i in range(n_corr_shells)]
# Second, let's read the file containing the Hamiltonian in WF basis # Second, let's read the file containing the Hamiltonian in WF basis
# produced by Wannier90 # produced by Wannier90
for isp in range(n_spin): for isp in range(n_spin_blocs):
# begin loop on isp # begin loop on isp
# build filename according to wannier90 conventions # build filename according to wannier90 conventions
if SP == 1: if SP == 1 and SO == 0:
mpi.report( mpi.report(
"Reading information for spin component n. %d" % isp) "Reading information for spin component n. %d" % isp)
hr_file = self.w90_seed + spin_w90name[isp] + '_hr.dat' file_seed = self.w90_seed + spin_w90name[isp]
else: else:
hr_file = self.w90_seed + '_hr.dat' file_seed = self.w90_seed
# now grab the data from the H(R) file # now grab the data from the H(R) file
mpi.report( mpi.report(
"The Hamiltonian in MLWF basis is extracted from %s ..." % hr_file) "\nThe Hamiltonian in MLWF basis is extracted from %s files..." % file_seed)
nr, rvec, rdeg, nw, hamr = self.read_wannier90hr(hr_file) (nr, rvec, rdeg, nw, hamr, u_mat, udis_mat,
band_mat, k_mesh_from_umat) = self.read_wannier90data(file_seed, kmesh_mode)
# number of R vectors, their indices, their degeneracy, number of # number of R vectors, their indices, their degeneracy, number of
# WFs, H(R) # WFs, H(R)
mpi.report("... done: %d R vectors, %d WFs found" % (nr, nw)) mpi.report("\n... done: %d R vectors, %d WFs found" % (nr, nw))
if isp == 0: if isp == 0:
# set or check some quantities that must be the same for both # set or check some quantities that must be the same for both
@ -241,7 +269,14 @@ class Wannier90Converter(ConverterTools):
self.nrpt = nr self.nrpt = nr
# k-point grid: (if not defined before) # k-point grid: (if not defined before)
if kmesh_mode == -1: if self.bloch_basis:
mpi.report('Reading k mesh from seedname_u.mat file')
kpts = k_mesh_from_umat
n_k = len(kpts)
kpt_weights = numpy.full(n_k, 1/n_k)
self.n_k = n_k
self.kpts = kpts
elif kmesh_mode == -1:
# the size of the k-point mesh is determined from the # the size of the k-point mesh is determined from the
# largest R vector # largest R vector
nki = [2 * rvec[:, idir].max() + 1 for idir in range(3)] nki = [2 * rvec[:, idir].max() + 1 for idir in range(3)]
@ -249,9 +284,10 @@ class Wannier90Converter(ConverterTools):
# wannier90 convention: if we have nki k-points along the i-th direction, # wannier90 convention: if we have nki k-points along the i-th direction,
# then we should get 2*(nki/2)+nki%2 R points along that # then we should get 2*(nki/2)+nki%2 R points along that
# direction # direction
n_k, k_mesh, bz_weights = self.kmesh_build(nki) n_k, kpts, kpt_weights = self.kmesh_build(nki)
self.n_k = n_k self.n_k = n_k
self.k_mesh = k_mesh self.kpts = kpts
# set the R vectors and their degeneracy # set the R vectors and their degeneracy
self.rvec = rvec self.rvec = rvec
@ -273,9 +309,10 @@ class Wannier90Converter(ConverterTools):
# we assume spin up and spin down always have same total number # we assume spin up and spin down always have same total number
# of WFs # of WFs
n_orbitals = numpy.ones( # get second dimension of udis_mat which corresponds to number of bands in window
[self.n_k, n_spin], numpy.int) * self.nwfs # n_bands_max corresponds to numpy.max(n_orbitals)
n_bands_max = udis_mat.shape[1]
n_orbitals = numpy.full([self.n_k, n_spin_blocs], n_bands_max)
else: else:
# consistency check between the _up and _down file contents # consistency check between the _up and _down file contents
if nr != self.nrpt: if nr != self.nrpt:
@ -286,8 +323,9 @@ class Wannier90Converter(ConverterTools):
"Different number of WFs for spin-up/spin-down!") "Different number of WFs for spin-up/spin-down!")
hamr_full.append(hamr) hamr_full.append(hamr)
# FIXME: when do we actually need deepcopy()? umat_full.append(u_mat)
# hamr_full.append(deepcopy(hamr)) udismat_full.append(udis_mat)
bandmat_full.append(band_mat)
for ir in range(nr): for ir in range(nr):
# checks if the Hamiltonian is real (it should, if # checks if the Hamiltonian is real (it should, if
@ -322,39 +360,92 @@ class Wannier90Converter(ConverterTools):
"Rotations for spin component n. %d do not match!" % isp) "Rotations for spin component n. %d do not match!" % isp)
# end loop on isp # end loop on isp
mpi.report("The k-point grid has dimensions: %d, %d, %d" % tuple(nki)) # Reads misc input needed for CSC calculations
if self.bloch_basis:
if os.path.isfile(self.w90_seed + '.nscf.out'):
fermi_weight_file = self.w90_seed + '.nscf.out'
print('Reading DFT band occupations from Quantum Espresso output {}'.format(fermi_weight_file))
elif os.path.isfile('LOCPROJ'):
fermi_weight_file = 'LOCPROJ'
print('Reading DFT band occupations from Vasp output {}'.format(fermi_weight_file))
else:
raise IOError('seedname.nscf.out or LOCPROJ required in bloch_basis mode')
f_weights, band_window, self.fermi_energy = self.convert_misc_input(fermi_weight_file,
self.w90_seed + '.nnkp', n_spin_blocs)
# Get density from k-point weighted average and sum over all spins and bands
density_required = numpy.sum(f_weights.T * kpt_weights) * (2 - SP)
print('Overwriting required density with DFT result {:.5f}'.format(density_required))
print('and using the DFT Fermi energy {:.5f} eV\n'.format(self.fermi_energy))
if not self.bloch_basis:
mpi.report("The k-point grid has dimensions: %d, %d, %d" % tuple(nki))
# if calculations are spin-polarized, then renormalize k-point weights # if calculations are spin-polarized, then renormalize k-point weights
if SP == 1: if SP == 1 and SO == 0:
bz_weights = 0.5 * bz_weights kpt_weights *= 0.5
# Third, compute the hoppings in reciprocal space # Third, initialise the projectors
hopping = numpy.zeros([self.n_k, n_spin, numpy.max( k_dep_projection = 0 # at the moment not really used, but might get important
n_orbitals), numpy.max(n_orbitals)], numpy.complex_) proj_mat = numpy.zeros([self.n_k, n_spin_blocs, n_corr_shells, max(
for isp in range(n_spin): [crsh['dim'] for crsh in corr_shells]), numpy.max(n_orbitals)], dtype=complex)
# make Fourier transform H(R) -> H(k) : it can be done one spin at
# a time
hamk = self.fourier_ham(self.nwfs, hamr_full[isp])
# copy the H(k) in the right place of hoppings... is there a better
# way to do this??
for ik in range(self.n_k):
#hopping[ik,isp,:,:] = deepcopy(hamk[ik][:,:])*energy_unit
hopping[ik, isp, :, :] = hamk[ik][:, :] * energy_unit
# Then, initialise the projectors
k_dep_projection = 0 # we always have the same number of WFs at each k-point
proj_mat = numpy.zeros([self.n_k, n_spin, n_corr_shells, max(
[crsh['dim'] for crsh in corr_shells]), numpy.max(n_orbitals)], numpy.complex_)
iorb = 0 iorb = 0
# Projectors simply consist in identity matrix blocks selecting those MLWFs that # Projectors are either identity matrix blocks to use with Wannier basis
# correspond to the specific correlated shell indexed by icrsh. # OR correspond to the overlap between Kohn-Sham and Wannier orbitals as
# P_{nu,alpha](k) = <w_{alpha,k}|psi_{nu,k}>
# NOTE: we assume that the correlated orbitals appear at the beginning of the H(R) # NOTE: we assume that the correlated orbitals appear at the beginning of the H(R)
# file and that the ordering of MLWFs matches the corr_shell info from # file and that the ordering of MLWFs matches the corr_shell info from
# the input. # the input.
for icrsh in range(n_corr_shells): for isp in range(n_spin_blocs):
norb = corr_shells[icrsh]['dim'] # now combine udismat and umat
proj_mat[:, :, icrsh, 0:norb, iorb:iorb + u_total = numpy.einsum('abc,acd->abd',udismat_full[isp],umat_full[isp])
norb] = numpy.identity(norb, numpy.complex_) # transpose and write into proj_mat
iorb += norb u_temp = numpy.transpose(u_total.conj(),(0,2,1))
for icrsh in range(n_corr_shells):
dim = corr_shells[icrsh]['dim']
proj_mat[:, isp, icrsh, 0:dim, :] = u_temp[:,iorb:iorb+dim,:]
iorb += dim
# Then, compute the hoppings in reciprocal space
hopping = numpy.zeros([self.n_k, n_spin_blocs, numpy.max(n_orbitals), numpy.max(n_orbitals)], dtype=complex)
for isp in range(n_spin_blocs):
# if bloch_basis is True, use Kohn-Sham eigenvalues as hamk
# this ensures that the calculation of the band-correlation energy
# is consistent with SumkDFT's calc_density_correction
if self.bloch_basis:
# diagonal Kohn-Sham bands
# TODO: test for system with self.nwfs > dim_corr_shells
hamk = [numpy.diag(bandmat_full[isp][ik]) for ik in range(self.n_k)]
# Sanity check if the local Hamiltonian with the projector method
# corresponds to W90 result
wannier_ham = self.fourier_ham(hamr_full[isp])
for ik in range(self.n_k):
proj_mat_flattened = proj_mat[ik, isp].reshape(self.nwfs, numpy.max(n_orbitals))
downfolded_ham = proj_mat_flattened.dot(hamk[ik].dot(proj_mat_flattened.conj().T))
if not numpy.allclose(downfolded_ham, wannier_ham[ik], atol=self._w90zero, rtol=0):
mpi.report('WARNING: mismatch between downfolded Hamiltonian and '
+ f'Fourier transformed H(R). First occurred at kpt {ik}:')
with numpy.printoptions(formatter={'complexfloat': '{:+.3f}'.format}):
mpi.report('Downfolded Hamiltonian, P H_eig P')
mpi.report(downfolded_ham)
mpi.report('\nWannier Hamiltonian, Fourier(H(r))')
mpi.report(wannier_ham[ik])
break
# else for an isolated set of bands use fourier transform of H(R)
else:
# make Fourier transform H(R) -> H(k) : it can be done one spin at a time
hamk = self.fourier_ham(hamr_full[isp])
# finally write hamk into hoppings
for ik in range(self.n_k):
hopping[ik, isp] = hamk[ik] - numpy.identity(numpy.max(n_orbitals)) * self.fermi_energy
hopping *= energy_unit
mpi.report("Subtracting {:.5f} eV from the Fermi level.".format(self.fermi_energy))
# bz_weights required by triqs h5 standard but soon to be replaced by kpt_weights
bz_weights = kpt_weights
# Finally, save all required data into the HDF archive: # Finally, save all required data into the HDF archive:
# use_rotations is supposed to be an int = 0, 1, no bool # use_rotations is supposed to be an int = 0, 1, no bool
@ -367,18 +458,25 @@ class Wannier90Converter(ConverterTools):
things_to_save = ['energy_unit', 'n_k', 'k_dep_projection', 'SP', 'SO', 'charge_below', 'density_required', things_to_save = ['energy_unit', 'n_k', 'k_dep_projection', 'SP', 'SO', 'charge_below', 'density_required',
'symm_op', 'n_shells', 'shells', 'n_corr_shells', 'corr_shells', 'use_rotations', 'rot_mat', 'symm_op', 'n_shells', 'shells', 'n_corr_shells', 'corr_shells', 'use_rotations', 'rot_mat',
'rot_mat_time_inv', 'n_reps', 'dim_reps', 'T', 'n_orbitals', 'proj_mat', 'bz_weights', 'hopping', 'rot_mat_time_inv', 'n_reps', 'dim_reps', 'T', 'n_orbitals', 'proj_mat', 'bz_weights', 'hopping',
'n_inequiv_shells', 'corr_to_inequiv', 'inequiv_to_corr'] 'n_inequiv_shells', 'corr_to_inequiv', 'inequiv_to_corr', 'kpt_weights', 'kpts']
for it in things_to_save: for it in things_to_save:
ar[self.dft_subgrp][it] = locals()[it] ar[self.dft_subgrp][it] = locals()[it]
def read_wannier90hr(self, hr_filename="wannier_hr.dat"): if self.bloch_basis:
# Store Fermi weights to 'dft_misc_input'
if not (self.misc_subgrp in ar):
ar.create_group(self.misc_subgrp)
ar[self.misc_subgrp]['dft_fermi_weights'] = f_weights
ar[self.misc_subgrp]['band_window'] = band_window+1 # Change to 1-based index
def read_wannier90data(self, wannier_seed="wannier", kmesh_mode=0):
""" """
Method for reading the seedname_hr.dat file produced by Wannier90 (http://wannier.org) Method for reading the seedname_hr.dat file produced by Wannier90 (http://wannier.org)
Parameters Parameters
---------- ----------
hr_filename : string wannier_seed : string
full name of the H(R) file produced by Wannier90 (usually seedname_hr.dat) seedname to read H(R) file produced by Wannier90 (usually seedname_hr.dat)
Returns Returns
------- -------
@ -392,21 +490,30 @@ class Wannier90Converter(ConverterTools):
number of Wannier functions found number of Wannier functions found
h_of_r : list of numpy.array h_of_r : list of numpy.array
<w_i|H(R)|w_j> = Hamilonian matrix elements in the Wannier basis <w_i|H(R)|w_j> = Hamilonian matrix elements in the Wannier basis
u_mat : numpy.array
U_mn^k = unitary matrix elements which mix the Kohn-Sham states
udis_mat : numpy.array
U^dis(k) = rectangular matrix for entangled bands
band_mat : numpy.array
\epsilon_nk = Kohn-Sham eigenvalues (in eV) needed for entangled bands
If not self.bloch_basis unused and therefore None
k_mesh : numpy.array
The k mesh read from the seedname_u.mat file to ensure consistency
If not self.bloch_basis unused and therefore None
""" """
# Read only from the master node # Read only from the master node
if not (mpi.is_master_node()): if not (mpi.is_master_node()):
return return
hr_filename = wannier_seed + '_hr.dat'
try: try:
with open(hr_filename, "r") as hr_filedesc: with open(hr_filename, 'r') as hr_filedesc:
hr_data = hr_filedesc.readlines() hr_data = hr_filedesc.readlines()
hr_filedesc.close()
except IOError: except IOError:
mpi.report("The file %s could not be read!" % hr_filename) mpi.report("The file %s could not be read!" % hr_filename)
mpi.report("Reading %s..." % hr_filename + hr_data[0]) mpi.report('reading {:20}...{}'.format(hr_filename,hr_data[0].strip('\n')))
try: try:
# reads number of Wannier functions per spin # reads number of Wannier functions per spin
@ -415,11 +522,79 @@ class Wannier90Converter(ConverterTools):
except ValueError: except ValueError:
mpi.report("Could not read number of WFs or R vectors") mpi.report("Could not read number of WFs or R vectors")
k_mesh = None
if not self.bloch_basis:
# For kmesh_mode == -1, size of automatic k mesh known when rvec_idx
# have been read. For kmesh_mode >= 0, kpts have been determined already
if kmesh_mode >= 0:
n_k = self.n_k
else:
# first, read u matrices from 'seedname_u.mat'
u_filename = wannier_seed + '_u.mat'
with open(u_filename,'r') as u_file:
u_data = u_file.readlines()
# reads number of kpoints and number of wannier functions
n_k, num_wf_u, _ = map(int, u_data[1].split())
assert num_wf_u == num_wf, '#WFs must be identical for *_u.mat and *_hr.dat'
mpi.report('reading {:20}...{}'.format(u_filename,u_data[0].strip('\n')))
# Reads k mesh from all lines with 3 floats
k_mesh = numpy.loadtxt((line for line in u_data if line.count('.') == 3))
assert k_mesh.shape == (n_k, 3)
# Reads u matrices from all lines with 2 floats
u_mat = numpy.loadtxt((line for line in u_data if line.count('.') == 2))
assert u_mat.shape == (n_k*num_wf*num_wf, 2)
mpi.report('Writing h5 archive in projector formalism: H(k) defined in KS Bloch basis')
try:
# read 'seedname_u_dis.mat'
udis_filename = wannier_seed + '_u_dis.mat'
# if it exists the Kohn-Sham eigenvalues and the window are needed
band_filename = wannier_seed + '.eig'
wout_filename = wannier_seed + '.wout'
with open(udis_filename,'r') as udis_file:
udis_data = udis_file.readlines()
disentangle = True
except IOError:
disentangle = False
mpi.report('WARNING: File {} missing.'.format(udis_filename))
mpi.report('Assuming an isolated set of bands. Check if this is what you want!')
# read Kohn-Sham eigenvalues from 'seedname.eig'
mpi.report('Reading {}'.format(band_filename))
band_data = numpy.loadtxt(band_filename, usecols=2)
if disentangle:
# reads number of kpoints, number of wannier functions and bands
num_k_udis, num_wf_udis, num_ks_bands = map(int, udis_data[1].split())
assert num_k_udis == n_k, '#k points must be identical for *.inp and *_u_dis.mat'
assert num_wf_udis == num_wf, '#WFs must be identical for *_u.mat and *_hr.dat'
mpi.report('Found {:22}...{}, '.format(udis_filename,udis_data[0].strip('\n')))
udis_data = numpy.loadtxt(udis_data, usecols=(0, 1), skiprows=2)
# read disentanglement window from 'seedname.wout'
with open(wout_filename) as wout_file:
for line in wout_file:
if 'Outer:' in line:
content = line.split()
index = content.index('Outer:') + 1
dis_window_min = float(content[index])
dis_window_max = float(content[index+2])
break
mpi.report('and {} for disentanglement energy window.'.format(wout_filename))
else:
num_ks_bands = num_wf
# allocate arrays to save the R vector indexes and degeneracies and the # allocate arrays to save the R vector indexes and degeneracies and the
# Hamiltonian # Hamiltonian
rvec_idx = numpy.zeros((nrpt, 3), dtype=int) rvec_idx = numpy.zeros((nrpt, 3), dtype=int)
rvec_deg = numpy.zeros(nrpt, dtype=int) rvec_deg = numpy.zeros(nrpt, dtype=int)
h_of_r = [numpy.zeros((num_wf, num_wf), dtype=numpy.complex_) h_of_r = [numpy.zeros((num_wf, num_wf), dtype=complex)
for n in range(nrpt)] for n in range(nrpt)]
# variable currpos points to the current line in the file # variable currpos points to the current line in the file
@ -437,7 +612,7 @@ class Wannier90Converter(ConverterTools):
ir += 1 ir += 1
# for each direct lattice vector R read the block of the # for each direct lattice vector R read the block of the
# Hamiltonian H(R) # Hamiltonian H(R)
for ir, jj, ii in product(list(range(nrpt)), list(range(num_wf)), list(range(num_wf))): for ir, jj, ii in product(range(nrpt), range(num_wf), range(num_wf)):
# advance one line, split the line into tokens # advance one line, split the line into tokens
currpos += 1 currpos += 1
cline = hr_data[currpos].split() cline = hr_data[currpos].split()
@ -445,8 +620,7 @@ class Wannier90Converter(ConverterTools):
if int(cline[3]) != ii + 1 or int(cline[4]) != jj + 1: if int(cline[3]) != ii + 1 or int(cline[4]) != jj + 1:
mpi.report( mpi.report(
"Inconsistent indices at %s%s of R n. %s" % (ii, jj, ir)) "Inconsistent indices at %s%s of R n. %s" % (ii, jj, ir))
rcurr = numpy.array( rcurr = numpy.array([int(cline[0]), int(cline[1]), int(cline[2])])
[int(cline[0]), int(cline[1]), int(cline[2])])
if ii == 0 and jj == 0: if ii == 0 and jj == 0:
rvec_idx[ir] = rcurr rvec_idx[ir] = rcurr
rprec = rcurr rprec = rcurr
@ -457,16 +631,57 @@ class Wannier90Converter(ConverterTools):
"Inconsistent indices for R vector n. %s" % ir) "Inconsistent indices for R vector n. %s" % ir)
# fill h_of_r with the matrix elements of the Hamiltonian # fill h_of_r with the matrix elements of the Hamiltonian
if not numpy.any(rcurr) and ii == jj: h_of_r[ir][ii, jj] = complex(float(cline[5]), float(cline[6]))
h_of_r[ir][ii, jj] = complex(float(cline[5]) - self.fermi_energy, float(cline[6]))
else:
h_of_r[ir][ii, jj] = complex(float(cline[5]), float(cline[6]))
except ValueError: except ValueError:
mpi.report("Wrong data or structure in file %s" % hr_filename) mpi.report("Wrong data or structure in file %s" % hr_filename)
# first, get the input for u_mat
if self.bloch_basis:
u_mat = u_mat[:, 0] + 1j * u_mat[:, 1]
u_mat = u_mat.reshape((n_k, num_wf, num_wf)).transpose((0, 2, 1))
else:
if kmesh_mode == -1:
n_k = numpy.prod([2 * rvec_idx[:, idir].max() + 1 for idir in range(3)])
# Wannier basis; fill u_mat with identity
u_mat = numpy.zeros([n_k, num_wf, num_wf], dtype=complex)
for ik in range(n_k):
u_mat[ik,:,:] = numpy.identity(num_wf,dtype=complex)
# now, check what is needed in the case of disentanglement:
# The file seedname_u_dis.mat contains only the bands inside the window
# and then fills the rest up with zeros. Therefore, we need to put the
# entries from udis_data in the correct position in udis_mat, i.e.
# shifting by the number of bands below dis_window_min
if self.bloch_basis:
# reshape band_data
band_mat = band_data.reshape(n_k, num_ks_bands)
else:
# Not used in wannier basis
band_mat = None
if self.bloch_basis and disentangle:
# Determine which bands are inside the band window
inside_window = numpy.logical_and(band_mat >= dis_window_min,
band_mat <= dis_window_max)
n_inside_per_k = numpy.sum(inside_window, axis=1)
# Reformats udis_data as complex, without header
udis_data = udis_data[:, 0] + 1j * udis_data[:, 1]
udis_data = udis_data.reshape((n_k, num_wf*num_ks_bands+1))[:, 1:]
udis_data = udis_data.reshape((n_k, num_wf, num_ks_bands))
#initiate U disentanglement matrices and fill from file "seedname_u_dis.mat"
udis_mat = numpy.zeros([n_k, num_ks_bands, num_wf], dtype=complex)
for ik in range(n_k):
udis_mat[ik, inside_window[ik]] = udis_data[ik, :, :n_inside_per_k[ik]].T
assert numpy.allclose(udis_data[ik, :, n_inside_per_k[ik]:], 0)
else:
# no disentanglement; fill udis_mat with identity
udis_mat = numpy.array([numpy.identity(num_wf,dtype=complex)] * n_k)
# return the data into variables # return the data into variables
return nrpt, rvec_idx, rvec_deg, num_wf, h_of_r return nrpt, rvec_idx, rvec_deg, num_wf, h_of_r, u_mat, udis_mat, band_mat, k_mesh
def find_rot_mat(self, n_sh, sh_lst, sh_map, ham0): def find_rot_mat(self, n_sh, sh_lst, sh_map, ham0):
""" """
@ -588,7 +803,7 @@ class Wannier90Converter(ConverterTools):
return succeeded, rot_mat return succeeded, rot_mat
def kmesh_build(self, msize=None, mmode=0): def kmesh_build(self, msize, mmode=0):
""" """
Method for the generation of the k-point mesh. Method for the generation of the k-point mesh.
Right now it only supports the option for generating a full grid containing k=0,0,0. Right now it only supports the option for generating a full grid containing k=0,0,0.
@ -604,7 +819,7 @@ class Wannier90Converter(ConverterTools):
------- -------
nkpt : integer nkpt : integer
total number of k-points in the mesh total number of k-points in the mesh
k_mesh : numpy.array[nkpt,3] of floats kpts : numpy.array[nkpt,3] of floats
the coordinates of all k-points the coordinates of all k-points
wk : numpy.array[nkpt] of floats wk : numpy.array[nkpt] of floats
the weight of each k-point the weight of each k-point
@ -614,30 +829,26 @@ class Wannier90Converter(ConverterTools):
if mmode != 0: if mmode != 0:
raise ValueError("Mesh generation mode not supported: %s" % mmode) raise ValueError("Mesh generation mode not supported: %s" % mmode)
assert len(msize) == 3
# a regular mesh including Gamma point # a regular mesh including Gamma point
# total number of k-points # total number of k-points
nkpt = msize[0] * msize[1] * msize[2] nkpt = numpy.prod(msize)
kmesh = numpy.zeros((nkpt, 3), dtype=float) kpts = numpy.array(list(product(range(msize[0]), range(msize[1]), range(msize[2]))))
ii = 0 kpts = kpts / numpy.array(msize)
for ix, iy, iz in product(list(range(msize[0])), list(range(msize[1])), list(range(msize[2]))):
kmesh[ii, :] = [float(ix) / msize[0], float(iy) /
msize[1], float(iz) / msize[2]]
ii += 1
# weight is equal for all k-points because wannier90 uses uniform grid on whole BZ # weight is equal for all k-points because wannier90 uses uniform grid on whole BZ
# (normalization is always 1 and takes into account spin degeneracy) # (normalization is always 1 and takes into account spin degeneracy)
wk = numpy.ones([nkpt], dtype=float) / float(nkpt) wk = numpy.full(nkpt, 1/nkpt)
return nkpt, kmesh, wk return nkpt, kpts, wk
def fourier_ham(self, norb, h_of_r): def fourier_ham(self, h_of_r):
""" """
Method for obtaining H(k) from H(R) via Fourier transform Method for obtaining H(k) from H(R) via Fourier transform
The R vectors and k-point mesh are read from global module variables The R vectors and k-point mesh are read from global module variables
Parameters Parameters
---------- ----------
norb : integer
number of orbitals
h_of_r : list of numpy.array[norb,norb] h_of_r : list of numpy.array[norb,norb]
Hamiltonian H(R) in Wannier basis Hamiltonian H(R) in Wannier basis
@ -648,14 +859,122 @@ class Wannier90Converter(ConverterTools):
""" """
twopi = 2 * numpy.pi h_of_k = [numpy.zeros((self.nwfs, self.nwfs), dtype=complex)
h_of_k = [numpy.zeros((norb, norb), dtype=numpy.complex_)
for ik in range(self.n_k)] for ik in range(self.n_k)]
ridx = numpy.array(list(range(self.nrpt))) ridx = numpy.array(list(range(self.nrpt)))
for ik, ir in product(list(range(self.n_k)), ridx): for ik, ir in product(list(range(self.n_k)), ridx):
rdotk = twopi * numpy.dot(self.k_mesh[ik], self.rvec[ir]) rdotk = numpy.dot(self.kpts[ik], self.rvec[ir])
factor = (math.cos(rdotk) + 1j * math.sin(rdotk)) / \ factor = numpy.exp(2j * numpy.pi * rdotk) / self.rdeg[ir]
float(self.rdeg[ir])
h_of_k[ik][:, :] += factor * h_of_r[ir][:, :] h_of_k[ik][:, :] += factor * h_of_r[ir][:, :]
return h_of_k return h_of_k
def convert_misc_input(self, out_filename, nnkp_filename, n_spin_blocs):
"""
Reads input from DFT code calculations to get occupations
Parameters
----------
out_filename : string
filename of DFT output file containing occupation data
nnkp_file : string
filename of Wannier postproc_setup run
n_spin_blocs : int
SP + 1 - SO
Returns
-------
fermi_weights : numpy.array[self.n_k, n_spin_blocs ,n_orbitals]
occupations from DFT calculation
band_window : numpy.array[self.n_k, n_spin_blocs ,n_orbitals]
band indices of correlated subspace
fermi_energy : float
the DFT Kohn-Sham Fermi energy
"""
# Read only from the master node
if not (mpi.is_master_node()):
return
assert n_spin_blocs == 1, 'spin-polarized not implemented'
assert 'nscf.out' in out_filename or out_filename == 'LOCPROJ'
occupations = []
if 'nscf.out' in out_filename:
occupations = []
with open(out_filename,'r') as out_file:
out_data = out_file.readlines()
# Reads number of Kohn-Sham states and total number of electrons
for line in out_data:
if 'number of electrons' in line:
n_total_electrons = float(line.split()[-1])
if 'number of Kohn-Sham states' in line:
n_ks = int(line.split()[-1])
if 'Fermi energy' in line:
fermi_energy = float(line.split()[-2])
# get occupations
for ct, line in enumerate(out_data):
if line.strip() == 'End of band structure calculation':
break
assert 'k = ' in out_data[ct + 2], 'Cannot read occupations. Set verbosity = "high" in {}'.format(out_filename)
out_data = out_data[ct+2:]
# block size of eigenvalues + occupations per k-point
n_block = int(2*numpy.ceil(n_ks/8)+5)
for ik in range(self.n_k):
# get data
k_block = [line.split() for line in out_data[ik*n_block+2:ik*n_block+n_block-1]]
# second half corresponds to occupations
occs = k_block[int(len(k_block)/2)+1:]
flattened_occs = [float(item) for sublist in occs for item in sublist]
occupations.append(flattened_occs)
else:
# Reads LOCPROJ
with open(out_filename, 'r') as file:
header = file.readline()
n_ks = int(header.split()[2])
fermi_energy = float(header.split()[4])
occupations = numpy.loadtxt((line for line in file if 'orbital' in line), usecols=5)
occupations = occupations.reshape((self.n_k, n_ks))
# assume that all bands contribute, then remove from exclude_bands; python indexing
corr_bands = list(range(n_ks))
# read exclude_bands from "seedname.nnkp" file
with open(nnkp_filename, 'r') as nnkp_file:
read = False
skip = False
for line in nnkp_file:
if line.strip() == 'begin exclude_bands':
read = True
# skip one more line that contains total number of excluded bands
skip = True
continue
elif line.strip() == 'end exclude_bands':
read = False
continue
elif skip:
skip = False
continue
elif read:
# wannier index -1
corr_bands.remove(int(line)-1)
# For now, it is only supported to exclude the lowest and highest bands
# If bands in the middle are supposed to be excluded, this doesn't work with the band_window
# We'd need to manually add rows of zeros to the projectors
if numpy.any(numpy.diff(corr_bands) != 1):
raise NotImplementedError('Can only exclude the lowest or highest bands')
band_window = numpy.array([[(min(corr_bands), max(corr_bands))]*self.n_k]*n_spin_blocs)
included_occupations = numpy.array(occupations)[:, corr_bands]
# Adds spin dimension without changing the array
f_weights = included_occupations.reshape(included_occupations.shape[0], 1,
included_occupations.shape[1])
return f_weights, band_window, fermi_energy