Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add basic Wannier.jl integration #899

Merged
merged 34 commits into from
Dec 12, 2023
Merged
Show file tree
Hide file tree
Changes from 14 commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
5839ce5
Add basic Wannier.jl integration
Technici4n Oct 30, 2023
c95ba0e
Add Wannier to docs deps
Technici4n Oct 30, 2023
1941d8e
Add kpath to example
Technici4n Oct 30, 2023
a9dc3ee
Fix wannier90 runs
Technici4n Oct 30, 2023
ac9dbb3
Add Wannier to test dependencies
mfherbst Nov 8, 2023
0e65e92
Merge branch 'master' into wannierjl
Technici4n Nov 13, 2023
1e88041
Empty after Wannier 0.3.2 release
Technici4n Nov 13, 2023
c9dc3f3
Try updating CI to 1.8 to see if that fixes checks
Technici4n Nov 13, 2023
502f32e
Optimize Amn computation by using less G vectors
Technici4n Oct 31, 2023
9abd0f9
Add hydrogenic initial Wannier projection
Technici4n Nov 13, 2023
4c2c6dd
Update example to use hydrogenic projections
Technici4n Nov 14, 2023
680df89
Merge remote-tracking branch 'upstream/master' into wannierjl
Technici4n Nov 14, 2023
2ea5d22
Merge remote-tracking branch 'upstream/master' into wannierjl
Technici4n Nov 21, 2023
298fc02
Merge remote-tracking branch 'upstream/master' into wannierjl
Technici4n Dec 8, 2023
1588782
Ensure that output of radial_hydrogenic is always of type T
Technici4n Dec 11, 2023
ace1d4e
`get_wannier_model` -> `wannier_model`
Technici4n Dec 11, 2023
bfdb38a
Remove random_gaussian_projection and fix centres typo
Technici4n Dec 11, 2023
34603d7
Move Wannier compat to extensions, remove WannierProjection abstract …
Technici4n Dec 11, 2023
b5feced
Fix wannier90 ext precompilation
Technici4n Dec 11, 2023
6abe064
Add test for type-stability of radial_hydrogenic
Technici4n Dec 11, 2023
5cf8726
Don't use begin and end in list comprehensions
Technici4n Dec 11, 2023
26e321e
Add basic Wannier.jl test
Technici4n Dec 11, 2023
59477b6
Place kpt parameter right after basis
Technici4n Dec 11, 2023
80e811b
Remove unused wannier_model stub
Technici4n Dec 11, 2023
62a6069
Update wannier example a bit
Technici4n Dec 11, 2023
43f52f4
Use tmpdir for wannierization tests
Technici4n Dec 11, 2023
f9b80c2
Tweak typing annotations
Technici4n Dec 11, 2023
45f6344
for in -> for =
Technici4n Dec 11, 2023
0a3b59b
Use a single SCF for the two Wannier tests
Technici4n Dec 11, 2023
f44c55f
Remove opposite_gaussians_projection
Technici4n Dec 11, 2023
50bbf40
Add custom projection example
Technici4n Dec 12, 2023
53dbc5b
Merge remote-tracking branch 'upstream/master' into wannierjl
Technici4n Dec 12, 2023
b8f5e35
Remove wannier90 example that got added back because of the merge
Technici4n Dec 12, 2023
7bf3a29
don't use begin with inline function definition
Technici4n Dec 12, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion Project.toml
Original file line number Diff line number Diff line change
Expand Up @@ -127,8 +127,9 @@ QuadGK = "1fd47b50-473d-5c70-9696-f719f8f3bcdc"
Random = "9a3f8284-a2c9-5f02-9a11-845980a1fd5c"
Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40"
TestItemRunner = "f8b46487-2199-4994-9208-9a1283c18c0a"
Wannier = "2b19380a-1f7e-4d7d-b1b8-8aa60b3321c9"
WriteVTK = "64499a7a-5c06-52f2-abe2-ccb03c286192"
wannier90_jll = "c5400fa0-8d08-52c2-913f-1e3f656c1ce9"

