From 3d54606e2f6751484724be94df758ef147ffb514 Mon Sep 17 00:00:00 2001 From: Avik Pal Date: Fri, 20 Oct 2023 18:20:46 -0400 Subject: [PATCH] Broyden with LineSearch --- src/NonlinearSolve.jl | 4 +- src/broyden.jl | 91 +++++++++++++++++++++---- src/klement.jl | 1 + src/linesearch.jl | 143 ++++++++++++++++++++++++++++++++++++--- src/raphson.jl | 5 +- src/utils.jl | 14 +++- test/23_test_problems.jl | 41 ++++++++--- test/basictests.jl | 90 ++++++++++++++++++++++++ 8 files changed, 352 insertions(+), 37 deletions(-) diff --git a/src/NonlinearSolve.jl b/src/NonlinearSolve.jl index ba93d0cf3..9b26e530b 100644 --- a/src/NonlinearSolve.jl +++ b/src/NonlinearSolve.jl @@ -25,6 +25,8 @@ import UnPack: @unpack const AbstractSparseADType = Union{ADTypes.AbstractSparseFiniteDifferences, ADTypes.AbstractSparseForwardMode, ADTypes.AbstractSparseReverseMode} +abstract type AbstractNonlinearSolveLineSearchAlgorithm end + abstract type AbstractNonlinearSolveAlgorithm <: AbstractNonlinearAlgorithm end abstract type AbstractNewtonAlgorithm{CJ, AD} <: AbstractNonlinearSolveAlgorithm end @@ -104,6 +106,6 @@ export NewtonRaphson, TrustRegion, LevenbergMarquardt, DFSane, GaussNewton, Pseu export LeastSquaresOptimJL, FastLevenbergMarquardtJL export RobustMultiNewton, FastShortcutNonlinearPolyalg -export LineSearch +export LineSearch, LiFukushimaLineSearch end # module diff --git a/src/broyden.jl b/src/broyden.jl index 6daf9ef1e..121a2ca71 100644 --- a/src/broyden.jl +++ b/src/broyden.jl @@ -1,19 +1,28 @@ # Sadly `Broyden` is taken up by SimpleNonlinearSolve.jl """ - GeneralBroyden(max_resets) - GeneralBroyden(; max_resets = 3) + GeneralBroyden(max_resets, linesearch) + GeneralBroyden(; max_resets = 3, linesearch = LineSearch()) -An implementation of `Broyden` with support for caching! +An implementation of `Broyden` with reseting and line search. ## Arguments - `max_resets`: the maximum number of resets to perform. Defaults to `3`. + - `linesearch`: the line search algorithm to use. Defaults to [`LineSearch()`](@ref), + which means that no line search is performed. Algorithms from `LineSearches.jl` can be + used here directly, and they will be converted to the correct `LineSearch`. It is + recommended to use [LiFukushimaLineSearchCache](@ref) -- a derivative free linesearch + specifically designed for Broyden's method. """ -struct GeneralBroyden <: AbstractNewtonAlgorithm{false, Nothing} +@concrete struct GeneralBroyden <: AbstractNewtonAlgorithm{false, Nothing} max_resets::Int + linesearch end -GeneralBroyden(; max_resets = 3) = GeneralBroyden(max_resets) +function GeneralBroyden(; max_resets = 3, linesearch = LineSearch()) + linesearch = linesearch isa LineSearch ? linesearch : LineSearch(; method = linesearch) + return GeneralBroyden(max_resets, linesearch) +end @concrete mutable struct GeneralBroydenCache{iip} <: AbstractNonlinearSolveCache{iip} f @@ -29,13 +38,14 @@ GeneralBroyden(; max_resets = 3) = GeneralBroyden(max_resets) J⁻¹df force_stop::Bool resets::Int - max_rests::Int + max_resets::Int maxiters::Int internalnorm retcode::ReturnCode.T abstol prob stats::NLStats + lscache end get_fu(cache::GeneralBroydenCache) = cache.fu @@ -46,11 +56,11 @@ function SciMLBase.__init(prob::NonlinearProblem{uType, iip}, alg::GeneralBroyde @unpack f, u0, p = prob u = alias_u0 ? u0 : deepcopy(u0) fu = evaluate_f(prob, u) - J⁻¹ = convert(parameterless_type(_mutable(u)), - Matrix{eltype(u)}(I, length(fu), length(u))) - return GeneralBroydenCache{iip}(f, alg, u, _mutable_zero(u), fu, similar(fu), - similar(fu), p, J⁻¹, similar(fu'), _mutable_zero(u), false, 0, alg.max_resets, - maxiters, internalnorm, ReturnCode.Default, abstol, prob, NLStats(1, 0, 0, 0, 0)) + J⁻¹ = __init_identity_jacobian(u, fu) + return GeneralBroydenCache{iip}(f, alg, u, _mutable_zero(u), fu, zero(fu), + zero(fu), p, J⁻¹, zero(fu'), _mutable_zero(u), false, 0, alg.max_resets, + maxiters, internalnorm, ReturnCode.Default, abstol, prob, NLStats(1, 0, 0, 0, 0), + init_linesearch_cache(alg.linesearch, f, u, p, fu, Val(iip))) end function perform_step!(cache::GeneralBroydenCache{true}) @@ -58,7 +68,8 @@ function perform_step!(cache::GeneralBroydenCache{true}) T = eltype(u) mul!(du, J⁻¹, -fu) - u .+= du + α = perform_linesearch!(cache.lscache, u, du) + axpy!(α, du, u) f(fu2, u, p) cache.internalnorm(fu2) < cache.abstol && (cache.force_stop = true) @@ -68,7 +79,7 @@ function perform_step!(cache::GeneralBroydenCache{true}) # Update the inverse jacobian dfu .= fu2 .- fu - if cache.resets < cache.max_rests && + if cache.resets < cache.max_resets && (all(x -> abs(x) ≤ 1e-12, du) || all(x -> abs(x) ≤ 1e-12, dfu)) fill!(J⁻¹, 0) J⁻¹[diagind(J⁻¹)] .= T(1) @@ -83,3 +94,57 @@ function perform_step!(cache::GeneralBroydenCache{true}) return nothing end + +function perform_step!(cache::GeneralBroydenCache{false}) + @unpack f, p = cache + T = eltype(cache.u) + + cache.du = cache.J⁻¹ * -cache.fu + α = perform_linesearch!(cache.lscache, cache.u, cache.du) + cache.u = cache.u .+ α * cache.du + cache.fu2 = f(cache.u, p) + + cache.internalnorm(cache.fu2) < cache.abstol && (cache.force_stop = true) + cache.stats.nf += 1 + + cache.force_stop && return nothing + + # Update the inverse jacobian + cache.dfu = cache.fu2 .- cache.fu + if cache.resets < cache.max_resets && + (all(x -> abs(x) ≤ 1e-12, cache.du) || all(x -> abs(x) ≤ 1e-12, cache.dfu)) + J⁻¹ = similar(cache.J⁻¹) + fill!(J⁻¹, 0) + J⁻¹[diagind(J⁻¹)] .= T(1) + cache.J⁻¹ = J⁻¹ + cache.resets += 1 + else + cache.J⁻¹df = cache.J⁻¹ * cache.dfu + cache.J⁻¹₂ = cache.du' * cache.J⁻¹ + cache.du = (cache.du .- cache.J⁻¹df) ./ (dot(cache.du, cache.J⁻¹df) .+ T(1e-5)) + cache.J⁻¹ = cache.J⁻¹ .+ cache.du * cache.J⁻¹₂ + end + cache.fu = cache.fu2 + + return nothing +end + +function SciMLBase.reinit!(cache::GeneralBroydenCache{iip}, u0 = cache.u; p = cache.p, + abstol = cache.abstol, maxiters = cache.maxiters) where {iip} + cache.p = p + if iip + recursivecopy!(cache.u, u0) + cache.f(cache.fu, cache.u, p) + else + # don't have alias_u0 but cache.u is never mutated for OOP problems so it doesn't matter + cache.u = u0 + cache.fu = cache.f(cache.u, p) + end + cache.abstol = abstol + cache.maxiters = maxiters + cache.stats.nf = 1 + cache.stats.nsteps = 1 + cache.force_stop = false + cache.retcode = ReturnCode.Default + return cache +end diff --git a/src/klement.jl b/src/klement.jl index e69de29bb..8b1378917 100644 --- a/src/klement.jl +++ b/src/klement.jl @@ -0,0 +1 @@ + diff --git a/src/linesearch.jl b/src/linesearch.jl index 1ef36206a..1f63d695c 100644 --- a/src/linesearch.jl +++ b/src/linesearch.jl @@ -26,7 +26,15 @@ function LineSearch(; method = Static(), autodiff = AutoFiniteDiff(), alpha = tr return LineSearch(method, autodiff, alpha) end -@concrete mutable struct LineSearchCache +@inline function init_linesearch_cache(ls::LineSearch, args...) + return init_linesearch_cache(ls.method, ls, args...) +end + +# LineSearches.jl doesn't have a supertype so default to that +init_linesearch_cache(_, ls, f, u, p, fu, iip) = LineSearchesJLCache(ls, f, u, p, fu, iip) + +# Wrapper over LineSearches.jl algorithms +@concrete mutable struct LineSearchesJLCache f ϕ dϕ @@ -35,11 +43,11 @@ end ls end -function LineSearchCache(ls::LineSearch, f, u::Number, p, _, ::Val{false}) +function LineSearchesJLCache(ls::LineSearch, f, u::Number, p, _, ::Val{false}) eval_f(u, du, α) = eval_f(u - α * du) eval_f(u) = f(u, p) - ls.method isa Static && return LineSearchCache(eval_f, nothing, nothing, nothing, + ls.method isa Static && return LineSearchesJLCache(eval_f, nothing, nothing, nothing, convert(typeof(u), ls.α), ls) g(u, fu) = last(value_derivative(Base.Fix2(f, p), u)) * fu @@ -73,11 +81,11 @@ function LineSearchCache(ls::LineSearch, f, u::Number, p, _, ::Val{false}) return ϕdϕ_internal end - return LineSearchCache(eval_f, ϕ, dϕ, ϕdϕ, convert(eltype(u), ls.α), ls) + return LineSearchesJLCache(eval_f, ϕ, dϕ, ϕdϕ, convert(eltype(u), ls.α), ls) end -function LineSearchCache(ls::LineSearch, f, u, p, fu1, IIP::Val{iip}) where {iip} - fu = iip ? fu1 : nothing +function LineSearchesJLCache(ls::LineSearch, f, u, p, fu1, IIP::Val{iip}) where {iip} + fu = iip ? deepcopy(fu1) : nothing u_ = _mutable_zero(u) function eval_f(u, du, α) @@ -86,7 +94,7 @@ function LineSearchCache(ls::LineSearch, f, u, p, fu1, IIP::Val{iip}) where {iip end eval_f(u) = evaluate_f(f, u, p, IIP; fu) - ls.method isa Static && return LineSearchCache(eval_f, nothing, nothing, nothing, + ls.method isa Static && return LineSearchesJLCache(eval_f, nothing, nothing, nothing, convert(eltype(u), ls.α), ls) g₀ = _mutable_zero(u) @@ -138,10 +146,10 @@ function LineSearchCache(ls::LineSearch, f, u, p, fu1, IIP::Val{iip}) where {iip return ϕdϕ_internal end - return LineSearchCache(eval_f, ϕ, dϕ, ϕdϕ, convert(eltype(u), ls.α), ls) + return LineSearchesJLCache(eval_f, ϕ, dϕ, ϕdϕ, convert(eltype(u), ls.α), ls) end -function perform_linesearch!(cache::LineSearchCache, u, du) +function perform_linesearch!(cache::LineSearchesJLCache, u, du) cache.ls.method isa Static && return cache.α ϕ = cache.ϕ(u, du) @@ -155,3 +163,120 @@ function perform_linesearch!(cache::LineSearchCache, u, du) return first(cache.ls.method(ϕ, cache.dϕ(u, du), cache.ϕdϕ(u, du), cache.α, ϕ₀, dϕ₀)) end + +""" + LiFukushimaLineSearch(; lambda_0 = 1.0, beta = 0.5, sigma_1 = 0.001, + eta = 0.1, nan_max_iter = 5, maxiters = 50) + +A derivative-free line search and global convergence of Broyden-like method for nonlinear +equations by Dong-Hui Li & Masao Fukushima. For more details see +https://doi.org/10.1080/10556780008805782 +""" +struct LiFukushimaLineSearch{T} <: AbstractNonlinearSolveLineSearchAlgorithm + λ₀::T + β::T + σ₁::T + σ₂::T + η::T + ρ::T + nan_max_iter::Int + maxiters::Int +end + +function LiFukushimaLineSearch(; lambda_0 = 1.0, beta = 0.1, sigma_1 = 0.001, + sigma_2 = 0.001, eta = 0.1, rho = 0.9, nan_max_iter = 5, maxiters = 50) + T = promote_type(typeof(lambda_0), typeof(beta), typeof(sigma_1), typeof(eta), + typeof(rho), typeof(sigma_2)) + return LiFukushimaLineSearch{T}(lambda_0, beta, sigma_1, sigma_2, eta, rho, + nan_max_iter, maxiters) +end + +@concrete mutable struct LiFukushimaLineSearchCache{iip} + f + p + u_cache + fu_cache + alg + α +end + +function init_linesearch_cache(alg::LiFukushimaLineSearch, ls::LineSearch, f, _u, p, _fu, + ::Val{iip}) where {iip} + fu = iip ? deepcopy(_fu) : nothing + u = iip ? deepcopy(_u) : nothing + return LiFukushimaLineSearchCache{iip}(f, p, u, fu, alg, ls.α) +end + +function perform_linesearch!(cache::LiFukushimaLineSearchCache{iip}, u, du) where {iip} + (; β, σ₁, σ₂, η, λ₀, ρ, nan_max_iter, maxiters) = cache.alg + λ₂ = λ₀ + λ₁ = λ₂ + + if iip + cache.f(cache.fu_cache, u, cache.p) + fx_norm = norm(cache.fu_cache, 2) + else + fx_norm = norm(cache.f(u, cache.p), 2) + end + + # Non-Blocking exit if the norm is NaN or Inf + !isfinite(fx_norm) && return cache.α + + # Early Terminate based on Eq. 2.7 + if iip + cache.u_cache .= u .+ du + cache.f(cache.fu_cache, cache.u_cache, cache.p) + fxλ_norm = norm(cache.fu_cache, 2) + else + fxλ_norm = norm(cache.f(u .+ du, cache.p), 2) + end + + fxλ_norm ≤ ρ * fx_norm - σ₂ * norm(du, 2)^2 && return cache.α + + if iip + cache.u_cache .= u .+ λ₂ .* du + cache.f(cache.fu_cache, cache.u_cache, cache.p) + fxλp_norm = norm(cache.fu_cache, 2) + else + fxλp_norm = norm(cache.f(u .+ λ₂ .* du, cache.p), 2) + end + + if !isfinite(fxλp_norm) + # Backtrack a finite number of steps + nan_converged = false + for _ in 1:nan_max_iter + λ₁, λ₂ = λ₂, β * λ₂ + + if iip + cache.u_cache .= u .+ λ₂ .* du + cache.f(cache.fu_cache, cache.u_cache, cache.p) + fxλp_norm = norm(cache.fu_cache, 2) + else + fxλp_norm = norm(cache.f(u .+ λ₂ .* du, cache.p), 2) + end + + nan_converged = isfinite(fxλp_norm) + nan_converged && break + end + + # Non-Blocking exit if the norm is still NaN or Inf + !nan_converged && return cache.α + end + + for _ in 1:maxiters + if iip + cache.u_cache .= u .+ λ₂ .* du + cache.f(cache.fu_cache, cache.u_cache, cache.p) + fxλp_norm = norm(cache.fu_cache, 2) + else + fxλp_norm = norm(cache.f(u .+ λ₂ .* du, cache.p), 2) + end + + converged = fxλp_norm ≤ (1 + η) * fx_norm - σ₁ * λ₂^2 * norm(du, 2)^2 + + converged && break + λ₁, λ₂ = λ₂, β * λ₂ + end + + return λ₂ +end diff --git a/src/raphson.jl b/src/raphson.jl index 642c7ee36..a3d9bbc6d 100644 --- a/src/raphson.jl +++ b/src/raphson.jl @@ -82,7 +82,8 @@ function SciMLBase.__init(prob::NonlinearProblem{uType, iip}, alg_::NewtonRaphso return NewtonRaphsonCache{iip}(f, alg, u, fu1, fu2, du, p, uf, linsolve, J, jac_cache, false, maxiters, internalnorm, ReturnCode.Default, abstol, prob, - NLStats(1, 0, 0, 0, 0), LineSearchCache(alg.linesearch, f, u, p, fu1, Val(iip))) + NLStats(1, 0, 0, 0, 0), + init_linesearch_cache(alg.linesearch, f, u, p, fu1, Val(iip))) end function perform_step!(cache::NewtonRaphsonCache{true}) @@ -96,7 +97,7 @@ function perform_step!(cache::NewtonRaphsonCache{true}) # Line Search α = perform_linesearch!(cache.lscache, u, du) - @. u = u - α * du + axpy!(α, du, u) f(cache.fu1, u, p) cache.internalnorm(fu1) < cache.abstol && (cache.force_stop = true) diff --git a/src/utils.jl b/src/utils.jl index 173c279be..5309ad120 100644 --- a/src/utils.jl +++ b/src/utils.jl @@ -35,8 +35,8 @@ function default_adargs_to_adtype(; chunk_size = missing, autodiff = nothing, if chunk_size !== missing || standardtag !== missing || diff_type !== missing || autodiff !== missing Base.depwarn("`chunk_size`, `standardtag`, `diff_type`, \ - `autodiff::Union{Val, Bool}` kwargs have been deprecated and will be removed in\ - v3. Update your code to directly specify autodiff=", + `autodiff::Union{Val, Bool}` kwargs have been deprecated and will be removed \ + in v3. Update your code to directly specify autodiff=", :default_adargs_to_adtype) end chunk_size === missing && (chunk_size = Val{0}()) @@ -208,3 +208,13 @@ function __get_concrete_algorithm(alg, prob) end return set_ad(alg, ad) end + +__init_identity_jacobian(u::Number, _) = u +function __init_identity_jacobian(u, fu) + return convert(parameterless_type(_mutable(u)), + Matrix{eltype(u)}(I, length(fu), length(u))) +end +function __init_identity_jacobian(u::StaticArray, fu) + return convert(MArray{Tuple{length(fu), length(u)}}, + Matrix{eltype(u)}(I, length(fu), length(u))) +end \ No newline at end of file diff --git a/test/23_test_problems.jl b/test/23_test_problems.jl index 5e0df8352..8c739c085 100644 --- a/test/23_test_problems.jl +++ b/test/23_test_problems.jl @@ -8,19 +8,28 @@ function test_on_library(problems, dicts, alg_ops, broken_tests, ϵ = 1e-4) x = dict["start"] res = similar(x) nlprob = NonlinearProblem(problem, x) - @testset "$(dict["title"])" begin + @testset "$idx: $(dict["title"])" begin for alg in alg_ops - sol = solve(nlprob, alg, abstol = 1e-18, reltol = 1e-18) - problem(res, sol.u, nothing) - broken = idx in broken_tests[alg] ? true : false - @test norm(res)≤ϵ broken=broken + try + sol = solve(nlprob, alg, abstol = 1e-18, reltol = 1e-18) + problem(res, sol.u, nothing) + broken = idx in broken_tests[alg] ? true : false + @test norm(res)≤ϵ broken=broken + catch + broken = idx in broken_tests[alg] ? true : false + if broken + @test false broken=true + else + @test 1 == 2 + end + end end end end end # NewtonRaphson -@testset "NewtonRaphson test problem library" begin +@testset "NewtonRaphson 23 Test Problems" begin alg_ops = (NewtonRaphson(),) # dictionary with indices of test problems where method does not converge to small residual @@ -30,7 +39,7 @@ end test_on_library(problems, dicts, alg_ops, broken_tests) end -@testset "TrustRegion test problem library" begin +@testset "TrustRegion 23 Test Problems" begin alg_ops = (TrustRegion(; radius_update_scheme = RadiusUpdateSchemes.Simple), TrustRegion(; radius_update_scheme = RadiusUpdateSchemes.Fan), TrustRegion(; radius_update_scheme = RadiusUpdateSchemes.Hei), @@ -50,7 +59,7 @@ end test_on_library(problems, dicts, alg_ops, broken_tests) end -@testset "TrustRegion test problem library" begin +@testset "LevenbergMarquardt 23 Test Problems" begin alg_ops = (LevenbergMarquardt(; linsolve = NormalCholeskyFactorization()), LevenbergMarquardt(; α_geodesic = 0.1, linsolve = NormalCholeskyFactorization())) @@ -62,8 +71,7 @@ end test_on_library(problems, dicts, alg_ops, broken_tests) end -# DFSane -@testset "DFSane test problem library" begin +@testset "DFSane 23 Test Problems" begin alg_ops = (DFSane(),) broken_tests = Dict(alg => Int[] for alg in alg_ops) @@ -71,3 +79,16 @@ end test_on_library(problems, dicts, alg_ops, broken_tests) end + +@testset "GeneralBroyden 23 Test Problems" begin + alg_ops = (GeneralBroyden(), + GeneralBroyden(; linesearch = LiFukushimaLineSearch(; beta = 0.1)), + GeneralBroyden(; linesearch = BackTracking())) + + broken_tests = Dict(alg => Int[] for alg in alg_ops) + broken_tests[alg_ops[1]] = [1, 3, 4, 5, 6, 8, 11, 12, 13, 14, 21] + broken_tests[alg_ops[2]] = [1, 2, 3, 4, 5, 6, 9, 11, 13, 22] + broken_tests[alg_ops[3]] = [1, 2, 4, 5, 6, 8, 11, 12, 13, 14, 21] + + test_on_library(problems, dicts, alg_ops, broken_tests) +end diff --git a/test/basictests.jl b/test/basictests.jl index f20417785..22ba9ab25 100644 --- a/test/basictests.jl +++ b/test/basictests.jl @@ -664,3 +664,93 @@ end @test all(abs.(newton_fails(sol.u, p)) .< 1e-10) end end + +# --- GeneralBroyden tests --- + +@testset "GeneralBroyden" begin + function benchmark_nlsolve_oop(f, u0, p = 2.0; linesearch = LineSearch()) + prob = NonlinearProblem{false}(f, u0, p) + return solve(prob, GeneralBroyden(; linesearch), abstol = 1e-9) + end + + function benchmark_nlsolve_iip(f, u0, p = 2.0; linesearch = LineSearch()) + prob = NonlinearProblem{true}(f, u0, p) + return solve(prob, GeneralBroyden(; linesearch), abstol = 1e-9) + end + + @testset "LineSearch: $(_nameof(lsmethod)) LineSearch AD: $(_nameof(ad))" for lsmethod in (Static(), + StrongWolfe(), BackTracking(), HagerZhang(), MoreThuente(), + LiFukushimaLineSearch()), + ad in (AutoFiniteDiff(), AutoZygote()) + + linesearch = LineSearch(; method = lsmethod, autodiff = ad) + u0s = ([1.0, 1.0], @SVector[1.0, 1.0], 1.0) + + @testset "[OOP] u0: $(typeof(u0))" for u0 in u0s + sol = benchmark_nlsolve_oop(quadratic_f, u0; linesearch) + @test SciMLBase.successful_retcode(sol) + @test all(abs.(sol.u .* sol.u .- 2) .< 1e-9) + + cache = init(NonlinearProblem{false}(quadratic_f, u0, 2.0), + GeneralBroyden(; linesearch), abstol = 1e-9) + @test (@ballocated solve!($cache)) < 200 + end + + @testset "[IIP] u0: $(typeof(u0))" for u0 in ([1.0, 1.0],) + ad isa AutoZygote && continue + sol = benchmark_nlsolve_iip(quadratic_f!, u0; linesearch) + @test SciMLBase.successful_retcode(sol) + @test all(abs.(sol.u .* sol.u .- 2) .< 1e-9) + + cache = init(NonlinearProblem{true}(quadratic_f!, u0, 2.0), + GeneralBroyden(; linesearch), abstol = 1e-9) + @test (@ballocated solve!($cache)) ≤ 64 + end + end + + @testset "[OOP] [Immutable AD]" begin + for p in 1.0:0.1:100.0 + @test begin + res = benchmark_nlsolve_oop(quadratic_f, @SVector[1.0, 1.0], p) + res_true = sqrt(p) + all(res.u .≈ res_true) + end + @test ForwardDiff.derivative(p -> benchmark_nlsolve_oop(quadratic_f, + @SVector[1.0, 1.0], p).u[end], p) ≈ 1 / (2 * sqrt(p)) + end + end + + @testset "[OOP] [Scalar AD]" begin + for p in 1.0:0.1:100.0 + @test begin + res = benchmark_nlsolve_oop(quadratic_f, 1.0, p) + res_true = sqrt(p) + res.u ≈ res_true + end + @test ForwardDiff.derivative(p -> benchmark_nlsolve_oop(quadratic_f, 1.0, p).u, + p) ≈ 1 / (2 * sqrt(p)) + end + end + + t = (p) -> [sqrt(p[2] / p[1])] + p = [0.9, 50.0] + @test benchmark_nlsolve_oop(quadratic_f2, 0.5, p).u ≈ sqrt(p[2] / p[1]) + @test ForwardDiff.jacobian(p -> [benchmark_nlsolve_oop(quadratic_f2, 0.5, p).u], + p) ≈ ForwardDiff.jacobian(t, p) + + # Iterator interface + function nlprob_iterator_interface(f, p_range, ::Val{iip}) where {iip} + probN = NonlinearProblem{iip}(f, iip ? [0.5] : 0.5, p_range[begin]) + cache = init(probN, GeneralBroyden(); maxiters = 100, abstol = 1e-10) + sols = zeros(length(p_range)) + for (i, p) in enumerate(p_range) + reinit!(cache, iip ? [cache.u[1]] : cache.u; p = p) + sol = solve!(cache) + sols[i] = iip ? sol.u[1] : sol.u + end + return sols + end + p = range(0.01, 2, length = 200) + @test nlprob_iterator_interface(quadratic_f, p, Val(false)) ≈ sqrt.(p) + @test nlprob_iterator_interface(quadratic_f!, p, Val(true)) ≈ sqrt.(p) +end