You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
295 lines
11 KiB
295 lines
11 KiB
# Copyright (c) [2024] [Daniel Pozsar]
|
|
#
|
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
# of this software and associated documentation files (the "Software"), to deal
|
|
# in the Software without restriction, including without limitation the rights
|
|
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
# copies of the Software, and to permit persons to whom the Software is
|
|
# furnished to do so, subject to the following conditions:
|
|
#
|
|
# The above copyright notice and this permission notice shall be included in all
|
|
# copies or substantial portions of the Software.
|
|
#
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
# SOFTWARE.
|
|
|
|
if __name__ == "__main__":
|
|
import argparse
|
|
import sys
|
|
from itertools import permutations, product
|
|
from timeit import default_timer as timer
|
|
|
|
import numpy as np
|
|
import numpy.linalg as nl
|
|
import sisl
|
|
import tqdm
|
|
from mpi4py import MPI
|
|
from scipy.special import roots_legendre
|
|
|
|
from grogu_magn.useful import hsk, make_atran, make_contour, make_kset
|
|
|
|
start = timer()
|
|
|
|
# Some input parsing
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument(
|
|
"--kset",
|
|
dest="kset",
|
|
default=2,
|
|
type=int,
|
|
help="k-space resolution of Jij calculation",
|
|
)
|
|
parser.add_argument(
|
|
"--kdirs",
|
|
dest="kdirs",
|
|
default="xyz",
|
|
help="Definition of k-space dimensionality",
|
|
)
|
|
parser.add_argument(
|
|
"--eset",
|
|
dest="eset",
|
|
default=42,
|
|
type=int,
|
|
help="Number of energy points on the contour",
|
|
)
|
|
parser.add_argument(
|
|
"--eset-p",
|
|
dest="esetp",
|
|
default=10,
|
|
type=int,
|
|
help="Parameter tuning the distribution on the contour",
|
|
)
|
|
parser.add_argument("--input", dest="infile", required=True, help="Input file name")
|
|
parser.add_argument(
|
|
"--output", dest="outfile", required=True, help="Output file name"
|
|
)
|
|
parser.add_argument(
|
|
"--Ebot",
|
|
dest="Ebot",
|
|
default=-20.0,
|
|
type=float,
|
|
help="Bottom energy of the contour",
|
|
)
|
|
parser.add_argument(
|
|
"--npairs",
|
|
dest="npairs",
|
|
default=1,
|
|
type=int,
|
|
help="Number of unitcell pairs in each direction for Jij calculation",
|
|
)
|
|
parser.add_argument(
|
|
"--adirs", dest="adirs", default=False, help="Definition of pair directions"
|
|
)
|
|
parser.add_argument(
|
|
"--use-tqdm",
|
|
dest="usetqdm",
|
|
default="not",
|
|
help="Use tqdm for progressbars or not",
|
|
)
|
|
parser.add_argument(
|
|
"--cutoff",
|
|
dest="cutoff",
|
|
default=100.0,
|
|
type=float,
|
|
help="Real space cutoff for pair generation in Angs",
|
|
)
|
|
parser.add_argument(
|
|
"--pairfile",
|
|
dest="pairfile",
|
|
default=False,
|
|
help="File to read pair information",
|
|
)
|
|
args = parser.parse_args()
|
|
# ----------------------------------------------------------------------
|
|
|
|
# MPI init
|
|
comm = MPI.COMM_WORLD
|
|
size = comm.Get_size()
|
|
rank = comm.Get_rank()
|
|
root_node = 0
|
|
if rank == root_node:
|
|
print("Number of nodes in the parallel cluster: ", size)
|
|
# ----------------------------------------------------------------------
|
|
|
|
# importing the necessary structures from SIESTA output
|
|
dat = sisl.get_sile(args.infile)
|
|
dh = dat.read_hamiltonian()
|
|
# update datastructure of the hamiltonian
|
|
# this is needed for quick Hk building
|
|
dh.hup = (
|
|
dh.tocsr(0)
|
|
.toarray()
|
|
.reshape(dh.no, dh.n_s, dh.no)
|
|
.transpose(0, 2, 1)
|
|
.astype("complex128")
|
|
)
|
|
dh.hdo = (
|
|
dh.tocsr(1)
|
|
.toarray()
|
|
.reshape(dh.no, dh.n_s, dh.no)
|
|
.transpose(0, 2, 1)
|
|
.astype("complex128")
|
|
)
|
|
dh.sov = (
|
|
dh.tocsr(2)
|
|
.toarray()
|
|
.reshape(dh.no, dh.n_s, dh.no)
|
|
.transpose(0, 2, 1)
|
|
.astype("complex128")
|
|
)
|
|
# ----------------------------------------------------------------------
|
|
|
|
# generate k space sampling
|
|
kset = make_kset(dirs=args.kdirs, NUMK=args.kset)
|
|
wk = 1 / len(kset) # weight of a kpoint in BZ integral
|
|
kpcs = np.array_split(kset, size)
|
|
if "k" in args.usetqdm:
|
|
kpcs[root_node] = tqdm.tqdm(kpcs[root_node], desc="k loop")
|
|
# ----------------------------------------------------------------------
|
|
# define pairs
|
|
|
|
if args.pairfile:
|
|
# if pair file is specified read in pair information from pairfile
|
|
if rank == root_node:
|
|
# read in pair on root node file in a format of five integer columns
|
|
# first two integers define sublattice
|
|
# second three define distance vectors of unitcells
|
|
dummy = np.loadtxt(args.pairfile, dtype=int)
|
|
atran = [
|
|
(dummy[p, 0], dummy[p, 1], [dummy[p, 2], dummy[p, 3], dummy[p, 4]])
|
|
for p in range(len(dummy))
|
|
]
|
|
else:
|
|
atran = None
|
|
# syncronize atran over all nodes
|
|
atran = comm.bcast(atran, root=root_node)
|
|
else:
|
|
# if pairfile is not specified generate pair as defined by npairs and adirs
|
|
# Take pair directions for k directions if adirs is not set
|
|
args.adirs = args.adirs if args.adirs else args.kdirs
|
|
# definition of pairs in terms of integer coordinates refering
|
|
# to unicell distances and atomic positions
|
|
atran = make_atran(len(dh.atoms), args.adirs, dist=args.npairs)
|
|
pairs = []
|
|
for i, j, uc in atran:
|
|
if nl.norm(np.dot(uc, dh.cell)) < args.cutoff:
|
|
pairs.append(
|
|
dict(
|
|
offset=uc, # lattice vector offset between the unitcells the two atoms are
|
|
aiij=[i, j], # indecies of the atoms in the unitcell
|
|
noij=[
|
|
dh.atoms.orbitals[i],
|
|
dh.atoms.orbitals[j],
|
|
], # number of orbitals on the appropriate atoms
|
|
slij=[
|
|
slice(
|
|
*(lambda x: [min(x), max(x) + 1])(dh.a2o(i, all=True))
|
|
), # slices for
|
|
slice(*(lambda x: [min(x), max(x) + 1])(dh.a2o(j, all=True))),
|
|
], # appropriate orbitals
|
|
rirj=[
|
|
dh.axyz()[i],
|
|
dh.axyz()[j],
|
|
], # real space vectors of atoms in the unit cell
|
|
Rij=np.dot(
|
|
uc, dh.cell
|
|
), # real space distance vector between unit cells
|
|
rij=np.dot(uc, dh.cell)
|
|
- dh.axyz()[i]
|
|
+ dh.axyz()[j], # real space vector between atoms
|
|
Jijz=[], # in this empty list are we going to gather the integrad of the energy integral
|
|
Jij=0, # the final results of the calculation are going to be here on the root node
|
|
)
|
|
)
|
|
|
|
if rank == root_node:
|
|
print("Number of pairs beeing calculated: ", len(pairs))
|
|
|
|
comm.Barrier()
|
|
# ----------------------------------------------------------------------
|
|
|
|
# make energy contour
|
|
# we are working in eV now !
|
|
# and sisil shifts E_F to 0 !
|
|
cont = make_contour(emin=args.Ebot, enum=args.eset, p=args.esetp)
|
|
eran = cont.ze
|
|
# ----------------------------------------------------------------------
|
|
|
|
# generating onsite matrix and overalp elements of all the atoms in the unitcell
|
|
# onsite of the origin supercell
|
|
orig_indx = np.arange(0, dh.no) + dh.sc_index([0, 0, 0]) * dh.no
|
|
# spin up
|
|
uc_up = dh.tocsr(dh.UP)[:, orig_indx].toarray()
|
|
# spin down
|
|
uc_down = dh.tocsr(dh.DOWN)[:, orig_indx].toarray()
|
|
Hs = []
|
|
# get number of atoms in the unit cell
|
|
for i in range(len(dh.atoms)):
|
|
at_indx = dh.a2o(i, all=True)
|
|
Hs.append(uc_up[:, at_indx][at_indx, :] - uc_down[:, at_indx][at_indx, :])
|
|
|
|
# ----------------------------------------------------------------------
|
|
|
|
# sampling the integrand on the contour and the BZ
|
|
|
|
for pair in pairs:
|
|
noi, noj = pair["noij"]
|
|
pair["Guij"] = np.zeros((args.eset, noi, noj), dtype="complex128")
|
|
pair["Gdji"] = np.zeros((args.eset, noj, noi), dtype="complex128")
|
|
pair["Guij_tmp"] = np.zeros((args.eset, noi, noj), dtype="complex128")
|
|
pair["Gdji_tmp"] = np.zeros((args.eset, noj, noi), dtype="complex128")
|
|
|
|
for k in kpcs[rank]:
|
|
HKU, HKD, SK = hsk(dh, k)
|
|
Gku = nl.inv(SK * eran.reshape(args.eset, 1, 1) - HKU)
|
|
Gkd = nl.inv(SK * eran.reshape(args.eset, 1, 1) - HKD)
|
|
for pair in pairs:
|
|
phase = np.exp(1j * np.dot(np.dot(k, dh.rcell), pair["Rij"]))
|
|
si, sj = pair["slij"]
|
|
pair["Guij_tmp"] += Gku[:, si, sj] * phase * wk
|
|
pair["Gdji_tmp"] += Gkd[:, sj, si] / phase * wk
|
|
|
|
# summ reduce partial results of mpi nodes
|
|
for pair in pairs:
|
|
comm.Reduce(pair["Guij_tmp"], pair["Guij"], root=root_node)
|
|
comm.Reduce(pair["Gdji_tmp"], pair["Gdji"], root=root_node)
|
|
|
|
if rank == root_node:
|
|
for pair in pairs:
|
|
i, j = pair["aiij"]
|
|
# The Szunyogh-Lichtenstein formula
|
|
pair["Jijz"] = np.trace(
|
|
(Hs[i] @ pair["Guij"]) @ (Hs[j] @ pair["Gdji"]), axis1=1, axis2=2
|
|
)
|
|
# evaluation of the contour integral
|
|
pair["Jij"] = np.trapz(np.imag(pair["Jijz"] * cont.we) / (2 * np.pi))
|
|
end = timer()
|
|
|
|
# ----------------------------------------------------------------------
|
|
# and saveing output of the calculation
|
|
np.savetxt(
|
|
args.outfile,
|
|
np.array(
|
|
[
|
|
[nl.norm(p["rij"]), p["Jij"] * sisl.unit_convert("eV", "Ry") * 1000]
|
|
+ p["aiij"]
|
|
+ list(p["offset"])
|
|
+ list(p["rij"])
|
|
for p in pairs
|
|
],
|
|
dtype=object,
|
|
),
|
|
header=str(args)
|
|
+ "\nnumber of cores = "
|
|
+ str(size)
|
|
+ "\ntime of calculation = "
|
|
+ str(end - start)
|
|
+ "\nnorm(rij),Jij[mRy],aiij,offset,rij",
|
|
fmt="%s",
|
|
)
|