Default backend and performance in ADNLPModels
As illustrated in the tutorial on backends, ADNLPModels.jl
use different backend for each method from the NLPModel API
that are implemented. By default, it uses the following:
using ADNLPModels, NLPModels
f(x) = 100 * (x[2] - x[1]^2)^2 + (x[1] - 1)^2
T = Float64
x0 = T[-1.2; 1.0]
lvar, uvar = zeros(T, 2), ones(T, 2) # must be of same type than `x0`
lcon, ucon = -T[0.5], T[0.5]
c!(cx, x) = begin
cx[1] = x[1] + x[2]
return cx
end
nlp = ADNLPModel!(f, x0, lvar, uvar, c!, lcon, ucon)
get_adbackend(nlp)
ADModelBackend{
ForwardDiffADGradient,
ForwardDiffADHvprod,
ForwardDiffADJprod,
ForwardDiffADJtprod,
SparseADJacobian,
SparseADHessian,
ForwardDiffADGHjvprod,
}
Note that ForwardDiff.jl
is mainly used as it is efficient and stable.
Predefined backends
Another way to know the default backends used is to check the constant ADNLPModels.default_backend
.
ADNLPModels.default_backend
Dict{Symbol, Type} with 12 entries:
:hprod_residual_backend => ForwardDiffADHvprod
:jprod_backend => ForwardDiffADJprod
:jtprod_backend => ForwardDiffADJtprod
:gradient_backend => ForwardDiffADGradient
:hprod_backend => ForwardDiffADHvprod
:hessian_residual_backend => SparseADHessian
:ghjvprod_backend => ForwardDiffADGHjvprod
:hessian_backend => SparseADHessian
:jprod_residual_backend => ForwardDiffADJprod
:jtprod_residual_backend => ForwardDiffADJtprod
:jacobian_residual_backend => SparseADJacobian
:jacobian_backend => SparseADJacobian
More generally, the package anticipates more uses
ADNLPModels.predefined_backend
Dict{Symbol, Dict{Symbol}} with 5 entries:
:zygote => Dict{Symbol, Type}(:hprod_residual_backend=>ForwardDiffADHvprod…
:default => Dict{Symbol, Type}(:hprod_residual_backend=>ForwardDiffADHvprod…
:generic => Dict{Symbol, DataType}(:hprod_residual_backend=>GenericForwardD…
:enzyme => Dict{Symbol, Type}(:hprod_residual_backend=>EnzymeReverseADHvpr…
:optimized => Dict{Symbol, Type}(:hprod_residual_backend=>ReverseDiffADHvprod…
The backend :optimized
will mainly focus on the most efficient approaches, for instance using ReverseDiff
to compute the gradient instead of ForwardDiff
.
ADNLPModels.predefined_backend[:optimized]
Dict{Symbol, Type} with 12 entries:
:hprod_residual_backend => ReverseDiffADHvprod
:jprod_backend => ForwardDiffADJprod
:jtprod_backend => ReverseDiffADJtprod
:gradient_backend => ReverseDiffADGradient
:hprod_backend => ReverseDiffADHvprod
:hessian_residual_backend => SparseReverseADHessian
:ghjvprod_backend => ForwardDiffADGHjvprod
:hessian_backend => SparseReverseADHessian
:jprod_residual_backend => ForwardDiffADJprod
:jtprod_residual_backend => ReverseDiffADJtprod
:jacobian_residual_backend => SparseADJacobian
:jacobian_backend => SparseADJacobian
The backend :generic
focuses on backend that make no assumptions on the element type, see Creating an ADNLPModels backend that supports multiple precisions.
It is possible to use these pre-defined backends using the keyword argument backend
when instantiating the model.
nlp = ADNLPModel!(f, x0, lvar, uvar, c!, lcon, ucon, backend = :optimized)
get_adbackend(nlp)
ADModelBackend{
ReverseDiffADGradient,
ReverseDiffADHvprod,
ForwardDiffADJprod,
ReverseDiffADJtprod,
SparseADJacobian,
SparseReverseADHessian,
ForwardDiffADGHjvprod,
}
The backend :enzyme
focuses on backend based on Enzyme.jl.
nlp = ADNLPModel!(f, x0, lvar, uvar, c!, lcon, ucon, backend = :enzyme)
get_adbackend(nlp)
ADModelBackend{
EnzymeReverseADGradient,
EnzymeReverseADHvprod,
EnzymeReverseADJprod,
EnzymeReverseADJtprod,
SparseEnzymeADJacobian,
SparseEnzymeADHessian,
ForwardDiffADGHjvprod,
}
The backend :zygote
focuses on backend based on Zygote.jl.