[targets]
test = ["Test", "TestItemRunner", "ASEconvert", "Aqua", "AtomsIO", "AtomsIOPython", "CUDA", "CUDA_Runtime_jll", "ComponentArrays", "DoubleFloats", "FiniteDiff", "FiniteDifferences", "GenericLinearAlgebra", "IntervalArithmetic", "JLD2", "JSON3", "Logging", "Plots", "QuadGK", "Random", "KrylovKit", "WriteVTK", "wannier90_jll"]
test = ["Test", "TestItemRunner", "ASEconvert", "Aqua", "AtomsIO", "AtomsIOPython", "CUDA", "CUDA_Runtime_jll", "ComponentArrays", "DoubleFloats", "FiniteDiff", "FiniteDifferences", "GenericLinearAlgebra", "IntervalArithmetic", "JLD2", "JSON3", "Logging", "Plots", "QuadGK", "Random", "KrylovKit", "Wannier", "WriteVTK", "wannier90_jll"]
1 change: 1 addition & 0 deletions docs/Project.toml
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ StaticArrays = "90137ffa-7385-5640-81b9-e52037218182"
TimerOutputs = "a759f4b9-e2f1-59dc-863e-4aeb61b1ea8f"
Unitful = "1986cc42-f94f-5a68-af5c-568840ba703d"
UnitfulAtomic = "a7773ee8-282e-5fa2-be4e-bd808c38a91a"
Wannier = "2b19380a-1f7e-4d7d-b1b8-8aa60b3321c9"
WriteVTK = "64499a7a-5c06-52f2-abe2-ccb03c286192"
wannier90_jll = "c5400fa0-8d08-52c2-913f-1e3f656c1ce9"

