Practical error bounds for the forces

DFTK includes an implementation of the strategy from [CDKL2022] to compute practical error bounds for forces and other quantities of interest.

This is an example showing how to compute error estimates for the forces on a ${\rm TiO}_2$ molecule, from which we can either compute asymptotically valid error bounds or increase the precision on the computation of the forces.

using DFTK
using Printf
using LinearAlgebra
using ForwardDiff

Setup

We setup manually the ${\rm TiO}_2$ configuration from Materials Project.

Ti = ElementPsp(:Ti, load_psp("hgh/lda/ti-q4.hgh"))
O  = ElementPsp(:O,  load_psp("hgh/lda/o-q6.hgh"))
atoms     = [Ti, Ti, O, O, O, O]
positions = [[0.5,     0.5,     0.5],  # Ti
             [0.0,     0.0,     0.0],  # Ti
             [0.19542, 0.80458, 0.5],  # O
             [0.80458, 0.19542, 0.5],  # O
             [0.30458, 0.30458, 0.0],  # O
             [0.69542, 0.69542, 0.0]]  # O
lattice   = [[8.79341  0.0      0.0];
             [0.0      8.79341  0.0];
             [0.0      0.0      5.61098]];

We apply a small displacement to one of the $\rm Ti$ atoms to get nonzero forces.

positions[1] .+= [-0.022, 0.028, 0.035]
3-element Vector{Float64}:
 0.478
 0.528
 0.535

We build a model with one $k$-point only, not too high Ecut_ref and small tolerance to limit computational time. These parameters can be increased for more precise results.

model = model_DFT(lattice, atoms, positions; functionals=LDA())
kgrid = [1, 1, 1]
Ecut_ref = 35
basis_ref = PlaneWaveBasis(model; Ecut=Ecut_ref, kgrid)
tol = 1e-5;

We also build a basis with smaller Ecut, to compute a variational approximation of the reference solution.

Choice of convergence parameters

Be careful to choose Ecut not too close to Ecut_ref. Note also that the current choice Ecut_ref = 35 is such that the reference solution is not converged and Ecut = 15 is such that the asymptotic regime (crucial to validate the approach) is barely established.

Ecut = 15
basis = PlaneWaveBasis(model; Ecut, kgrid);

Computations

Compute the solution on the smaller basis:

scfres = self_consistent_field(basis; tol, callback=identity);

Compute first order corrections refinement.δψ and refinement.δρ. Note that refinement.ψ and refinement.ρ are the quantities computed with Ecut and then extended to the reference grid. This step is roughly as expensive as the self_consistent_field call above.

refinement = refine_scfres(scfres, basis_ref; tol, callback=identity);

Error estimates

  • Computation of the force from the variational solution without any post-processing:
f = compute_forces(scfres)
6-element Vector{StaticArraysCore.SVector{3, Float64}}:
 [1.1354184452625729, -1.0152645394508164, 0.4001653714349991]
 [-0.5924264166214437, 0.8448004032874371, 0.4691168761113661]
 [-3.6627223445059123, 1.4344300159896834, -0.4095793772324037]
 [0.9252057057833252, 0.9709773166587574, -0.14998513094082724]
 [2.653082729482258, 0.3994435381730339, -0.12659130483844616]
 [-0.45854366370691224, -2.6344226208303567, -0.1827126595147277]
  • Computation of the forces by a linearization argument when replacing the error $P-P_*$ by the modified residual $R_{\rm Schur}(P)$. The latter quantity is computable in practice.
force_refinement = refine_forces(refinement)
forces_refined = f + force_refinement.dF
6-element Vector{StaticArraysCore.SVector{3, Float64}}:
 [1.2910325057452392, -1.101588826728895, 0.690803095413975]
 [-0.40995236879950514, 0.710974502572875, 0.5337593814169823]
 [-4.20315476795873, 1.625836260227495, -0.6027079536641577]
 [0.8244391246133329, 1.2387717942942058, -0.34575530517229186]
 [3.015574653530292, 0.4906053024523104, -0.09430366510749069]
 [-0.5197714441827641, -2.9624198779741553, -0.17946827058824488]

A practical estimate of the error on the forces is then the following:

