Source code for dft_managers.qe_manager

#!/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 function to run a QuantumEspresso iteration. Needed for CSC calculations.
"""

import os
import subprocess
import sys

import triqs.utility.mpi as mpi

from solid_dmft.dft_managers import mpi_helpers


def _start_with_piping(mpi_exe, mpi_arguments, qe_file_ext, env_vars, seedname):
    """
    Handles the piping of the output when starting QE.

    Parameters
    ----------
    mpi_exe: string, mpi command
    mpi_arguments: list of string, arguments to start mpi with
    qe_file_ext : string, file name for QE
    env_vars: dict of string, environment variables containing PATH
    seedname: string, QE input file

    Returns
    -------
    int: id of the VASP child process
    """

    if qe_file_ext in ['scf', 'nscf', 'pw2wan', 'mod_scf', 'bnd', 'bands', 'proj']:

        inp = open(f'{seedname}.{qe_file_ext}.in', 'r')
        out = open(f'{seedname}.{qe_file_ext}.out', 'w')
        err = open(f'{seedname}.{qe_file_ext}.err', 'w')

        print('  solid_dmft: Starting {} calculation...'.format(qe_file_ext))

        # start subprocess
        qe_result = subprocess.run(mpi_arguments, stdin=inp, env=env_vars, capture_output=True,
                                   text=True, shell=False)

        # write output and error file
        output = qe_result.stdout
        error = qe_result.stderr
        out.writelines(output)
        err.writelines(error)

        if qe_result.returncode != 0:
            mpi.report('QE calculation failed. Exiting programm.')
            sys.exit(1)

    elif 'win' in qe_file_ext:
        print('  solid_dmft: Starting Wannier90 {}...'.format(qe_file_ext))
        # don't need any piping for Wannier90
        subprocess.check_call(mpi_arguments + [seedname], env=env_vars, shell=False)


[docs] def run(number_cores, qe_file_ext, qe_exec, mpi_profile, mpi_exe_param, seedname): """ 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. Parameters ---------- number_cores: int, the number of cores that vasp runs on qe_file_ext: string, qe executable qe_exec: string, path to qe executables mpi_profile: string, name of the cluster so that settings can be tailored to it """ # get MPI env hostfile = mpi_helpers.create_hostfile(number_cores, mpi_profile) qe_exec_path = qe_exec.strip(qe_exec.rsplit('/')[-1]) qe_exec = qe_exec_path 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 QE:', mpi_exe) if qe_file_ext in ['scf', 'nscf', 'mod_scf', 'bnd']: qe_exec += f'pw.x -nk {number_cores}' elif qe_file_ext in ['pw2wan']: qe_exec += 'pw2wannier90.x -nk 1 -pd .true.' elif qe_file_ext in ['bands']: qe_exec += f'bands.x -nk {number_cores}' elif qe_file_ext in ['proj']: qe_exec += f'projwfc.x -nk {number_cores}' elif qe_file_ext in ['win_pp']: qe_exec += 'wannier90.x -pp' elif qe_file_ext in ['win']: qe_exec += 'wannier90.x' arguments = mpi_helpers.get_mpi_arguments(mpi_profile, mpi_exe, number_cores, qe_exec, hostfile) _start_with_piping(mpi_exe, arguments, qe_file_ext, env_vars, seedname) mpi_helpers.poll_barrier(mpi.MPI.COMM_WORLD)
[docs] def read_dft_energy(seedname, iter_dmft): """ Reads DFT energy from quantum espresso's out files 1. At the first iteration, the DFT energy is read from the scf file. 2. After the first iteration the band energy computed in the mod_scf calculation is wrong, and needs to be subtracted from the reported total energy. The correct band energy is computed in the nscf calculation. """ dft_energy = 0.0 RYDBERG = 13.605693123 # eV if iter_dmft == 1: with open(f'{seedname}.scf.out', 'r') as file: dft_output = file.readlines() for line in dft_output: if '!' in line: print("\nReading total energy from the scf calculation \n") dft_energy = float(line.split()[-2]) * RYDBERG print(f"The DFT energy is: {dft_energy} eV") break if line =="": raise EOFError("Did not find scf total energy") else: with open(f'{seedname}.mod_scf.out', 'r') as file: dft_output = file.readlines() for line in dft_output: #if 'eband, Ef (eV)' in line: if "(sum(wg*et))" in line: print("\nReading band energy from the mod_scf calculation \n") #band_energy = float(line.split()) band_energy_modscf = float(line.split()[-2])*RYDBERG print(f"The mod_scf band energy is: {band_energy_modscf} eV") if 'total energy' in line: print("\nReading total energy from the mod_scf calculation \n") dft_energy = float(line.split()[-2]) * RYDBERG print(f"The uncorrected DFT energy is: {dft_energy} eV") dft_energy -= band_energy_modscf print(f"The DFT energy without kinetic part is: {dft_energy} eV") with open(f'{seedname}.nscf.out', 'r') as file: dft_output = file.readlines() for line in dft_output: if 'The nscf band energy' in line: print("\nReading band energy from the nscf calculation\n") band_energy_nscf = float(line.split()[-2]) * RYDBERG dft_energy += band_energy_nscf print(f"The nscf band energy is: {band_energy_nscf} eV") print(f"The corrected DFT energy is: {dft_energy} eV") break return dft_energy