Source code for abipy.embedding.embedding_ifc

from __future__ import annotations

import numpy as np

from phonopy import Phonopy
from pymatgen.io.phonopy import get_pmg_structure,get_phonopy_structure
from abipy.core.abinit_units import eV_to_THz
from abipy.core.structure import Structure
from abipy.dfpt.converters import phonopy_to_abinit
from abipy.embedding.utils_ifc import map_two_structures_coords, clean_structure # accoustic_sum


[docs] class Embedded_phonons(Phonopy): """ Extends Phonopy object implementing the Interatomic Force Constants embedding method as defined in: https://pubs.acs.org/doi/10.1021/acs.chemmater.3c00537 https://iopscience.iop.org/article/10.1088/1367-2630/16/7/073026/meta https://journals.aps.org/prmaterials/abstract/10.1103/PhysRevMaterials.5.084603 """ def __init__(self, stru_pristine: Structure, stru_defect: Structure, stru_emb: Structure, ifc_emb, nac_params: dict): """ Args: stru_pristine: Supercell pristine structure. stru_defect: Supercell defect structure. stru_emb: Supercell embedded structure. ifc_emb: Interatomic force constant associated to the supercell embedded structure nac_params: Non-analytical parameters associated to the supercell embedded structure, in phonopy format. """ super().__init__(unitcell=get_phonopy_structure(stru_emb), supercell_matrix=[1,1,1], primitive_matrix=np.eye(3),) self.force_constants = ifc_emb self.nac_params = nac_params self.stru_pristine = stru_pristine self.stru_defect = stru_defect self.stru_emb = stru_emb
[docs] @classmethod def from_phonopy_instances(cls, phonopy_pristine: Phonopy, phonopy_defect: Phonopy, structure_defect_wo_relax: Structure, main_defect_coords_in_pristine: Structure, main_defect_coords_in_defect, substitutions_list: list = None, # index in pristine,specie ex: [0,"Eu"] vacancies_list: list = None, # index in pristine ex: [13,14] interstitial_list: list = None, # species, cart_coord ex: [['Eu',[0,0,0]],['Ce','[0,0,3]']] tol_mapping: float = 0.01, cut_off_mode: str = 'auto', rc_1: float | None = None, rc_2: float | None = None, factor_ifc: float = 1.0, verbose: int = 0, asr: bool = True) -> Phonopy: """ Args: phonopy_pristine: Phonopy object of the pristine structure. phonopy_defect: Phonopy object of the defect structure. structure_defect_wo_relax: Supercell structure associated to the defect structure, but without relaxation. Needed for an easier mapping. Should corresponds to order of phonopy_defect structure. main_defect_coords_in_pristine: Main coordinates of the defect in pristine structure, if defect complex, can be set to the center of mass of the complex. main_defect_coords_in_defect: Main coordinates of the defect in defect structure, if defect complex, can be set to the center of mass of the complex substitutions_list: List of substitutions info [index, specie], ex: [[0, "Eu"],[1,"N"]] vacancies_list: List of indices where the vacancies are located, ex: [13, 14]. interstitial_list: List of interstitial info [specie, cart_coord], ex [['Eu',[0,0,0]],['Ce','[0,0,3]']]. tol_mapping: Tolerance in angstrom for the mapping between structures cut_off_mode: Cut off mode for the radii of the sphere centered around the defect (rc_2). If 'auto' the code tries to find the largest sphere inscribed in the defect supercell. If 'manual': rc_1 and rc_2 should be provided. rc_1: Radii of the sphere centered around the defect outside which the IFCs are set to zero, allows to get sparse matrix. rc_2: Radii of the sphere centered around the defect where the IFCs of the defect computation are included factor_ifc: Multiply the IFCs inside the sphere of radii rc_2 by factor_ifc, useful to introduce fictitious high-frequency local mode. verbose: Print explicitly all the IFCs replacements. asr: If True, re-enforce acoustic sum rule after IFCs embedding, following eq. (S4) of https://pubs.acs.org/doi/10.1021/acs.chemmater.3c00537 Returns: A new phonopy object with the embedded structure and embedded IFCs. """ ######################## # Structures manipulations and mapping ######################## stru_defect = get_pmg_structure(phonopy_defect.supercell) stru_pristine = get_pmg_structure(phonopy_pristine.supercell) stru_emb = stru_pristine.copy() if vacancies_list is not None: stru_emb.remove_sites(indices=vacancies_list) if substitutions_list is not None: for sub in substitutions_list: stru_emb.replace(sub[0],sub[1]) if interstitial_list is not None: for inter in interstitial_list: stru_emb.append(species=inter[0],coords=inter[1],coords_are_cartesian=True) stru_emb = clean_structure(stru_emb,defect_coord=main_defect_coords_in_pristine) structure_defect_wo_relax = clean_structure(structure_defect_wo_relax,defect_coord=main_defect_coords_in_defect) mapping = map_two_structures_coords(structure_defect_wo_relax,stru_emb,tol=tol_mapping) ################### # Init of the IFCs ################### ifc_pristine = phonopy_pristine.force_constants ifc_defect = phonopy_defect.force_constants # in case of IFC with vacancy, remove ifcs of the vac site if vacancies_list is not None: ifc_pristine = np.delete(ifc_pristine,vacancies_list,0) ifc_pristine = np.delete(ifc_pristine,vacancies_list,1) # interstitial if interstitial_list is not None: for inter in interstitial_list: ifc_pristine = np.append(ifc_pristine,np.zeros(shape=[1, len(ifc_pristine), 3, 3]), axis=0) ifc_pristine = np.append(ifc_pristine,np.zeros(shape=[len(ifc_pristine), 1, 3, 3]),axis=1) ######################## # Print infos ######################## print(f"Number of atoms in the pristine supercell: {len(stru_pristine)}") print(f"Number of atoms in the defective supercel: {len(stru_defect)}") print("Defect infos") if substitutions_list is not None: print(" Substitutions:") for sub in substitutions_list: print(f" {sub[0]}, {stru_pristine[sub[0]].coords}, {stru_pristine[sub[0]].species} replaced by {sub[1]}") if vacancies_list is not None: print(" Vacancies:") for vac in vacancies_list: print(f" {vac}, {stru_pristine[vac].coords}, {stru_pristine[vac].species} removed") if interstitial_list is not None: print(" Interstitials:") for inter in interstitial_list: print(f" {inter[0]}, {inter[1]} added") print(f"Mapping after structure manipulation: {len(mapping)}/{len(stru_defect)}") ######################## # change the IFCs ######################## ifc_emb = np.zeros(shape=np.shape(ifc_pristine)) if cut_off_mode == 'auto': rc_1 = 100000 # very large value to include all the ifcs, no sparse matrix. rc_2 = 0.99*min(np.array(structure_defect_wo_relax.lattice.abc)/2) # largest sphere inscribed in defect supercell, # 0.99 to avoid problem with atoms at the border if cut_off_mode == 'manual': rc_1 = rc_1 rc_2 = rc_2 # Set to zero if 2 atoms distance is bigger than rc_1 for i, atom1 in enumerate(stru_emb): for j, atom2 in enumerate(stru_emb): dist_ij = np.sqrt(sum((atom1.coords-atom2.coords)**2)) if dist_ij > rc_1: ifc_emb[i][j] = np.zeros(shape=(3, 3)) else: ifc_emb[i][j] = ifc_pristine[i][j] # Set to doped phonons if 2 atoms are separated from defect by distance < R_c2 print(f"\n Set IFC to explicit defect phonons calculations if both atoms are separated from defect by a distance < R_c2 = {round(rc_2,3)}") for i, atom1 in enumerate(stru_emb): for j, atom2 in enumerate(stru_emb): # structure centered around defect!!!, main defect is at [0,0,0] dist_1_from_defect = np.sqrt(sum((atom1.coords)**2)) dist_2_from_defect = np.sqrt(sum((atom2.coords)**2)) if dist_1_from_defect < rc_2 and dist_2_from_defect < rc_2: if verbose > 0: print(f"\n \n Atomic pair: {i,atom1} - {j,atom2} \n") print(f"atom1: Dist. from. defect = {dist_1_from_defect}") print(f"atom2: Dist. from. defect = {dist_2_from_defect}") print(f"Replacing pristine cell IFC = \n {ifc_pristine[i][j]}") print(f"by defect cell IFC = \n {ifc_defect[mapping.index(i)][mapping.index(j)]}") print(f"Diff IFC = \n {ifc_pristine[i][j] - ifc_defect[mapping.index(i)][mapping.index(j)]}") ifc_emb[i][j] = factor_ifc * ifc_defect[mapping.index(i)][mapping.index(j)] # enforce ASR. if asr: print("\n Enforcing ASR") sum_ac = np.sum(ifc_emb,axis=1) for i, atom1 in enumerate(stru_emb): for alpha in [0, 1, 2]: ifc_emb[i][i][alpha][alpha] = - (sum_ac[i][alpha][alpha]-ifc_emb[i][i][alpha][alpha]) ######################## # change the nac params ######################## if phonopy_pristine.nac_params is not None: # Simply copy the nac params from the pristine nac_params_emb = phonopy_pristine.nac_params.copy() if vacancies_list is not None: nac_params_emb["born"] = np.delete(nac_params_emb["born"],vacancies_list,0) if interstitial_list is not None: for interstial in interstitial_list: nac = np.append(nac_params_emb["born"],(stru_emb[-1].specie.common_oxidation_states[0])*np.eye(3)) nac_params_emb["born"] = nac.reshape(len(stru_emb),3,3) else: nac_params_emb = None ######################## print("\n Embedding procedure done") return cls(stru_pristine, stru_defect, stru_emb, ifc_emb, nac_params_emb)
[docs] def get_gamma_freq_with_vec_abipy_fmt(self) -> tuple[np.ndarray, np.ndarray]: """ Compute with phonopy the Gamma phonons freq and vectors and returns them in abipy format such that ph_vec[iband][iatom] gives vector of iband,iatom Returns: phonons frequencies, phonon eigenvectors """ ph_freq_phonopy, ph_vec_phonopy = self.get_frequencies_with_eigenvectors(q=[0,0,0]) ph_freq = ph_freq_phonopy / (eV_to_THz) # put it in eV ph_vec = ph_vec_phonopy.transpose().reshape(3*len(self.supercell),len(self.supercell),3) return ph_freq, ph_vec
[docs] def to_ddb(self, embedded_ddb_path='out_DDB', workdir=None): """ Call the converter to go from phonopy to Abinit DDB. Args: embedded_ddb_path: filepath where to save the DDB. workdir: work directory for the conversion. """ ddb_sc = phonopy_to_abinit(unit_cell=get_pmg_structure(self.supercell), supercell_matrix=[1,1,1], qpt_list=[[0,0,0]], out_ddb_path=embedded_ddb_path, force_constants=self.force_constants, born=self.nac_params, primitive_matrix=np.eye(3), symprec=1e-5, tolsym=None,workdir=workdir, nsym=1) return ddb_sc