#!/usr/bin/env python3
# -*- coding: utf-8 -*-
################################################################################
#
# solid_dmft - A versatile python wrapper to perform DFT+DMFT calculations
# utilizing the TRIQS software library
#
# Copyright (C) 2018-2020, ETH Zurich
# Copyright (C) 2021, The Simons Foundation
# authors: A. Hampel, M. Merkel, and S. Beck
#
# solid_dmft 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.
#
# solid_dmft 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
# solid_dmft (in the file COPYING.txt in this directory). If not, see
# <http://www.gnu.org/licenses/>.
#
################################################################################
"""
Contains the handling of the VASP process. It can start VASP, reactivate it,
check if the lock file is there and finally kill VASP. Needed for CSC calculations.
This functionality is contained in the simpler public functions.
"""
import os
import signal
import time
import numpy as np
import triqs.utility.mpi as mpi
from h5 import HDFArchive
from solid_dmft.dft_managers import mpi_helpers
def _fork_and_start_vasp(mpi_exe, arguments, env_vars):
"""
Forks a process from the master process that then calls mpi to start vasp.
The child process running VASP never leaves this function whereas the main
process returns the child's process id and continues. We use explicitly
os.fork as os.execve here because subprocess does not return, and as VASP
needs to keep running throughout the whole DFT+DMFT calculation that is
not a viable solution here.
Parameters
----------
mpi_exe: string, mpi command
arguments: list of string, arguments to start mpi with
env_vars: dict of string, environment variables containing PATH
Returns
-------
int: id of the VASP child process
"""
# fork process
vasp_process_id = os.fork()
if vasp_process_id == 0:
# close file descriptors, if rank0 had open files
for fd in range(3, 256):
try:
os.close(fd)
except OSError:
pass
print('\n Starting VASP now\n')
os.execve(mpi_exe, arguments, env_vars)
print('\n VASP exec failed\n')
os._exit(127)
return vasp_process_id
def _is_lock_file_present():
"""
Checks if the lock file 'vasp.lock' is there, i.e. if VASP is still working.
"""
res_bool = False
if mpi.is_master_node():
res_bool = os.path.isfile('./vasp.lock')
res_bool = mpi.bcast(res_bool)
return res_bool
[docs]
def remove_legacy_projections_suppressed():
""" Removes legacy file vasp.suppress_projs if present. """
if mpi.is_master_node():
if os.path.isfile('./vasp.suppress_projs'):
print(' solid_dmft: Removing legacy file vasp.suppress_projs', flush=True)
os.remove('./vasp.suppress_projs')
mpi.barrier()
[docs]
def run_initial_scf(number_cores, vasp_command, mpi_exe_param, cluster_name):
"""
Starts the VASP child process. Takes care of initializing a clean
environment for the child process. This is needed so that VASP does not
get confused with all the standard slurm environment variables. Returns when
VASP has completed its initial scf cycle.
Parameters
----------
number_cores: int, the number of cores that vasp runs on
vasp_command: string, the command to start vasp
cluster_name: string, name of the cluster so that settings can be tailored to it
"""
# Removes STOPCAR
if mpi.is_master_node() and os.path.isfile('STOPCAR'):
os.remove('STOPCAR')
mpi.barrier()
# get MPI env
vasp_process_id = 0
hostfile = mpi_helpers.create_hostfile(number_cores, cluster_name)
if mpi.is_master_node():
# clean environment
env_vars = {}
for var_name in ['PATH', 'LD_LIBRARY_PATH', 'SHELL', 'PWD', 'HOME',
'OMP_NUM_THREADS', 'OMPI_MCA_btl_vader_single_copy_mechanism']:
var = os.getenv(var_name)
if var:
env_vars[var_name] = var
# assuming that mpirun points to the correct mpi env
mpi_exe = mpi_helpers.find_path_to_mpi_command(env_vars, mpi_exe_param)
print('\nMPI executable for Vasp:', mpi_exe)
arguments = mpi_helpers.get_mpi_arguments(cluster_name, mpi_exe, number_cores, vasp_command, hostfile)
vasp_process_id = _fork_and_start_vasp(mpi_exe, arguments, env_vars)
mpi_helpers.poll_barrier(mpi.MPI.COMM_WORLD)
vasp_process_id = mpi.bcast(vasp_process_id)
# Waits for VASP to start
while not _is_lock_file_present():
time.sleep(1)
mpi.barrier()
# Waits for VASP to finish
while _is_lock_file_present():
time.sleep(1)
mpi.barrier()
return vasp_process_id
[docs]
def run_charge_update():
"""
Performs one step of the charge update with VASP by creating the vasp.lock
file and then waiting until it gets delete by VASP when it has finished.
"""
if mpi.is_master_node():
open('./vasp.lock', 'a').close()
mpi.barrier()
# Waits for VASP to finish
while _is_lock_file_present():
time.sleep(1)
mpi.barrier()
[docs]
def read_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
[docs]
def read_dft_iter():
"""
Reads DFT iteration number from the last line of the OSZICAR or from Vasp's vasptriqs.h5
"""
h5_iter = False
if os.path.isfile('vaspout.h5'):
with HDFArchive('vaspout.h5', 'r') as h5:
if 'oszicar' in h5['intermediate/ion_dynamics']:
dft_iter = int(h5['intermediate/ion_dynamics/oszicar'][-1,0])
h5_iter = True
# as backup use OSZICAR file
if not h5_iter:
with open('OSZICAR', 'r') as file:
nextline = file.readline()
while nextline.strip():
line = nextline
nextline = file.readline()
dft_iter = int(line.split()[1])
return dft_iter
[docs]
def read_irred_kpoints(kpts):
""" Reads the indices of the irreducible k-points from the OUTCAR. """
def read_outcar(file):
has_started_reading = False
for line in file:
if 'IBZKPT_HF' in line:
has_started_reading = True
continue
if not has_started_reading:
continue
if 't-inv' in line:
yield line
continue
if '-'*10 in line:
break
irred_indices = None
if mpi.is_master_node():
with open('OUTCAR', 'r') as file:
outcar_data_raw = np.loadtxt(read_outcar(file), usecols=[0, 1, 2, 4])
outcar_kpoints = outcar_data_raw[:, :3]
outcar_indices = (outcar_data_raw[:, 3]-.5).astype(int)
assert np.allclose(outcar_kpoints, kpts)
symmetry_mapping = np.full(outcar_kpoints.shape[0], -1, dtype=int)
for i, (kpt_outcar, outcar_index) in enumerate(zip(outcar_kpoints, outcar_indices)):
for j, kpt in enumerate(kpts):
if np.allclose(kpt_outcar, kpt):
# Symmetry-irreducible k points
if i == outcar_index:
symmetry_mapping[j] = outcar_index
# Symmetry-reducible
else:
symmetry_mapping[j] = outcar_index
break
# Asserts that loop left through break, i.e. a pair was found
assert np.allclose(kpt_outcar, kpt)
irreds, irred_indices = np.unique(symmetry_mapping, return_index=True)
assert np.all(np.diff(irreds) == 1)
assert np.all(symmetry_mapping >= 0)
return mpi.bcast(irred_indices)
[docs]
def kill(vasp_process_id):
""" Kills the VASP process. """
# TODO: if we kill the process in the next step, does it make a difference if we write the STOPCAR
with open('STOPCAR', 'wt') as f_stop:
f_stop.write('LABORT = .TRUE.\n')
os.kill(vasp_process_id, signal.SIGTERM)
mpi.MPI.COMM_WORLD.Abort(1)