From 716209ff03a6de6d389b217a9215c58962b4d981 Mon Sep 17 00:00:00 2001 From: Christopher Rackauckas Date: Mon, 26 Feb 2024 04:53:22 -0600 Subject: [PATCH] Make default polyalgs respect autodiff --- Project.toml | 2 +- src/default.jl | 43 +++++++++++++++++++++----------------- test/misc/polyalg_tests.jl | 26 +++++++++++++++++++++++ 3 files changed, 51 insertions(+), 20 deletions(-) diff --git a/Project.toml b/Project.toml index a06689370..bc1919c29 100644 --- a/Project.toml +++ b/Project.toml @@ -1,7 +1,7 @@ name = "NonlinearSolve" uuid = "8913a72c-1f9b-4ce2-8d82-65094dcecaec" authors = ["SciML"] -version = "3.7.2" +version = "3.7.3" [deps] ADTypes = "47edcb42-4c32-4615-8424-f2b9edc5f35b" diff --git a/src/default.jl b/src/default.jl index 805001c0e..bae3e2c57 100644 --- a/src/default.jl +++ b/src/default.jl @@ -307,12 +307,13 @@ function FastShortcutNonlinearPolyalg( # and thus are not included in the polyalgorithm if SA if __is_complex(T) - algs = (SimpleBroyden(), Broyden(; init_jacobian = Val(:true_jacobian)), + algs = (SimpleBroyden(), + Broyden(; init_jacobian = Val(:true_jacobian), autodiff), SimpleKlement(), NewtonRaphson(; concrete_jac, linsolve, precs, autodiff)) else algs = (SimpleBroyden(), - Broyden(; init_jacobian = Val(:true_jacobian)), + Broyden(; init_jacobian = Val(:true_jacobian), autodiff), SimpleKlement(), NewtonRaphson(; concrete_jac, linsolve, precs, autodiff), NewtonRaphson(; concrete_jac, linsolve, precs, @@ -322,13 +323,13 @@ function FastShortcutNonlinearPolyalg( end else if __is_complex(T) - algs = (Broyden(), Broyden(; init_jacobian = Val(:true_jacobian)), - Klement(; linsolve, precs), + algs = (Broyden(), Broyden(; init_jacobian = Val(:true_jacobian), autodiff), + Klement(; linsolve, prec, autodiff), NewtonRaphson(; concrete_jac, linsolve, precs, autodiff)) else - algs = (Broyden(), - Broyden(; init_jacobian = Val(:true_jacobian)), - Klement(; linsolve, precs), + algs = (Broyden(; autodiff), + Broyden(; init_jacobian = Val(:true_jacobian), autodiff), + Klement(; linsolve, precs, autodiff), NewtonRaphson(; concrete_jac, linsolve, precs, autodiff), NewtonRaphson(; concrete_jac, linsolve, precs, linesearch = LineSearchesJL(; method = BackTracking()), autodiff), @@ -343,7 +344,7 @@ end """ FastShortcutNLLSPolyalg(::Type{T} = Float64; concrete_jac = nothing, linsolve = nothing, - precs = DEFAULT_PRECS, kwargs...) + precs = DEFAULT_PRECS, autodiff = nothing, kwargs...) A polyalgorithm focused on balancing speed and robustness. It first tries less robust methods for more performance and then tries more robust techniques if the faster ones fail. @@ -353,21 +354,25 @@ for more performance and then tries more robust techniques if the faster ones fa - `T`: The eltype of the initial guess. It is only used to check if some of the algorithms are compatible with the problem type. Defaults to `Float64`. """ -function FastShortcutNLLSPolyalg(::Type{T} = Float64; concrete_jac = nothing, - linsolve = nothing, precs = DEFAULT_PRECS, kwargs...) where {T} +function FastShortcutNLLSPolyalg( + ::Type{T} = Float64; concrete_jac = nothing, linsolve = nothing, + precs = DEFAULT_PRECS, autodiff = nothing, kwargs...) where {T} if __is_complex(T) - algs = (GaussNewton(; concrete_jac, linsolve, precs, kwargs...), - LevenbergMarquardt(; linsolve, precs, disable_geodesic = Val(true), kwargs...), - LevenbergMarquardt(; linsolve, precs, kwargs...)) + algs = (GaussNewton(; concrete_jac, linsolve, precs, autodiff, kwargs...), + LevenbergMarquardt(; + linsolve, precs, autodiff, disable_geodesic = Val(true), kwargs...), + LevenbergMarquardt(; linsolve, precs, autodiff, kwargs...)) else - algs = (GaussNewton(; concrete_jac, linsolve, precs, kwargs...), - LevenbergMarquardt(; linsolve, precs, disable_geodesic = Val(true), kwargs...), - TrustRegion(; concrete_jac, linsolve, precs, kwargs...), + algs = (GaussNewton(; concrete_jac, linsolve, precs, autodiff, kwargs...), + LevenbergMarquardt(; + linsolve, precs, disable_geodesic = Val(true), autodiff, kwargs...), + TrustRegion(; concrete_jac, linsolve, precs, autodiff, kwargs...), GaussNewton(; concrete_jac, linsolve, precs, - linesearch = LineSearchesJL(; method = BackTracking()), kwargs...), + linesearch = LineSearchesJL(; method = BackTracking()), + autodiff, kwargs...), TrustRegion(; concrete_jac, linsolve, precs, - radius_update_scheme = RadiusUpdateSchemes.Bastin, kwargs...), - LevenbergMarquardt(; linsolve, precs, kwargs...)) + radius_update_scheme = RadiusUpdateSchemes.Bastin, autodiff, kwargs...), + LevenbergMarquardt(; linsolve, precs, autodiff, kwargs...)) end return NonlinearSolvePolyAlgorithm(algs, Val(:NLLS)) end diff --git a/test/misc/polyalg_tests.jl b/test/misc/polyalg_tests.jl index 6c1f17639..7b7fe6880 100644 --- a/test/misc/polyalg_tests.jl +++ b/test/misc/polyalg_tests.jl @@ -71,6 +71,32 @@ end @test SciMLBase.successful_retcode(sol) end +@testitem "PolyAlgorithms Autodiff" begin + cache = zeros(2) + function f(du, u, p) + cache .= u .* u + du .= cache .- 2 + end + u0 = [1.0, 1.0] + probN = NonlinearProblem{true}(f, u0) + + custom_polyalg = NonlinearSolvePolyAlgorithm(( + Broyden(; autodiff = AutoFiniteDiff()), LimitedMemoryBroyden())) + + # Uses the `__solve` function + solver = solve(probN; abstol = 1e-9) + @test SciMLBase.successful_retcode(solver) + @test_throws MethodError solve(probN, RobustMultiNewton(); abstol = 1e-9) + @test SciMLBase.successful_retcode(solver) + solver = solve(probN, RobustMultiNewton(; autodiff = AutoFiniteDiff()); abstol = 1e-9) + @test SciMLBase.successful_retcode(solver) + solver = solve( + probN, FastShortcutNonlinearPolyalg(; autodiff = AutoFiniteDiff()); abstol = 1e-9) + @test SciMLBase.successful_retcode(solver) + solver = solve(probN, custom_polyalg; abstol = 1e-9) + @test SciMLBase.successful_retcode(solver) +end + @testitem "Simple Scalar Problem #187" begin using NaNMath