dF_estimate = forces_refined - f
6-element Vector{StaticArraysCore.SVector{3, Float64}}:
 [0.15561406048266635, -0.0863242872780785, 0.2906377239789759]
 [0.18247404782193855, -0.13382590071456213, 0.06464250530561622]
 [-0.5404324234528177, 0.1914062442378115, -0.19312857643175407]
 [-0.10076658116999226, 0.2677944776354484, -0.19577017423146462]
 [0.362491924048034, 0.09116176427927647, 0.03228763973095547]
 [-0.06122778047585187, -0.32799725714379857, 0.003244388926482822]

Comparisons against non-practical estimates.

For practical computations one can stop at forces_refined and dF_estimate. We continue here with a comparison of different ways to obtain the refined forces, noting that the computational cost is much higher.

Computations

We compute the reference solution $P_*$ from which we will compute the references forces.

scfres_ref = self_consistent_field(basis_ref; tol, callback=identity)
ψ_ref = DFTK.select_occupied_orbitals(basis_ref, scfres_ref.ψ, scfres_ref.occupation).ψ;
  • Compute the error $P-P_*$ on the associated orbitals $ϕ-ψ$ after aligning them: this is done by solving $\min |ϕ - ψU|$ for $U$ unitary matrix of size $N×N$ ($N$ being the number of electrons) whose solution is $U = S(S^*S)^{-1/2}$ where $S$ is the overlap matrix $ψ^*ϕ$.
function compute_error(ϕ, ψ)
    map(zip(ϕ, ψ)) do (ϕk, ψk)
        S = ψk'ϕk
        U = S*(S'S)^(-1/2)
        ϕk - ψk*U
    end
end
error = compute_error(refinement.ψ, ψ_ref);

Error estimates

We start with different estimations of the forces:

  • Force from the reference solution
f_ref = compute_forces(scfres_ref)
forces   = Dict("F(P_*)" => f_ref)
relerror = Dict("F(P_*)" => 0.0)
compute_relerror(f) = norm(f - f_ref) / norm(f_ref);
  • Force from the variational solution and relative error without any post-processing:
forces["F(P)"]   = f

relerror["F(P)"] = compute_relerror(f);

We then try to improve $F(P)$ using the first order linearization:

\[F(P) = F(P_*) + {\rm d}F(P)·(P-P_*).\]

To this end, we use the ForwardDiff.jl package to compute ${\rm d}F(P)$ using automatic differentiation.

function df(basis, occupation, ψ, δψ, ρ)
    δρ = DFTK.compute_δρ(basis, ψ, δψ, occupation)
    ForwardDiff.derivative(ε -> compute_forces(basis, ψ.+ε.*δψ, occupation; ρ=ρ+ε.*δρ), 0)
end;
  • Computation of the forces by a linearization argument if we have access to the actual error $P-P_*$. Usually this is of course not the case, but this is the "best" improvement we can hope for with a linearisation, so we are aiming for this precision.
df_err = df(basis_ref, refinement.occupation, refinement.ψ,
            DFTK.proj_tangent(error, refinement.ψ), refinement.ρ)
forces["F(P) - df(P)⋅(P-P_*)"]   = f - df_err
relerror["F(P) - df(P)⋅(P-P_*)"] = compute_relerror(f - df_err);
  • Computation of the forces by a linearization argument when replacing the error $P-P_*$ by the modified residual $R_{\rm Schur}(P)$. The latter quantity is computable in practice.
forces["F(P) - df(P)⋅Rschur(P)"]   = forces_refined
relerror["F(P) - df(P)⋅Rschur(P)"] = compute_relerror(forces_refined);

Summary of all forces on the first atom (Ti)

for (key, value) in pairs(forces)
    @printf("%30s = [%7.5f, %7.5f, %7.5f]   (rel. error: %7.5f)\n",
            key, (value[1])..., relerror[key])
end
                        F(P_*) = [1.47893, -1.25373, 0.81010]   (rel. error: 0.00000)
                          F(P) = [1.13542, -1.01526, 0.40017]   (rel. error: 0.20483)
        F(P) - df(P)⋅Rschur(P) = [1.29103, -1.10159, 0.69080]   (rel. error: 0.07836)
          F(P) - df(P)⋅(P-P_*) = [1.50900, -1.28632, 0.86146]   (rel. error: 0.08072)

Notice how close the computable expression $F(P) - {\rm d}F(P)⋅R_{\rm Schur}(P)$ is to the best linearization ansatz $F(P) - {\rm d}F(P)⋅(P-P_*)$.