Expand Down
2 changes: 1 addition & 1 deletion docs/make.jl
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@ PAGES = [
# options we have
"examples/atomsbase.jl",
"examples/input_output.jl",
"examples/wannier90.jl",
"examples/wannier.jl",
],
"Tipps and tricks" => [
# Resolving convergence issues, what solver to use, improving performance or
Expand Down
2 changes: 1 addition & 1 deletion docs/src/features.md
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@
- Integration with [ASE](https://wiki.fysik.dtu.dk/ase/) and
[AtomsBase](https://github.com/JuliaMolSim/AtomsBase.jl) for passing
atomic structures (see [AtomsBase integration](@ref)).
- [Wannierization using Wannier90](@ref)
- [Wannierization using Wannier.jl or Wannier90](@ref)


* Runs out of the box on Linux, macOS and Windows
Expand Down
144 changes: 144 additions & 0 deletions examples/wannier.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
# # Wannierization using Wannier.jl or Wannier90
#
# DFTK features an interface with the programs
# [Wannier.jl](https://wannierjl.org) and [Wannier90](http://www.wannier.org/),
# in order to compute maximally-localized Wannier functions (MLWFs)
# from an initial self consistent field calculation.
# All processes are handled by calling the routine `get_wannier_model` (for Wannier.jl) or `run_wannier90` (for Wannier90).
#
# !!! warning "No guarantees on Wannier interface"
# This code is at an early stage and has so far not been fully tested.
# Bugs are likely and we welcome issues in case you find any!
#
# This example shows how to obtain the MLWFs corresponding
# to the first five bands of graphene. Since the bands 2 to 11 are entangled,
# 15 bands are first computed to obtain 5 MLWFs by a disantanglement procedure.

using DFTK
using Plots
using Unitful
using UnitfulAtomic

d = 10u"Å"
a = 2.641u"Å" # Graphene Lattice constant
lattice = [a -a/2 0;
0 √3*a/2 0;
0 0 d]

C = ElementPsp(:C; psp=load_psp("hgh/pbe/c-q4"))
atoms = [C, C]
positions = [[0.0, 0.0, 0.0], [1//3, 2//3, 0.0]]
model = model_PBE(lattice, atoms, positions)
basis = PlaneWaveBasis(model; Ecut=15, kgrid=[5, 5, 1])
nbandsalg = AdaptiveBands(basis.model; n_bands_converge=15)
scfres = self_consistent_field(basis; nbandsalg, tol=1e-5);

# Plot bandstructure of the system

plot_bandstructure(scfres; kline_density=10)

# ## Wannierization with Wannier.jl
#
# Now we use the `get_wannier_model` routine to generate a Wannier.jl model
# that can be used to perform the wannierization procedure.
# For now, this model generation produces file in the Wannier90 convention,
# where all files are named with the same prefix and only differ by
# their extensions. By default all generated input and output files are stored
# in the subfolder "wannierjl" under the prefix "wannier" (i.e. "wannierjl/wannier.win",
# "wannierjl/wannier.wout", etc.). A different file prefix can be given with the
# keyword argument `fileprefix` as shown below.
#
# We now produce a simple Wannier model for 5 MLFWs.
#
# For a good MLWF, we need to provide initial projections that resemble the expected shape
# of the Wannier functions.
# Here we will use:
# - 3 bond-centered 2s hydrogenic orbitals for the expected σ bonds
# - 2 atom-centered 2pz hydrogenic orbitals for the expected π bands

using Wannier # Needed to make get_wannier_model available

## Helper functions to produce hydrogenic starting guesses
s_guess(center) = DFTK.HydrogenicWannierProjection(center, 2, 0, 0, C.Z)
pz_guess(center) = DFTK.HydrogenicWannierProjection(center, 2, 1, 0, C.Z)

wannier_model = get_wannier_model(scfres;
fileprefix="wannier/graphene",
n_bands=scfres.n_bands_converge,
n_wannier=5,
projections=[
## Note: fractional coordinates for the centers!
## 3 bond-centered 2s hydrogenic orbitals to imitate σ bonds
s_guess((positions[1] + positions[2]) / 2),
s_guess((positions[1] + positions[2] + [0, -1, 0]) / 2),
s_guess((positions[1] + positions[2] + [-1, -1, 0]) / 2),
## 2 atom-centered 2pz hydrogenic orbitals
pz_guess(positions[1]),
pz_guess(positions[2]),
],
dis_froz_max=ustrip(auconvert(u"eV", scfres.εF))+1) # maximum frozen window, for example 1 eV above Fermi level

# Once we have the `wannier_model`, we can use the functions in the Wannier.jl package:
#
# Compute MLWF:
U = disentangle(wannier_model, max_iter=200);

# Inspect localization before and after Wannierization:
omega(wannier_model)
omega(wannier_model, U)

# Build a Wannier interpolation model:
kpath = irrfbz_path(model)
interp_model = Wannier.InterpModel(wannier_model; kpath=kpath)

# And so on...
# Refer to the Wannier.jl documentation for further examples.

# (Delete temporary files when done.)
rm("wannier", recursive=true)

# This example assumes that Wannier.jl version 0.3.2 is used,
# and may need to be updated to accomodate for changes in Wannier.jl.
#
# Note: Some parameters supported by Wannier90 may have to be passed to Wannier.jl differently,
# for example the max number of iterations is passed to `disentangle` in Wannier.jl,
# but as `num_iter` to `run_wannier90`.

# ## Wannierization with Wannier90
#
# We can use the `run_wannier90` routine to generate all required files and perform the wannierization procedure:

using wannier90_jll # Needed to make run_wannier90 available
run_wannier90(scfres;
fileprefix="wannier/graphene",
n_wannier=5,
projections=[
## Note: fractional coordinates for the centers!
## 3 bond-centered 2s hydrogenic orbitals to imitate σ bonds
s_guess((positions[1] + positions[2]) / 2),
s_guess((positions[1] + positions[2] + [0, -1, 0]) / 2),
s_guess((positions[1] + positions[2] + [-1, -1, 0]) / 2),
## 2 atom-centered 2pz hydrogenic orbitals
pz_guess(positions[1]),
pz_guess(positions[2]),
],
mfherbst marked this conversation as resolved.
Show resolved Hide resolved
num_print_cycles=25,
num_iter=200,
##
dis_win_max=19.0,
dis_froz_max=ustrip(auconvert(u"eV", scfres.εF))+1, # 1 eV above Fermi level
dis_num_iter=300,
dis_mix_ratio=1.0,
##
wannier_plot=true,
wannier_plot_format="cube",
wannier_plot_supercell=5,
write_xyz=true,
translate_home_cell=true,
);

# As can be observed standard optional arguments for the disentanglement
# can be passed directly to `run_wannier90` as keyword arguments.

# (Delete temporary files.)
rm("wannier", recursive=true)
72 changes: 0 additions & 72 deletions examples/wannier90.jl

This file was deleted.

6 changes: 6 additions & 0 deletions src/DFTK.jl
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ include("common/zeros_like.jl")
include("common/norm.jl")
include("common/quadrature.jl")
include("common/hankel.jl")
include("common/hydrogenic.jl")
mfherbst marked this conversation as resolved.
Show resolved Hide resolved

export PspHgh
export PspUpf
Expand Down Expand Up @@ -184,9 +185,11 @@ include("pseudo/list_psp.jl")
include("pseudo/attach_psp.jl")

export atomic_system, periodic_system # Reexport from AtomsBase
export get_wannier_model
export run_wannier90
include("external/atomsbase.jl")
include("external/stubs.jl") # Function stubs for conditionally defined methods
include("external/wannier_shared.jl")

export compute_bands
export plot_bandstructure
Expand Down Expand Up @@ -240,6 +243,9 @@ function __init__()
@require wannier90_jll="c5400fa0-8d08-52c2-913f-1e3f656c1ce9" begin
include("external/wannier90.jl")
end
@require Wannier="2b19380a-1f7e-4d7d-b1b8-8aa60b3321c9" begin
include("external/wannier.jl")
end

# TODO Move out into extension module
@require CUDA="052768ef-5323-5732-b1bb-66c8b64840ba" begin
Expand Down
22 changes: 22 additions & 0 deletions src/common/hydrogenic.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
@doc raw"""
Radial functions from solutions of Hydrogenic Schrödinger equation.
Same as Wannier90 user guide Table 3.3.
# Arguments
- `r`: radial grid
- `n`: principal quantum number
- `α`: diffusivity, ``\frac{Z}/{a}`` where ``Z`` is the atomic number and
``a`` is the Bohr radius.
"""
function radial_hydrogenic(r::AbstractVector{T}, n::Integer, α::Real=1.0) where {T<:Real}
Technici4n marked this conversation as resolved.
Show resolved Hide resolved
@assert n > 0
if n == 1
f = 2 * α^(3/2) * exp.(-α * r)

Check warning on line 13 in src/common/hydrogenic.jl

View check run for this annotation

Codecov / codecov/patch

src/common/hydrogenic.jl#L13

Added line #L13 was not covered by tests
elseif n == 2
f = 2^(-3/2) * α^(3/2) * (2 .- α * r) .* exp.(-α * r/2)
elseif n == 3
f = sqrt(4/27) * α^(3/2) * (1 .- 2/3 * α * r .+ 2/27 * α^2 * r.^2) .* exp.(-α * r/3)

Check warning on line 17 in src/common/hydrogenic.jl

View check run for this annotation

Codecov / codecov/patch

src/common/hydrogenic.jl#L16-L17

Added lines #L16 - L17 were not covered by tests
else
error("n = $n is not supported")

Check warning on line 19 in src/common/hydrogenic.jl

View check run for this annotation

Codecov / codecov/patch

src/common/hydrogenic.jl#L19

Added line #L19 was not covered by tests
end
f
end
20 changes: 17 additions & 3 deletions src/external/stubs.jl
Original file line number Diff line number Diff line change
@@ -1,13 +1,27 @@
# Stubs for conditionally defined functions

"""
Wannerize the obtained bands using wannier90. By default all converged
Build a Wannier.jl model for the obtained bands. By default all converged
bands from the `scfres` are employed (change with `n_bands` kwargs)
and `n_wannier = n_bands` wannier functions are computed using
random Gaussians as guesses. All keyword arguments supported by
and `n_wannier = n_bands` wannier functions are computed.
Random Gaussians are used as guesses by default, can be changed using the `projections` kwarg.
All keyword arguments supported by
Wannier90 for the disentanglement may be added as keyword arguments.
The function returns the `fileprefix`.

!!! warning "Experimental feature"
Currently this is an experimental feature, which has not yet been tested
to full depth. The interface is considered unstable and may change
incompatibly in the future. Use at your own risk and please report bugs
in case you encounter any.
"""
function get_wannier_model end

"""
Wannerize the obtained bands using wannier90.
Same arguments as `get_wannier_model`.
Technici4n marked this conversation as resolved.
Show resolved Hide resolved
The function returns the `fileprefix`.

!!! warning "Experimental feature"
Currently this is an experimental feature, which has not yet been tested
to full depth. The interface is considered unstable and may change
Expand Down
36 changes: 36 additions & 0 deletions src/external/wannier.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
"""
Use Wannier.jl to read a .win file and produce the nnkp data (i.e. the b-vectors used in the Mmn matrix).
"""
function get_nnkpt_from_wannier(fileprefix)
model = Wannier.read_w90(fileprefix; amn=false, mmn=false, eig=false)
bvectors = model.bvectors

nnkpts = reduce(vcat,
map(1:model.n_kpts) do ik
zp = zip(bvectors.kpb_k[:, ik], eachcol(bvectors.kpb_b[:, :, ik]))
map(zp) do (ikpb, G_shift)
(ik, ikpb, G_shift=copy(G_shift))
end
end)

(nntot=model.n_bvecs, nnkpts)
end

"""
Build a Wannier.jl model that can be used for Wannierization.
"""
@timing function get_wannier_model(scfres;

Check warning on line 22 in src/external/wannier.jl

View check run for this annotation

Codecov / codecov/patch

src/external/wannier.jl#L22

Added line #L22 was not covered by tests
mfherbst marked this conversation as resolved.
Show resolved Hide resolved
n_bands=scfres.n_bands_converge,
n_wannier=n_bands,
projections::AbstractVector{<:WannierProjection}=default_wannier_centres(n_wannier),
fileprefix=joinpath("wannierjl", "wannier"),
wannier_plot=false,
kwargs...)
# Write the files
write_wannier90_files(scfres; n_bands, n_wannier, projections, fileprefix, wannier_plot, kwargs...) do
get_nnkpt_from_wannier(fileprefix)
end

# Read Wannier.jl model
Wannier.read_w90(fileprefix)
end
Loading
Loading