2015-11-20 18:54:51 +01:00
|
|
|
r"""
|
|
|
|
vasp.plotools
|
|
|
|
=============
|
2015-02-13 21:58:42 +01:00
|
|
|
|
2015-11-20 18:54:51 +01:00
|
|
|
Set of routines for processing and outputting PLOs.
|
|
|
|
|
|
|
|
This is the main module containing routines responsible for checking
|
|
|
|
the consistency of the input data, generation of projected localized
|
|
|
|
orbitals (PLOs) out of raw VASP projectors, and outputting data
|
|
|
|
required by DFTTools.
|
|
|
|
"""
|
2015-02-19 21:22:51 +01:00
|
|
|
import itertools as it
|
2015-02-13 21:58:42 +01:00
|
|
|
import numpy as np
|
2015-11-13 19:09:25 +01:00
|
|
|
from proj_group import ProjectorGroup
|
|
|
|
from proj_shell import ProjectorShell
|
2015-02-13 21:58:42 +01:00
|
|
|
|
2015-10-16 18:10:48 +02:00
|
|
|
np.set_printoptions(suppress=True)
|
|
|
|
|
2015-08-10 13:52:29 +02:00
|
|
|
# 'simplejson' is supposed to be faster than 'json' in stdlib.
|
2015-09-21 11:59:04 +02:00
|
|
|
try:
|
|
|
|
import simplejson as json
|
|
|
|
except ImportError:
|
|
|
|
import json
|
2015-08-10 13:52:29 +02:00
|
|
|
|
2015-11-11 12:43:51 +01:00
|
|
|
def issue_warning(message):
|
|
|
|
"""
|
|
|
|
Issues a warning.
|
|
|
|
"""
|
|
|
|
print
|
|
|
|
print " !!! WARNING !!!: " + message
|
|
|
|
print
|
|
|
|
|
2015-02-13 21:58:42 +01:00
|
|
|
################################################################################
|
2015-03-01 11:42:18 +01:00
|
|
|
# check_data_consistency()
|
2015-02-13 21:58:42 +01:00
|
|
|
################################################################################
|
2015-03-01 11:42:18 +01:00
|
|
|
def check_data_consistency(pars, el_struct):
|
2015-02-13 21:58:42 +01:00
|
|
|
"""
|
|
|
|
Check the consistency of the VASP data.
|
|
|
|
"""
|
2015-09-16 11:40:01 +02:00
|
|
|
# Check that ions inside each shell are of the same sort
|
|
|
|
for sh in pars.shells:
|
2015-10-15 11:44:50 +02:00
|
|
|
assert max(sh['ion_list']) <= el_struct.natom, "Site index in the projected shell exceeds the number of ions in the structure"
|
2015-09-16 11:56:54 +02:00
|
|
|
sorts = set([el_struct.type_of_ion[io] for io in sh['ion_list']])
|
2015-09-16 11:40:01 +02:00
|
|
|
assert len(sorts) == 1, "Each projected shell must contain only ions of the same sort"
|
2015-02-13 21:58:42 +01:00
|
|
|
|
2015-10-14 19:32:12 +02:00
|
|
|
# Check that ion and orbital lists in shells match those of projectors
|
|
|
|
ion_list = sh['ion_list']
|
|
|
|
lshell = sh['lshell']
|
|
|
|
for ion in ion_list:
|
|
|
|
for par in el_struct.proj_params:
|
|
|
|
if par['isite'] - 1 == ion and par['l'] == lshell:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
errmsg = "Projector for isite = %s, l = %s does not match PROJCAR"%(ion + 1, lshell)
|
|
|
|
raise Exception(errmsg)
|
|
|
|
|
2015-03-01 13:09:31 +01:00
|
|
|
################################################################################
|
|
|
|
#
|
2015-03-03 10:53:46 +01:00
|
|
|
# generate_plo()
|
2015-03-01 13:09:31 +01:00
|
|
|
#
|
|
|
|
################################################################################
|
2015-03-03 10:27:52 +01:00
|
|
|
def generate_plo(conf_pars, el_struct):
|
2015-02-13 21:58:42 +01:00
|
|
|
"""
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
|
|
|
|
conf_pars (dict) : dictionary of input parameters (from conf-file)
|
2015-03-01 11:42:18 +01:00
|
|
|
el_struct : ElectronicStructure object
|
2015-02-13 21:58:42 +01:00
|
|
|
"""
|
|
|
|
|
2015-03-01 11:42:18 +01:00
|
|
|
check_data_consistency(conf_pars, el_struct)
|
2015-02-13 21:58:42 +01:00
|
|
|
|
2015-03-01 11:42:18 +01:00
|
|
|
proj_raw = el_struct.proj_raw
|
2015-02-17 19:42:34 +01:00
|
|
|
try:
|
|
|
|
efermi = conf_pars.general['efermi']
|
2015-03-01 11:42:18 +01:00
|
|
|
except (KeyError, AttributeError):
|
|
|
|
efermi = el_struct.efermi
|
2015-02-17 19:42:34 +01:00
|
|
|
|
2015-02-13 21:58:42 +01:00
|
|
|
# eigvals(nktot, nband, ispin) are defined with respect to the Fermi level
|
2015-10-14 19:32:12 +02:00
|
|
|
eigvals = el_struct.eigvals - efermi
|
2015-02-13 21:58:42 +01:00
|
|
|
|
2015-11-10 12:24:14 +01:00
|
|
|
nshell = len(conf_pars.shells)
|
|
|
|
print
|
|
|
|
print " Generating %i shell%s..."%(nshell, '' if nshell == 1 else 's')
|
2015-03-01 11:42:18 +01:00
|
|
|
pshells = []
|
2015-02-17 19:42:34 +01:00
|
|
|
for sh_par in conf_pars.shells:
|
2015-11-11 20:30:49 +01:00
|
|
|
pshell = ProjectorShell(sh_par, proj_raw, el_struct.proj_params, el_struct.nc_flag)
|
2015-11-10 12:24:14 +01:00
|
|
|
print
|
|
|
|
print " Shell : %s"%(pshell.user_index)
|
|
|
|
print " Orbital l : %i"%(pshell.lorb)
|
|
|
|
print " Number of ions: %i"%(len(pshell.ion_list))
|
2015-11-11 18:58:38 +01:00
|
|
|
print " Dimension : %i"%(pshell.ndim)
|
2015-11-10 12:24:14 +01:00
|
|
|
pshells.append(pshell)
|
2015-02-17 19:42:34 +01:00
|
|
|
|
2015-03-01 11:42:18 +01:00
|
|
|
pgroups = []
|
2015-02-17 21:12:57 +01:00
|
|
|
for gr_par in conf_pars.groups:
|
2015-11-19 16:01:05 +01:00
|
|
|
pgroup = ProjectorGroup(gr_par, pshells, eigvals)
|
2015-10-20 17:37:17 +02:00
|
|
|
pgroup.orthogonalize()
|
2015-11-20 18:54:51 +01:00
|
|
|
# DEBUG output
|
2015-10-16 18:10:48 +02:00
|
|
|
print "Density matrix:"
|
2015-11-20 16:58:36 +01:00
|
|
|
dm_all, ov_all = pshells[pgroup.ishells[0]].density_matrix(el_struct)
|
|
|
|
nimp = 0.0
|
2015-12-09 11:55:29 +01:00
|
|
|
spin_fac = 2 if dm_all.shape[0] == 1 else 1
|
|
|
|
for io in xrange(dm_all.shape[1]):
|
2015-11-20 16:58:36 +01:00
|
|
|
print " Site %i"%(io + 1)
|
2015-12-09 11:55:29 +01:00
|
|
|
dm = spin_fac * dm_all[:, io, : ,:].sum(0)
|
|
|
|
for row in dm:
|
|
|
|
print ''.join(map("{0:12.7f}".format, row))
|
|
|
|
ndm = dm.trace()
|
2015-11-20 16:58:36 +01:00
|
|
|
nimp += ndm
|
|
|
|
print " trace: ", ndm
|
|
|
|
print
|
|
|
|
print " Impurity density:", nimp
|
2015-10-16 18:10:48 +02:00
|
|
|
print
|
|
|
|
print "Overlap:"
|
2015-11-20 16:58:36 +01:00
|
|
|
for io, ov in enumerate(ov_all[0]):
|
|
|
|
print " Site %i"%(io + 1)
|
|
|
|
print ov
|
2015-11-20 18:54:51 +01:00
|
|
|
# END DEBUG output
|
2015-11-10 16:40:46 +01:00
|
|
|
if 'dosmesh' in conf_pars.general:
|
|
|
|
print
|
|
|
|
print "Evaluating DOS..."
|
|
|
|
mesh_pars = conf_pars.general['dosmesh']
|
|
|
|
if np.isnan(mesh_pars['emin']):
|
|
|
|
dos_emin = pgroup.emin
|
|
|
|
dos_emax = pgroup.emax
|
|
|
|
else:
|
|
|
|
dos_emin = mesh_pars['emin']
|
|
|
|
dos_emax = mesh_pars['emax']
|
|
|
|
n_points = mesh_pars['n_points']
|
|
|
|
|
|
|
|
emesh = np.linspace(dos_emin, dos_emax, n_points)
|
|
|
|
dos = pshells[pgroup.ishells[0]].density_of_states(el_struct, emesh)
|
|
|
|
de = emesh[1] - emesh[0]
|
|
|
|
ntot = (dos[1:,...] + dos[:-1,...]).sum(0) / 2 * de
|
|
|
|
print " Total number of states:", ntot
|
|
|
|
for io in xrange(dos.shape[2]):
|
|
|
|
np.savetxt('pdos_%i.dat'%(io), np.vstack((emesh.T, dos[:, 0, io, :].T)).T)
|
|
|
|
|
2015-03-01 11:42:18 +01:00
|
|
|
pgroups.append(pgroup)
|
2015-02-17 21:12:57 +01:00
|
|
|
|
2015-03-01 11:42:18 +01:00
|
|
|
return pshells, pgroups
|
2015-02-13 21:58:42 +01:00
|
|
|
|
2015-11-20 18:54:51 +01:00
|
|
|
################################################################################
|
|
|
|
#
|
|
|
|
# output_as_text
|
|
|
|
#
|
|
|
|
################################################################################
|
|
|
|
def output_as_text(pars, el_struct, pshells, pgroups):
|
|
|
|
"""
|
|
|
|
Output all information necessary for the converter as text files.
|
|
|
|
"""
|
|
|
|
ctrl_output(pars, el_struct, len(pgroups))
|
|
|
|
plo_output(pars, el_struct, pshells, pgroups)
|
|
|
|
|
2015-03-01 13:09:31 +01:00
|
|
|
|
2015-03-03 10:53:46 +01:00
|
|
|
# TODO: k-points with weights should be stored once and for all
|
|
|
|
################################################################################
|
|
|
|
#
|
|
|
|
# kpoints_output
|
|
|
|
#
|
|
|
|
################################################################################
|
|
|
|
def kpoints_output(basename, el_struct):
|
|
|
|
"""
|
|
|
|
Outputs k-point data into a text file.
|
|
|
|
"""
|
|
|
|
kmesh = el_struct.kmesh
|
|
|
|
fname = basename + '.kpoints'
|
|
|
|
with open(fname, 'wt') as f:
|
|
|
|
f.write("# Number of k-points: nktot\n")
|
|
|
|
nktot = kmesh['nktot']
|
|
|
|
f.write("%i\n"%(nktot))
|
|
|
|
# TODO: add the output of reciprocal lattice vectors
|
|
|
|
f.write("# List of k-points with weights\n")
|
|
|
|
for ik in xrange(nktot):
|
|
|
|
kx, ky, kz = kmesh['kpoints'][ik, :]
|
|
|
|
kwght = kmesh['kweights'][ik]
|
|
|
|
f.write("%15.10f%15.10f%15.10f%20.10f\n"%(kx, ky, kz, kwght))
|
|
|
|
|
|
|
|
# Check if there are tetrahedra defined and if they are, output them
|
|
|
|
try:
|
|
|
|
ntet = kmesh['ntet']
|
|
|
|
volt = kmesh['volt']
|
|
|
|
f.write("\n# Number of tetrahedra and volume: ntet, volt\n")
|
|
|
|
f.write("%i %s\n"%(ntet, volt))
|
|
|
|
f.write("# List of tetrahedra: imult, ik1, ..., ik4\n")
|
|
|
|
for it in xrange(ntet):
|
|
|
|
f.write(' '.join(map("{0:d}".format, *kmesh['itet'][it, :])) + '\n')
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2015-08-12 17:09:30 +02:00
|
|
|
################################################################################
|
|
|
|
#
|
|
|
|
# ctrl_output
|
|
|
|
#
|
|
|
|
################################################################################
|
|
|
|
def ctrl_output(conf_pars, el_struct, ng):
|
|
|
|
"""
|
|
|
|
Outputs a ctrl-file.
|
|
|
|
"""
|
|
|
|
ctrl_fname = conf_pars.general['basename'] + '.ctrl'
|
|
|
|
head_dict = {}
|
|
|
|
|
2015-08-13 11:50:49 +02:00
|
|
|
# TODO: Add output of tetrahedra
|
2015-08-12 17:09:30 +02:00
|
|
|
# Construct the header dictionary
|
|
|
|
head_dict['ngroups'] = ng
|
|
|
|
head_dict['nk'] = el_struct.kmesh['nktot']
|
|
|
|
head_dict['ns'] = el_struct.nspin
|
|
|
|
head_dict['nc_flag'] = 1 if el_struct.nc_flag else 0
|
|
|
|
# head_dict['efermi'] = conf_pars.general['efermi'] # We probably don't need Efermi
|
|
|
|
|
|
|
|
header = json.dumps(head_dict, indent=4, separators=(',', ': '))
|
|
|
|
|
2015-08-13 10:59:32 +02:00
|
|
|
print " Storing ctrl-file..."
|
2015-08-12 17:09:30 +02:00
|
|
|
with open(ctrl_fname, 'wt') as f:
|
2015-08-13 10:59:32 +02:00
|
|
|
f.write(header + "\n")
|
|
|
|
f.write("#END OF HEADER\n")
|
|
|
|
|
|
|
|
f.write("# k-points and weights\n")
|
|
|
|
labels = ['kx', 'ky', 'kz', 'kweight']
|
2015-10-15 13:26:48 +02:00
|
|
|
out = "".join(map(lambda s: s.center(15), labels))
|
2015-08-13 10:59:32 +02:00
|
|
|
f.write("#" + out + "\n")
|
|
|
|
for ik, kp in enumerate(el_struct.kmesh['kpoints']):
|
|
|
|
tmp1 = "".join(map("{0:15.10f}".format, kp))
|
|
|
|
out = tmp1 + "{0:16.10f}".format(el_struct.kmesh['kweights'][ik])
|
|
|
|
f.write(out + "\n")
|
|
|
|
|
2015-08-12 17:09:30 +02:00
|
|
|
|
2015-03-01 13:09:31 +01:00
|
|
|
################################################################################
|
|
|
|
#
|
|
|
|
# plo_output
|
|
|
|
#
|
|
|
|
################################################################################
|
2015-08-12 17:09:30 +02:00
|
|
|
def plo_output(conf_pars, el_struct, pshells, pgroups):
|
2015-03-01 13:09:31 +01:00
|
|
|
"""
|
|
|
|
Outputs PLO groups into text files.
|
|
|
|
|
|
|
|
Filenames are defined by <basename> that is passed from config-file.
|
|
|
|
All necessary general parameters are stored in a file '<basename>.ctrl'.
|
|
|
|
|
|
|
|
Each group is stored in a '<basename>.plog<Ng>' file. The format is the
|
|
|
|
following:
|
|
|
|
|
|
|
|
# Energy window: emin, emax
|
|
|
|
ib_min, ib_max
|
2015-03-02 21:29:54 +01:00
|
|
|
nelect
|
2015-03-01 13:09:31 +01:00
|
|
|
# Eigenvalues
|
2015-03-02 21:29:54 +01:00
|
|
|
isp, ik1, kx, ky, kz, kweight
|
2015-03-01 13:09:31 +01:00
|
|
|
ib1, ib2
|
|
|
|
eig1
|
|
|
|
eig2
|
|
|
|
...
|
|
|
|
eigN
|
|
|
|
ik2, kx, ky, kz, kweight
|
|
|
|
...
|
|
|
|
|
|
|
|
# Projected shells
|
|
|
|
Nshells
|
|
|
|
# Shells: <shell indices>
|
2015-08-10 13:52:29 +02:00
|
|
|
# Shell <1>
|
2015-03-01 13:09:31 +01:00
|
|
|
Shell 1
|
|
|
|
ndim
|
|
|
|
# complex arrays: plo(ns, nion, ndim, nb)
|
|
|
|
...
|
|
|
|
# Shells: <shell indices>
|
2015-08-10 13:52:29 +02:00
|
|
|
# Shell <2>
|
2015-03-01 13:09:31 +01:00
|
|
|
Shell 2
|
|
|
|
...
|
|
|
|
|
|
|
|
"""
|
2015-08-13 12:29:41 +02:00
|
|
|
for ig, pgroup in enumerate(pgroups):
|
|
|
|
plo_fname = conf_pars.general['basename'] + '.pg%i'%(ig + 1)
|
|
|
|
print " Storing PLO-group file '%s'..."%(plo_fname)
|
|
|
|
head_dict = {}
|
|
|
|
|
|
|
|
head_dict['ewindow'] = (pgroup.emin, pgroup.emax)
|
|
|
|
head_dict['nb_max'] = pgroup.nb_max
|
|
|
|
|
2015-09-17 12:40:55 +02:00
|
|
|
# Number of electrons within the window
|
|
|
|
head_dict['nelect'] = pgroup.nelect_window(el_struct)
|
2015-10-15 13:26:48 +02:00
|
|
|
print " Density within window:", head_dict['nelect']
|
2015-09-17 12:40:55 +02:00
|
|
|
|
2015-08-13 12:29:41 +02:00
|
|
|
head_shells = []
|
2015-08-18 11:36:08 +02:00
|
|
|
for ish in pgroup.ishells:
|
|
|
|
shell = pgroup.shells[ish]
|
2015-08-13 12:29:41 +02:00
|
|
|
sh_dict = {}
|
2015-08-18 11:36:08 +02:00
|
|
|
sh_dict['shell_index'] = ish
|
2015-08-13 12:29:41 +02:00
|
|
|
sh_dict['lorb'] = shell.lorb
|
|
|
|
sh_dict['ndim'] = shell.ndim
|
|
|
|
# Convert ion indices from the internal representation (starting from 0)
|
|
|
|
# to conventional VASP representation (starting from 1)
|
|
|
|
ion_output = [io + 1 for io in shell.ion_list]
|
|
|
|
sh_dict['ion_list'] = ion_output
|
2015-09-16 11:56:54 +02:00
|
|
|
sh_dict['ion_sort'] = el_struct.type_of_ion[shell.ion_list[0]]
|
|
|
|
|
2015-08-13 12:29:41 +02:00
|
|
|
# TODO: add the output of transformation matrices
|
|
|
|
|
|
|
|
head_shells.append(sh_dict)
|
|
|
|
|
|
|
|
head_dict['shells'] = head_shells
|
|
|
|
|
|
|
|
header = json.dumps(head_dict, indent=4, separators=(',', ': '))
|
|
|
|
|
|
|
|
with open(plo_fname, 'wt') as f:
|
|
|
|
f.write(header + "\n")
|
2015-08-18 11:36:08 +02:00
|
|
|
f.write("#END OF HEADER\n")
|
2015-08-13 12:29:41 +02:00
|
|
|
|
2015-08-18 11:36:08 +02:00
|
|
|
# Eigenvalues within the window
|
|
|
|
f.write("# Eigenvalues within the energy window: %s, %s\n"%(pgroup.emin, pgroup.emax))
|
|
|
|
nk, nband, ns_band = el_struct.eigvals.shape
|
2015-03-02 21:29:54 +01:00
|
|
|
for isp in xrange(ns_band):
|
2015-08-18 11:36:08 +02:00
|
|
|
f.write("# is = %i\n"%(isp + 1))
|
2015-03-02 21:29:54 +01:00
|
|
|
for ik in xrange(nk):
|
2015-08-18 11:36:08 +02:00
|
|
|
ib1, ib2 = pgroup.ib_win[ik, isp, 0], pgroup.ib_win[ik, isp, 1]
|
2015-12-04 19:55:37 +01:00
|
|
|
# Output band indices in Fortran convention!
|
|
|
|
f.write(" %i %i\n"%(ib1 + 1, ib2 + 1))
|
2015-08-18 11:36:08 +02:00
|
|
|
for ib in xrange(ib1, ib2 + 1):
|
2015-10-20 12:02:46 +02:00
|
|
|
eigv_ef = el_struct.eigvals[ik, ib, isp] - el_struct.efermi
|
2015-11-19 16:06:25 +01:00
|
|
|
f_weight = el_struct.ferw[isp, ik, ib]
|
|
|
|
f.write("%13.8f %12.7f\n"%(eigv_ef, f_weight))
|
2015-08-18 11:36:08 +02:00
|
|
|
|
|
|
|
# Projected shells
|
|
|
|
f.write("# Projected shells\n")
|
|
|
|
f.write("# Shells: %s\n"%(pgroup.ishells))
|
|
|
|
for ish in pgroup.ishells:
|
|
|
|
shell = pgroup.shells[ish]
|
|
|
|
f.write("# Shell %i\n"%(ish))
|
|
|
|
|
|
|
|
nion, ns, nk, nlm, nb = shell.proj_win.shape
|
|
|
|
for isp in xrange(ns):
|
|
|
|
f.write("# is = %i\n"%(isp + 1))
|
|
|
|
for ik in xrange(nk):
|
|
|
|
f.write("# ik = %i\n"%(ik + 1))
|
2015-03-02 21:29:54 +01:00
|
|
|
for ion in xrange(nion):
|
2015-08-18 11:36:08 +02:00
|
|
|
for ilm in xrange(nlm):
|
|
|
|
ib1, ib2 = pgroup.ib_win[ik, isp, 0], pgroup.ib_win[ik, isp, 1]
|
2015-10-20 17:37:17 +02:00
|
|
|
ib_win = ib2 - ib1 + 1
|
|
|
|
for ib in xrange(ib_win):
|
2015-08-18 11:36:08 +02:00
|
|
|
p = shell.proj_win[ion, isp, ik, ilm, ib]
|
|
|
|
f.write("{0:16.10f}{1:16.10f}\n".format(p.real, p.imag))
|
2015-03-02 21:29:54 +01:00
|
|
|
f.write("\n")
|
2015-08-18 11:36:08 +02:00
|
|
|
|
|
|
|
|