mirror of
https://github.com/triqs/dft_tools
synced 2025-01-07 03:43:12 +01:00
278 lines
9.2 KiB
Python
278 lines
9.2 KiB
Python
|
|
################################################################################
|
|
#
|
|
# TRIQS: a Toolbox for Research in Interacting Quantum Systems
|
|
#
|
|
# Copyright (C) 2011 by M. Ferrero, O. Parcollet
|
|
#
|
|
# DFT tools: Copyright (C) 2011 by M. Aichhorn, L. Pourovskii, V. Vildosola
|
|
#
|
|
# PLOVasp: Copyright (C) 2015 by O. E. Peil
|
|
#
|
|
# TRIQS is free software: you can redistribute it and/or modify it under the
|
|
# terms of the GNU General Public License as published by the Free Software
|
|
# Foundation, either version 3 of the License, or (at your option) any later
|
|
# version.
|
|
#
|
|
# TRIQS 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 General Public License for more
|
|
# details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License along with
|
|
# TRIQS. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
################################################################################
|
|
|
|
import errno
|
|
import os
|
|
import re
|
|
import time
|
|
import signal
|
|
import sys
|
|
import triqs.utility.mpi as mpi
|
|
from h5 import HDFArchive
|
|
from . import converter
|
|
from shutil import copyfile
|
|
|
|
xch = sys.excepthook
|
|
def excepthook(typ, value, traceback):
|
|
xch(typ, value, traceback)
|
|
if mpi.MPI.COMM_WORLD.size > 1:
|
|
mpi.MPI.COMM_WORLD.Abort(1)
|
|
sys.excepthook = excepthook
|
|
|
|
debug = False
|
|
#
|
|
# Helper functions
|
|
#
|
|
def sigint_handler(signal, frame):
|
|
raise SystemExit(1)
|
|
|
|
def is_vasp_lock_present():
|
|
res_bool = False
|
|
if mpi.is_master_node():
|
|
res_bool = os.path.isfile('./vasp.lock')
|
|
res_bool = mpi.bcast(res_bool)
|
|
return res_bool
|
|
|
|
def is_vasp_running(vasp_pid):
|
|
"""
|
|
Tests if VASP initial process is still alive.
|
|
"""
|
|
pid_exists = False
|
|
if mpi.is_master_node():
|
|
try:
|
|
os.kill(vasp_pid, 0)
|
|
except OSError as e:
|
|
pid_exists = e.errno == errno.EPERM
|
|
else:
|
|
pid_exists = True
|
|
|
|
pid_exists = mpi.bcast(pid_exists)
|
|
return pid_exists
|
|
|
|
|
|
def get_dft_energy():
|
|
"""
|
|
Reads DFT energy from the last line of Vasp's OSZICAR or from vasptriqs.h5
|
|
"""
|
|
h5_energy = False
|
|
if os.path.isfile('vaspout.h5'):
|
|
with HDFArchive('vaspout.h5', 'r') as h5:
|
|
if 'oszicar' in h5['intermediate/ion_dynamics']:
|
|
dft_energy = h5['intermediate/ion_dynamics/oszicar'][-1,1]
|
|
h5_energy = True
|
|
|
|
# as backup use OSZICAR file
|
|
if not h5_energy:
|
|
with open('OSZICAR', 'r') as file:
|
|
nextline = file.readline()
|
|
while nextline.strip():
|
|
line = nextline
|
|
nextline = file.readline()
|
|
dft_energy = float(line.split()[2])
|
|
|
|
return dft_energy
|
|
|
|
class bcolors:
|
|
MAGENTA = '\033[95m'
|
|
BLUE = '\033[94m'
|
|
GREEN = '\033[92m'
|
|
YELLOW = '\033[93m'
|
|
RED = '\033[91m'
|
|
ENDC = '\033[0m'
|
|
|
|
# Main self-consistent cycle
|
|
def run_all(vasp_pid, dmft_cycle, cfg_file, n_iter, n_iter_dft, vasp_version):
|
|
"""
|
|
"""
|
|
mpi.report(" Waiting for VASP lock to appear...")
|
|
while not is_vasp_lock_present():
|
|
time.sleep(1)
|
|
|
|
vasp_running = True
|
|
|
|
iter = 0
|
|
while vasp_running:
|
|
if debug: print(bcolors.RED + "rank %s"%(mpi.rank) + bcolors.ENDC)
|
|
mpi.report(" Waiting for VASP lock to disappear...")
|
|
mpi.barrier()
|
|
while is_vasp_lock_present():
|
|
time.sleep(1)
|
|
if debug: print(bcolors.YELLOW + " waiting: rank %s"%(mpi.rank) + bcolors.ENDC)
|
|
if not is_vasp_running(vasp_pid):
|
|
mpi.report(" VASP stopped")
|
|
vasp_running = False
|
|
break
|
|
|
|
# Tell VASP to stop if the maximum number of iterations is reached
|
|
|
|
if debug: print(bcolors.MAGENTA + "rank %s"%(mpi.rank) + bcolors.ENDC)
|
|
err = 0
|
|
exc = None
|
|
if debug: print(bcolors.BLUE + "plovasp: rank %s"%(mpi.rank) + bcolors.ENDC)
|
|
if mpi.is_master_node():
|
|
converter.generate_and_output_as_text(cfg_file, vasp_dir='./')
|
|
# Read energy from OSZICAR
|
|
dft_energy = get_dft_energy()
|
|
mpi.barrier()
|
|
|
|
if debug: print(bcolors.GREEN + "rank %s"%(mpi.rank) + bcolors.ENDC)
|
|
corr_energy, sum_k = dmft_cycle()
|
|
mpi.barrier()
|
|
|
|
if mpi.is_master_node():
|
|
total_energy = dft_energy + corr_energy - sum_k.dc_energ[0]
|
|
print()
|
|
print("="*80)
|
|
print(" Total energy: ", total_energy)
|
|
print(" DFT energy: ", dft_energy)
|
|
print(" Corr. energy: ", corr_energy)
|
|
print(" DFT DC: ", sum_k.dc_energ[0])
|
|
print("="*80)
|
|
print()
|
|
|
|
# check if we should do additional VASP calculations
|
|
# in the standard VASP version, VASP writes out GAMMA itself
|
|
# so that if we want to keep GAMMA fixed we have to copy it to
|
|
# GAMMA_recent and copy it back after VASP has completed an iteration
|
|
# if we are using a hacked Version of VASP where the write out is
|
|
# disabled we can skip this step.
|
|
# the hack consists of removing the call of LPRJ_LDApU in VASP src file
|
|
# electron.F around line 644
|
|
iter_dft = 0
|
|
|
|
if vasp_version == 'standard' or vasp_version == 'ncl':
|
|
copyfile(src='GAMMA',dst='GAMMA_recent')
|
|
while iter_dft < n_iter_dft:
|
|
# insert recalculation of GAMMA here
|
|
# Recalculates the density correction
|
|
# Reads in new projectors and hopping and updates chemical potential
|
|
# rot_mat is not updated since it's more closely related to the local problem than DFT
|
|
# New fermi weights are directly read in calc_density_correction
|
|
if iter > 0 and not iter == n_iter and mpi.is_master_node():
|
|
with HDFArchive('vasp.h5', 'r') as archive:
|
|
sum_k.proj_mat = archive['dft_input/proj_mat']
|
|
sum_k.hopping = archive['dft_input/hopping']
|
|
sum_k.proj_mat = mpi.bcast(sum_k.proj_mat)
|
|
sum_k.hopping = mpi.bcast(sum_k.hopping)
|
|
sum_k.calc_mu(precision=0.001)
|
|
|
|
# Writes out GAMMA file
|
|
sum_k.calc_density_correction(dm_type='vasp')
|
|
|
|
mpi.barrier()
|
|
if mpi.is_master_node():
|
|
open('./vasp.lock', 'a').close()
|
|
while is_vasp_lock_present():
|
|
time.sleep(1)
|
|
if not is_vasp_running(vasp_pid):
|
|
mpi.report(" VASP stopped")
|
|
vasp_running = False
|
|
break
|
|
iter_dft += 1
|
|
if vasp_version == 'standard' or vasp_version == 'ncl':
|
|
copyfile(src='GAMMA_recent',dst='GAMMA')
|
|
iter += 1
|
|
if iter == n_iter:
|
|
print("\n Maximum number of iterations reached.")
|
|
print(" Aborting VASP iterations...\n")
|
|
f_stop = open('STOPCAR', 'wt')
|
|
f_stop.write("LABORT = .TRUE.\n")
|
|
f_stop.close()
|
|
if mpi.is_master_node():
|
|
total_energy = dft_energy + corr_energy - sum_k.dc_energ[0]
|
|
with open('TOTENERGY', 'w') as f:
|
|
f.write(" Total energy: %s\n"%(total_energy))
|
|
f.write(" DFT energy: %s\n"%(dft_energy))
|
|
f.write(" Corr. energy: %s\n"%(corr_energy))
|
|
f.write(" DFT DC: %s\n"%(sum_k.dc_energ[0]))
|
|
f.write(" Energy correction: %s\n"%(corr_energy - sum_k.dc_energ[0]))
|
|
|
|
mpi.report("***Done")
|
|
|
|
def main():
|
|
|
|
import importlib
|
|
|
|
try:
|
|
vasp_pid = int(sys.argv[1])
|
|
except (ValueError, KeyError):
|
|
if mpi.is_master_node():
|
|
print("ERROR: VASP process pid must be provided as the first argument")
|
|
raise
|
|
|
|
try:
|
|
n_iter = int(sys.argv[2])
|
|
except (ValueError, KeyError):
|
|
if mpi.is_master_node():
|
|
print("ERROR: Number of iterations must be provided as the second argument")
|
|
raise
|
|
|
|
try:
|
|
n_iter_dft = int(sys.argv[3])
|
|
except (ValueError, KeyError):
|
|
if mpi.is_master_node():
|
|
print("ERROR: Number of VASP iterations with fixed charge density must be provided as the third argument")
|
|
raise
|
|
|
|
try:
|
|
dmft_script = re.sub("\.py$", "", sys.argv[4])
|
|
except:
|
|
if mpi.is_master_node():
|
|
print("ERROR: User-defined DMFT script must be provided as the fourth argument")
|
|
raise
|
|
|
|
# Optional parameter: config-file name
|
|
try:
|
|
cfg_file = sys.argv[5]
|
|
except KeyError:
|
|
cfg_file = 'plo.cfg'
|
|
|
|
try:
|
|
vasp_version = sys.argv[6]
|
|
except KeyError:
|
|
vasp_version = 'standard'
|
|
|
|
#if vasp_version != 'standard' and vasp_version != 'no_gamma_write':
|
|
# raise Exception('vasp_version has to be standard or no_gamma_write')
|
|
|
|
# if len(sys.argv) > 1:
|
|
# vasp_path = sys.argv[1]
|
|
# else:
|
|
# try:
|
|
# vasp_path = os.environ['VASP_DIR']
|
|
# except KeyError:
|
|
# if mpi.is_master_node():
|
|
# print "Path to VASP must be specified either as an argument of in VASP_DIR"
|
|
# raise
|
|
signal.signal(signal.SIGINT, sigint_handler)
|
|
|
|
dmft_mod = importlib.import_module(dmft_script)
|
|
|
|
run_all(vasp_pid, dmft_mod.dmft_cycle, cfg_file, n_iter, n_iter_dft, vasp_version)
|
|
|
|
if __name__ == '__main__':
|
|
main()
|