From 195ac47d161143a23b3e245ea22f08cc4a523c00 Mon Sep 17 00:00:00 2001 From: Marcelo Forets Date: Fri, 13 Oct 2023 21:10:31 -0300 Subject: [PATCH 1/7] add ExponentiationModule --- ...onentiation.jl => ExponentiationModule.jl} | 29 +++++++++++++++++-- src/Discretization/discretization.jl | 3 ++ src/ReachabilityAnalysis.jl | 3 +- test/discretization/discretization.jl | 22 +------------- test/discretization/exponentiation_module.jl | 26 +++++++++++++++++ 5 files changed, 58 insertions(+), 25 deletions(-) rename src/Discretization/{exponentiation.jl => ExponentiationModule.jl} (94%) create mode 100644 test/discretization/exponentiation_module.jl diff --git a/src/Discretization/exponentiation.jl b/src/Discretization/ExponentiationModule.jl similarity index 94% rename from src/Discretization/exponentiation.jl rename to src/Discretization/ExponentiationModule.jl index a78d8c8802..b1a4313d5e 100644 --- a/src/Discretization/exponentiation.jl +++ b/src/Discretization/ExponentiationModule.jl @@ -1,7 +1,28 @@ +""" +Interface to matrix exponential backends of different kinds. + +Includes common integral computations arising in the discretization of linear differential equations +using matrix methods. For applications see e.g. [1] and references therein. + +[1] Conservative Time Discretization: A Comparative Study. + Marcelo Forets and Christian Schilling (2022). + Proceedings of the 17th International Conference on integrated Formal Methods (iFM), + LNCS, vol. 13274, pp. 149-167. doi: 10.1007/978-3-031-07727-2_9, arXiv: 2111.01454. +""" +module ExponentiationModule + +using StaticArrays +using MathematicalSystems +using ReachabilityBase.Require using LinearAlgebra: checksquare +using SparseArrays: SparseMatrixCSC +using IntervalMatrices: AbstractIntervalMatrix +using LazySets + +export BaseExp, _exp, _alias, _elementwise_abs, _Φ₂ # ========================== -# Exponentiation functions +# Exponentiation interface # ========================== """ @@ -118,7 +139,7 @@ _alias(alg::Val{:pade}) = PadeExp end @inline function _exp(::SparseMatrixCSC, ::PadeExpAlg) - @requires Expokit + return require(@__MODULE__, Expokit) end function load_expokit_pade() @@ -440,7 +461,7 @@ function _Eplus(A::SparseMatrixCSC{N,D}, X0::AbstractHyperrectangle{N}, δt; m=m v = V.radius Aabs = copy(abs.(A)) - @requires ExponentialUtilities + require(@__MODULE__, ExponentialUtilities) Pv = _phiv(Aabs, v, 1, δt; m, tol) return E⁺ = Hyperrectangle(zeros(n), Pv) end @@ -489,3 +510,5 @@ function _Cδ(A, δ, order) E = IntervalMatrices._exp_remainder(A, δ, order; n=n) return M + E * δ end + +end # module diff --git a/src/Discretization/discretization.jl b/src/Discretization/discretization.jl index eecd34a5d9..d65114fe61 100644 --- a/src/Discretization/discretization.jl +++ b/src/Discretization/discretization.jl @@ -1,4 +1,7 @@ using IntervalMatrices: correction_hull, input_correction +using Reexport + +@reexport import ..ExponentiationModule: _alias # ================================== # Abstract interface diff --git a/src/ReachabilityAnalysis.jl b/src/ReachabilityAnalysis.jl index 01a5278f82..59c4ae034b 100644 --- a/src/ReachabilityAnalysis.jl +++ b/src/ReachabilityAnalysis.jl @@ -24,7 +24,8 @@ include("Continuous/fields.jl") include("Continuous/normalization.jl") include("Continuous/homogeneization.jl") include("Continuous/linearization.jl") -include("Discretization/exponentiation.jl") +include("Discretization/ExponentiationModule.jl") +using .ExponentiationModule include("Discretization/discretization.jl") # =========================================================== diff --git a/test/discretization/discretization.jl b/test/discretization/discretization.jl index 0d978205cd..a18cd0a780 100644 --- a/test/discretization/discretization.jl +++ b/test/discretization/discretization.jl @@ -2,24 +2,4 @@ using SparseArrays using Expokit using ReachabilityAnalysis: _exp, BaseExpAlg, LazyExpAlg, IntervalExpAlg, PadeExpAlg -@testset "Exponentiation" begin - A = [10.0 0; 0 20] - δ = 0.1 - expAδ = [exp(1) 0; 0 exp(2)] - - # default algorithm - @test _exp(A, δ) == expAδ - - # base algorithm - @test _exp(A, δ, BaseExpAlg()) == expAδ - - # Krylov algorithm - @test _exp(sparse(A), δ, LazyExpAlg()) == expAδ - - # interval matrix - @test all((_exp(A, δ, IntervalExpAlg()) .- IntervalMatrix(expAδ)) .<= 1e-4) - - # Padé approximation - @test_throws ArgumentError _exp(A, δ, PadeExpAlg()) - @test _exp(sparse(A), δ, PadeExpAlg()) ≈ expAδ -end +# TODO Add unit tests. diff --git a/test/discretization/exponentiation_module.jl b/test/discretization/exponentiation_module.jl new file mode 100644 index 0000000000..2a7104a052 --- /dev/null +++ b/test/discretization/exponentiation_module.jl @@ -0,0 +1,26 @@ +# using SparseArrays +# using Expokit +# using ReachabilityAnalysis: _exp, BaseExpAlg, LazyExpAlg, IntervalExpAlg, PadeExpAlg +using ReachabilityAnalysis.ExponentiationModule + +@testset "Exponentiation" begin + A = [10.0 0; 0 20] + δ = 0.1 + expAδ = [exp(1) 0; 0 exp(2)] + + # default algorithm + @test _exp(A, δ) == expAδ + + # base algorithm + @test _exp(A, δ, BaseExpAlg()) == expAδ + + # Krylov algorithm + @test _exp(sparse(A), δ, LazyExpAlg()) == expAδ + + # interval matrix + @test all((_exp(A, δ, IntervalExpAlg()) .- IntervalMatrix(expAδ)) .<= 1e-4) + + # Padé approximation + @test_throws ArgumentError _exp(A, δ, PadeExpAlg()) + @test _exp(sparse(A), δ, PadeExpAlg()) ≈ expAδ +end From 9ecb9d2eb50e1cfc7c07e1d4adcc8a64d9bbad9f Mon Sep 17 00:00:00 2001 From: Marcelo Forets Date: Sat, 14 Oct 2023 07:11:43 -0300 Subject: [PATCH 2/7] update module --- src/Discretization/ExponentiationModule.jl | 23 +++++++++++----------- 1 file changed, 11 insertions(+), 12 deletions(-) diff --git a/src/Discretization/ExponentiationModule.jl b/src/Discretization/ExponentiationModule.jl index b1a4313d5e..c761e88d75 100644 --- a/src/Discretization/ExponentiationModule.jl +++ b/src/Discretization/ExponentiationModule.jl @@ -14,16 +14,17 @@ module ExponentiationModule using StaticArrays using MathematicalSystems using ReachabilityBase.Require -using LinearAlgebra: checksquare -using SparseArrays: SparseMatrixCSC -using IntervalMatrices: AbstractIntervalMatrix +using LinearAlgebra: checksquare, I, Diagonal +using SparseArrays +using IntervalArithmetic +using IntervalMatrices using LazySets -export BaseExp, _exp, _alias, _elementwise_abs, _Φ₂ +export BaseExp, BaseExpAlg, IntervalExpAlg, LazyExpAlg, PadeExpAlg, _exp, _alias, _elementwise_abs, + _Φ₂, _Φ₁, _Φ₁_u, load_expokit_pade, _interval_matrix, _Cδ -# ========================== +# ------------------------- # Exponentiation interface -# ========================== """ AbstractExpAlg @@ -144,7 +145,7 @@ end function load_expokit_pade() return quote - @inline _exp(A::SparseMatrixCSC, ::PadeExpAlg) = Expokit.padm(A) + @inline ExponentiationModule._exp(A::SparseMatrixCSC, ::PadeExpAlg) = Expokit.padm(A) end end # quote / load_expokit_pade @@ -466,9 +467,8 @@ function _Eplus(A::SparseMatrixCSC{N,D}, X0::AbstractHyperrectangle{N}, δt; m=m return E⁺ = Hyperrectangle(zeros(n), Pv) end -# ================ +# --------------- # Absolute values -# ================ @inline _elementwise_abs(A::AbstractMatrix) = abs.(A) @inline function _elementwise_abs(A::SparseMatrixCSC) @@ -476,9 +476,8 @@ end end @inline _elementwise_abs(A::IdentityMultiple) = IdentityMultiple(abs(A.M.λ), size(A, 1)) -# ==================================== +# ------------------------------------ # Exponentiation of interval matrices -# ==================================== # TODO: outsource to IntervalMatrices.jl @@ -487,7 +486,7 @@ function _Cδ(A, δ, order) n = size(A, 1) A² = A * A if isa(A, IntervalMatrix) - Iδ = IntervalMatrix(Diagonal(fill(IA.interval(δ), n))) + Iδ = IntervalMatrix(Diagonal(fill(IntervalArithmetic.interval(δ), n))) else Iδ = Matrix(δ * I, n, n) end From ae9b25f3db9a31750346662eb21561a20f8a5041 Mon Sep 17 00:00:00 2001 From: Marcelo Forets Date: Sat, 14 Oct 2023 08:26:48 -0300 Subject: [PATCH 3/7] remove empty test file --- test/discretization/discretization.jl | 5 ----- 1 file changed, 5 deletions(-) delete mode 100644 test/discretization/discretization.jl diff --git a/test/discretization/discretization.jl b/test/discretization/discretization.jl deleted file mode 100644 index a18cd0a780..0000000000 --- a/test/discretization/discretization.jl +++ /dev/null @@ -1,5 +0,0 @@ -using SparseArrays -using Expokit -using ReachabilityAnalysis: _exp, BaseExpAlg, LazyExpAlg, IntervalExpAlg, PadeExpAlg - -# TODO Add unit tests. From 942ebc6944ebfe24e7a2f8cc88940e77aeef2bef Mon Sep 17 00:00:00 2001 From: Marcelo Forets Date: Sun, 15 Oct 2023 19:40:21 -0300 Subject: [PATCH 4/7] update tests --- docs/src/lib/discretize.md | 17 +++++++++-------- test/discretization/exponentiation_module.jl | 3 +-- test/runtests.jl | 2 +- 3 files changed, 11 insertions(+), 11 deletions(-) diff --git a/docs/src/lib/discretize.md b/docs/src/lib/discretize.md index 96cfcc4760..b352c08d04 100644 --- a/docs/src/lib/discretize.md +++ b/docs/src/lib/discretize.md @@ -44,14 +44,15 @@ over vectors when needed, that is, $e^{δA} v$ for each $v$. This method is part well-suited if `A` is vert sparse. Use the option `exp=:krylov` (or `exp=:lazy`) for this purpose. ```@docs -ReachabilityAnalysis._exp -ReachabilityAnalysis.Φ₁ -ReachabilityAnalysis.Φ₂ -ReachabilityAnalysis.AbstractExpAlg -ReachabilityAnalysis.BaseExpAlg -ReachabilityAnalysis.LazyExpAlg -ReachabilityAnalysis.IntervalExpAlg -ReachabilityAnalysis.PadeExpAlg +ReachabilityAnalysis.ExponentiationModule +ReachabilityAnalysis.ExponentiationModule._exp +ReachabilityAnalysis.ExponentiationModule.Φ₁ +ReachabilityAnalysis.ExponentiationModule.Φ₂ +ReachabilityAnalysis.ExponentiationModule.AbstractExpAlg +ReachabilityAnalysis.ExponentiationModule.BaseExpAlg +ReachabilityAnalysis.ExponentiationModule.LazyExpAlg +ReachabilityAnalysis.ExponentiationModule.IntervalExpAlg +ReachabilityAnalysis.ExponentiationModule.PadeExpAlg ``` ## References diff --git a/test/discretization/exponentiation_module.jl b/test/discretization/exponentiation_module.jl index 2a7104a052..ed85e38545 100644 --- a/test/discretization/exponentiation_module.jl +++ b/test/discretization/exponentiation_module.jl @@ -1,6 +1,5 @@ # using SparseArrays -# using Expokit -# using ReachabilityAnalysis: _exp, BaseExpAlg, LazyExpAlg, IntervalExpAlg, PadeExpAlg +using Expokit using ReachabilityAnalysis.ExponentiationModule @testset "Exponentiation" begin diff --git a/test/runtests.jl b/test/runtests.jl index f9431e6e16..ef7bb37ff7 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -17,7 +17,7 @@ TEST_MODELS = ["models/harmonic_oscillator.jl", TEST_FILES = ["continuous/solve.jl", "continuous/symbolics.jl", "continuous/traces.jl", - "discretization/discretization.jl", + "discretization/exponentiation_module.jl", "flowpipes/flowpipes.jl", "flowpipes/setops.jl", "reachsets/reachsets.jl", From 7fd992b604134213702e5003008fe4fb6efe4dc4 Mon Sep 17 00:00:00 2001 From: Marcelo Forets Date: Fri, 20 Oct 2023 11:42:02 -0300 Subject: [PATCH 5/7] review changes --- src/Discretization/Backward.jl | 2 + src/Discretization/CorrectionHull.jl | 34 +++++++- src/Discretization/ExponentiationModule.jl | 97 +++++++--------------- src/Discretization/FirstOrder.jl | 2 + src/Discretization/Forward.jl | 10 ++- src/Discretization/ForwardBackward.jl | 10 ++- src/Discretization/NoBloating.jl | 6 +- src/Discretization/SecondOrderddt.jl | 2 + src/Discretization/StepIntersect.jl | 2 + src/Initialization/init_Expokit.jl | 2 +- 10 files changed, 87 insertions(+), 80 deletions(-) diff --git a/src/Discretization/Backward.jl b/src/Discretization/Backward.jl index 0737358892..422f4e868c 100644 --- a/src/Discretization/Backward.jl +++ b/src/Discretization/Backward.jl @@ -2,6 +2,8 @@ # Backward approximation # ================================== +using ..ExponentiationModule: _alias + """ Backward{EM, SO, SI, IT, BT} <: AbstractApproximationModel diff --git a/src/Discretization/CorrectionHull.jl b/src/Discretization/CorrectionHull.jl index 2d631d4fb4..e7cacf3cf0 100644 --- a/src/Discretization/CorrectionHull.jl +++ b/src/Discretization/CorrectionHull.jl @@ -2,6 +2,8 @@ # Discretize using the correction hull of the matrix exponential # =============================================================== +using ..ExponentiationModule: _exp, _alias + """ CorrectionHull{EM} <: AbstractApproximationModel @@ -117,7 +119,7 @@ function discretize(ivp::IVP{<:CLCCS,<:LazySet}, δ, alg::CorrectionHull) Φ = _exp(A, δ, alg.exp) - A_interval = _interval_matrix(A) + A_interval = interval_matrix(A) origin_not_contained_in_U = zeros(dim(U)) ∉ Uz @@ -166,3 +168,33 @@ function _convert_intervals_to_box_with_vectors(intervals) H = Hyperrectangle(Vector(center(H)), Vector(radius_hyperrectangle(H))) return H end + +# TODO: outsource to IntervalMatrices.jl +# compute Iδ + 1/2 * δ^2 * A + 1/6 * δ^3 * A² + ... + 1/(η+1)! * δ^(η+1) * A^η + E(δ) * δ +function _Cδ(A, δ, order) + n = size(A, 1) + A² = A * A + if isa(A, IntervalMatrix) + Iδ = IntervalMatrix(Diagonal(fill(IntervalArithmetic.interval(δ), n))) + else + Iδ = Matrix(δ * I, n, n) + end + + IδW = Iδ + 1 / 2 * δ^2 * A + 1 / 6 * δ^3 * A² + M = IδW + + if order > 2 + # i = 2 + αᵢ₊₁ = 6 # factorial of (i+1) + Aⁱ = A² + δⁱ⁺¹ = δ^3 + @inbounds for i in 3:order + αᵢ₊₁ *= i + 1 + δⁱ⁺¹ *= δ + Aⁱ *= A + M += (δⁱ⁺¹ / αᵢ₊₁) * Aⁱ + end + end + E = IntervalMatrices._exp_remainder(A, δ, order; n=n) + return M + E * δ +end diff --git a/src/Discretization/ExponentiationModule.jl b/src/Discretization/ExponentiationModule.jl index c761e88d75..3f7ccf9262 100644 --- a/src/Discretization/ExponentiationModule.jl +++ b/src/Discretization/ExponentiationModule.jl @@ -20,8 +20,8 @@ using IntervalArithmetic using IntervalMatrices using LazySets -export BaseExp, BaseExpAlg, IntervalExpAlg, LazyExpAlg, PadeExpAlg, _exp, _alias, _elementwise_abs, - _Φ₂, _Φ₁, _Φ₁_u, load_expokit_pade, _interval_matrix, _Cδ +export BaseExp, BaseExpAlg, IntervalExpAlg, LazyExpAlg, PadeExpAlg, elementwise_abs, Φ₂, Φ₁, Φ₁_u, + interval_matrix # ------------------------- # Exponentiation interface @@ -122,26 +122,21 @@ const PadeExp = PadeExpAlg() _alias(alg::Val{:pade}) = PadeExp # general case: convert to Matrix -@inline _exp(A::AbstractMatrix, ::BaseExpAlg) = exp(Matrix(A)) -@inline _exp(A::Matrix, ::BaseExpAlg) = exp(A) +_exp(A::AbstractMatrix, ::BaseExpAlg) = exp(Matrix(A)) +_exp(A::Matrix, ::BaseExpAlg) = exp(A) # static arrays have their own exp method -@inline _exp(A::StaticArray, ::BaseExpAlg) = exp(A) +_exp(A::StaticArray, ::BaseExpAlg) = exp(A) # exponential of an identity multiple, defined in MathematicalSystems.jl -@inline _exp(A::IdentityMultiple, ::BaseExpAlg) = exp(A) +_exp(A::IdentityMultiple, ::BaseExpAlg) = exp(A) # lazy wrapper of the matrix exponential -@inline _exp(A::AbstractMatrix, ::LazyExpAlg) = SparseMatrixExp(A) +_exp(A::AbstractMatrix, ::LazyExpAlg) = SparseMatrixExp(A) # pade approximants (requires Expokit.jl) -@inline function _exp(::AbstractMatrix, ::PadeExpAlg) - throw(ArgumentError("algorithm requires a sparse matrix")) -end - -@inline function _exp(::SparseMatrixCSC, ::PadeExpAlg) - return require(@__MODULE__, Expokit) -end +_exp(::AbstractMatrix, ::PadeExpAlg) = throw(ArgumentError("algorithm requires a sparse matrix")) +_exp(::SparseMatrixCSC, ::PadeExpAlg) = require(@__MODULE__, Expokit) function load_expokit_pade() return quote @@ -150,18 +145,18 @@ function load_expokit_pade() end # quote / load_expokit_pade function _exp(A::AbstractMatrix, alg::IntervalExpAlg) - return exp_overapproximation(_interval_matrix(A), one(eltype(A)), alg.order) + return exp_overapproximation(interval_matrix(A), one(eltype(A)), alg.order) end function _exp(A::AbstractMatrix, δ, alg::IntervalExpAlg) - return exp_overapproximation(_interval_matrix(A), δ, alg.order) + return exp_overapproximation(interval_matrix(A), δ, alg.order) end -function _interval_matrix(A::AbstractIntervalMatrix) +# TODO Refactor to IntervalMatrices.jl? +function interval_matrix(A::AbstractIntervalMatrix) return A end - -function _interval_matrix(A::AbstractMatrix) +function interval_matrix(A::AbstractMatrix) return IntervalMatrix(A) end @@ -283,13 +278,13 @@ It can be shown that where ``Φ(A, δ) = e^{Aδ}``. In particular, `Φ₁(A, δ) = P[1:n, (n+1):2*n]`. This method can be found in [[FRE11]](@ref). """ -function Φ₁(A::AbstractMatrix, δ, alg::AbstractExpAlg=BaseExp, isinv=false, Φ=nothing) - return _Φ₁(A, δ, alg, Val(isinv), Φ) +function Φ₁(A::AbstractMatrix, δ::Real, alg::AbstractExpAlg=BaseExp, isinv=false, Φ=nothing) + return Φ₁(A, δ, alg, Val(isinv), Φ) end # dispatch for discretization methods -_Φ₁(A, δ, alg, isinv::Val{:false}, Φ) = _Φ₁_blk(A, δ, alg) -_Φ₁(A, δ, alg, isinv::Val{:true}, Φ) = _Φ₁_inv(A, δ, alg, Φ) +Φ₁(A, δ, alg, isinv::Val{:false}, Φ) = _Φ₁_blk(A, δ, alg) +Φ₁(A, δ, alg, isinv::Val{:true}, Φ) = _Φ₁_inv(A, δ, alg, Φ) # evaluate the series Φ₁(A, δ) = ∑_{i=0}^∞ \\dfrac{δ^{i+1}}{(i+1)!}A^i # without assuming invertibility of A, by taking the exponential of a 2n x 2n matrix @@ -325,11 +320,11 @@ function _Φ₁_inv(A::IdentityMultiple, δ, alg, Φ=nothing) end # method to compute Φ₁ * u, where u is a vector -function _Φ₁_u(A, δ, alg, isinv::Val{:true}, u::AbstractVector, Φ=nothing) - return _Φ₁_u(A, δ, alg, Φ, u) +function Φ₁_u(A, δ, alg, isinv::Val{:true}, u::AbstractVector, Φ=nothing) + return Φ₁_u(A, δ, alg, Φ, u) end -function _Φ₁_u(A, δ, alg, isinv::Val{:false}, u::AbstractVector, Φ=nothing) +function Φ₁_u(A, δ, alg, isinv::Val{:false}, u::AbstractVector, Φ=nothing) M = _Φ₁_blk(A, δ, alg) return M * u end @@ -337,7 +332,7 @@ end # compute the vector Φ₁(A, δ)u = A^{-1}(exp(Aδ) - I) u # assuming that A is invertible and solving the associated linear system # (the inverse of A is not explicitly computed) -function _Φ₁_u(A, δ, alg, u::AbstractVector, Φ=nothing) +function Φ₁_u(A, δ, alg, u::AbstractVector, Φ=nothing) if isnothing(Φ) Φ = _exp(A, δ, alg) end @@ -347,7 +342,7 @@ end # compute Φ₁(A, δ)u = A^{-1}(exp(Aδ) - I) u without explicitly computing exp(Aδ) # and assuming that A is invertible -function _Φ₁_u(A, δ, alg::LazyExpAlg, u::AbstractVector, ::Nothing) +function Φ₁_u(A, δ, alg::LazyExpAlg, u::AbstractVector, ::Nothing) w = LazySets._expmv(δ, A, u; m=alg.m, tol=alg.tol) x = w - u return A \ x @@ -408,12 +403,12 @@ where ``Φ(A, δ) = e^{Aδ}``. In particular, `Φ₂ = P_{3n}[1:n, (2*n+1):3*n]` This method can be found in [[FRE11]](@ref). """ function Φ₂(A::AbstractMatrix, δ, alg::AbstractExpAlg=BaseExp, isinv=false, Φ=nothing) - return _Φ₂(A, δ, alg, Val(isinv), Φ) + return Φ₂(A, δ, alg, Val(isinv), Φ) end # dispatch for discretization methods -_Φ₂(A, δ, alg, isinv::Val{:false}, Φ) = _Φ₂_blk(A, δ, alg) -_Φ₂(A, δ, alg, isinv::Val{:true}, Φ) = _Φ₂_inv(A, δ, alg, Φ) +Φ₂(A, δ, alg, isinv::Val{:false}, Φ) = _Φ₂_blk(A, δ, alg) +Φ₂(A, δ, alg, isinv::Val{:true}, Φ) = _Φ₂_inv(A, δ, alg, Φ) @inline _P₂_blk(P, n, ::AbstractExpAlg) = P[1:n, (2 * n + 1):(3 * n)] @inline _P₂_blk(P, n, ::LazyExpAlg) = sparse(get_columns(P, (2 * n + 1):(3 * n))[1:n, :]) @@ -470,44 +465,10 @@ end # --------------- # Absolute values -@inline _elementwise_abs(A::AbstractMatrix) = abs.(A) -@inline function _elementwise_abs(A::SparseMatrixCSC) +elementwise_abs(A::AbstractMatrix) = abs.(A) +function elementwise_abs(A::SparseMatrixCSC) return SparseMatrixCSC(A.m, A.n, A.colptr, A.rowval, abs.(nonzeros(A))) end -@inline _elementwise_abs(A::IdentityMultiple) = IdentityMultiple(abs(A.M.λ), size(A, 1)) - -# ------------------------------------ -# Exponentiation of interval matrices - -# TODO: outsource to IntervalMatrices.jl - -# compute Iδ + 1/2 * δ^2 * A + 1/6 * δ^3 * A² + ... + 1/(η+1)! * δ^(η+1) * A^η + E(δ) * δ -function _Cδ(A, δ, order) - n = size(A, 1) - A² = A * A - if isa(A, IntervalMatrix) - Iδ = IntervalMatrix(Diagonal(fill(IntervalArithmetic.interval(δ), n))) - else - Iδ = Matrix(δ * I, n, n) - end - - IδW = Iδ + 1 / 2 * δ^2 * A + 1 / 6 * δ^3 * A² - M = IδW - - if order > 2 - # i = 2 - αᵢ₊₁ = 6 # factorial of (i+1) - Aⁱ = A² - δⁱ⁺¹ = δ^3 - @inbounds for i in 3:order - αᵢ₊₁ *= i + 1 - δⁱ⁺¹ *= δ - Aⁱ *= A - M += (δⁱ⁺¹ / αᵢ₊₁) * Aⁱ - end - end - E = IntervalMatrices._exp_remainder(A, δ, order; n=n) - return M + E * δ -end +elementwise_abs(A::IdentityMultiple) = IdentityMultiple(abs(A.M.λ), size(A, 1)) end # module diff --git a/src/Discretization/FirstOrder.jl b/src/Discretization/FirstOrder.jl index 0ab949e5cf..9eeb9c5416 100644 --- a/src/Discretization/FirstOrder.jl +++ b/src/Discretization/FirstOrder.jl @@ -2,6 +2,8 @@ # First-order approximation model # =============================== +using ..ExponentiationModule: _exp + """ FirstOrder{EM} <: AbstractApproximationModel diff --git a/src/Discretization/Forward.jl b/src/Discretization/Forward.jl index 4dfb736fe4..cdabe671df 100644 --- a/src/Discretization/Forward.jl +++ b/src/Discretization/Forward.jl @@ -2,6 +2,8 @@ # Forward approximation # ================================== +using ..ExponentiationModule: _exp, _alias + """ Forward{EM, SO, SI, IT, BT} <: AbstractApproximationModel @@ -66,9 +68,9 @@ function discretize(ivp::IVP{<:CLCS,<:LazySet}, δ, alg::Forward) X0 = initial_state(ivp) Φ = _exp(A, δ, alg.exp) - A_abs = _elementwise_abs(A) + A_abs = elementwise_abs(A) Φcache = sum(A) == abs(sum(A)) ? Φ : nothing - P2A_abs = _Φ₂(A_abs, δ, alg.exp, alg.inv, Φcache) + P2A_abs = Φ₂(A_abs, δ, alg.exp, alg.inv, Φcache) E₊ = sih(P2A_abs * sih((A * A) * X0, alg.sih), alg.sih) Ω0 = ConvexHull(X0, Φ * X0 ⊕ E₊) Ω0 = _apply_setops(Ω0, alg) @@ -114,9 +116,9 @@ function discretize(ivp::IVP{<:CLCCS,<:LazySet}, δ, alg::Forward) X0 = initial_state(ivp) Φ = _exp(A, δ, alg.exp) - A_abs = _elementwise_abs(A) + A_abs = elementwise_abs(A) Φcache = sum(A) == abs(sum(A)) ? Φ : nothing - P2A_abs = _Φ₂(A_abs, δ, alg.exp, alg.inv, Φcache) + P2A_abs = Φ₂(A_abs, δ, alg.exp, alg.inv, Φcache) Einit = sih(P2A_abs * sih((A * A) * X0, alg.sih), alg.sih) diff --git a/src/Discretization/ForwardBackward.jl b/src/Discretization/ForwardBackward.jl index a56c538ce0..f8291edb9a 100644 --- a/src/Discretization/ForwardBackward.jl +++ b/src/Discretization/ForwardBackward.jl @@ -1,3 +1,5 @@ +using ..ExponentiationModule: _exp, _alias + # obs: S should normally be <:JuMP.MOI.AbstractOptimizer struct ForwardBackward{EM,SO,SI,IT,BT,S} <: AbstractApproximationModel exp::EM @@ -40,9 +42,9 @@ function discretize(ivp::IVP{<:CLCS,<:LazySet}, δ, alg::ForwardBackward) X0 = initial_state(ivp) Φ = _exp(A, δ, alg.exp) - A_abs = _elementwise_abs(A) + A_abs = elementwise_abs(A) Φcache = A == A_abs ? Φ : nothing - P2A_abs = _Φ₂(A_abs, δ, alg.exp, alg.inv, Φcache) + P2A_abs = Φ₂(A_abs, δ, alg.exp, alg.inv, Φcache) A² = A * A E₊ = sih(P2A_abs * sih(A² * X0, alg.sih), alg.sih) @@ -72,9 +74,9 @@ function discretize(ivp::IVP{<:CLCCS,<:LazySet}, δ, alg::ForwardBackward) Φ = _exp(A, δ, alg.exp) U = next_set(inputset(ivp), 1) - A_abs = _elementwise_abs(A) + A_abs = elementwise_abs(A) Φcache = A == A_abs ? Φ : nothing - P2A_abs = _Φ₂(A_abs, δ, alg.exp, alg.inv, Φcache) + P2A_abs = Φ₂(A_abs, δ, alg.exp, alg.inv, Φcache) A² = A * A E₊ = sih(P2A_abs * sih(A² * X0, alg.sih), alg.sih) diff --git a/src/Discretization/NoBloating.jl b/src/Discretization/NoBloating.jl index 5a588f2010..72c7433c13 100644 --- a/src/Discretization/NoBloating.jl +++ b/src/Discretization/NoBloating.jl @@ -2,6 +2,8 @@ # Approximation model in discrete time, i.e. without bloating # ============================================================ +using ..ExponentiationModule: _exp, _alias + """ NoBloating{EM, SO, IT} <: AbstractApproximationModel @@ -68,10 +70,10 @@ function discretize(ivp::IVP{<:CLCCS,<:LazySet}, δ, alg::NoBloating) Φ = _exp(A, δ, alg.exp) if isa(U, AbstractSingleton) - Mu = _Φ₁_u(A, δ, alg.exp, alg.inv, element(U), Φ) + Mu = Φ₁_u(A, δ, alg.exp, alg.inv, element(U), Φ) V = Singleton(Mu) else - M = _Φ₁(A, δ, alg.exp, alg.inv, Φ) + M = Φ₁(A, δ, alg.exp, alg.inv, Φ) V = _apply_setops(M * U, alg.setops) end diff --git a/src/Discretization/SecondOrderddt.jl b/src/Discretization/SecondOrderddt.jl index 2df7308388..1aa913a20d 100644 --- a/src/Discretization/SecondOrderddt.jl +++ b/src/Discretization/SecondOrderddt.jl @@ -2,6 +2,8 @@ # Second-order approximation from d/dt # =================================== +using ..ExponentiationModule: _exp, _alias + """ SecondOrderddt{EM, SO, SI, IT, BT} <: AbstractApproximationModel diff --git a/src/Discretization/StepIntersect.jl b/src/Discretization/StepIntersect.jl index e81b845590..40719443db 100644 --- a/src/Discretization/StepIntersect.jl +++ b/src/Discretization/StepIntersect.jl @@ -3,6 +3,8 @@ # backward of the same model # ============================================================================ +using ..ExponentiationModule: _exp, _alias + """ StepIntersect{DM<:AbstractApproximationModel} <: AbstractApproximationModel diff --git a/src/Initialization/init_Expokit.jl b/src/Initialization/init_Expokit.jl index 5129b9d075..b243c39bf6 100644 --- a/src/Initialization/init_Expokit.jl +++ b/src/Initialization/init_Expokit.jl @@ -1 +1 @@ -eval(load_expokit_pade()) +eval(ExponentiationModule.load_expokit_pade()) From 080997be10a98262a408e7d443d4bd0ce7abf168 Mon Sep 17 00:00:00 2001 From: Marcelo Forets Date: Fri, 20 Oct 2023 12:09:36 -0300 Subject: [PATCH 6/7] update --- src/Discretization/Backward.jl | 2 +- src/Discretization/CorrectionHull.jl | 2 +- .../{ExponentiationModule.jl => Exponentiation.jl} | 14 +++++++------- src/Discretization/FirstOrder.jl | 2 +- src/Discretization/Forward.jl | 2 +- src/Discretization/ForwardBackward.jl | 2 +- src/Discretization/NoBloating.jl | 2 +- src/Discretization/SecondOrderddt.jl | 2 +- src/Discretization/StepIntersect.jl | 2 +- src/Discretization/discretization.jl | 3 ++- src/Initialization/init_Expokit.jl | 2 +- src/ReachabilityAnalysis.jl | 3 +-- ...{exponentiation_module.jl => exponentiation.jl} | 3 ++- test/runtests.jl | 2 +- 14 files changed, 22 insertions(+), 21 deletions(-) rename src/Discretization/{ExponentiationModule.jl => Exponentiation.jl} (95%) rename test/discretization/{exponentiation_module.jl => exponentiation.jl} (86%) diff --git a/src/Discretization/Backward.jl b/src/Discretization/Backward.jl index 422f4e868c..2852937abc 100644 --- a/src/Discretization/Backward.jl +++ b/src/Discretization/Backward.jl @@ -2,7 +2,7 @@ # Backward approximation # ================================== -using ..ExponentiationModule: _alias +using ..Exponentiation: _alias """ Backward{EM, SO, SI, IT, BT} <: AbstractApproximationModel diff --git a/src/Discretization/CorrectionHull.jl b/src/Discretization/CorrectionHull.jl index e7cacf3cf0..713bde845d 100644 --- a/src/Discretization/CorrectionHull.jl +++ b/src/Discretization/CorrectionHull.jl @@ -2,7 +2,7 @@ # Discretize using the correction hull of the matrix exponential # =============================================================== -using ..ExponentiationModule: _exp, _alias +using ..Exponentiation: _exp, _alias """ CorrectionHull{EM} <: AbstractApproximationModel diff --git a/src/Discretization/ExponentiationModule.jl b/src/Discretization/Exponentiation.jl similarity index 95% rename from src/Discretization/ExponentiationModule.jl rename to src/Discretization/Exponentiation.jl index 3f7ccf9262..19536f0e03 100644 --- a/src/Discretization/ExponentiationModule.jl +++ b/src/Discretization/Exponentiation.jl @@ -9,7 +9,7 @@ using matrix methods. For applications see e.g. [1] and references therein. Proceedings of the 17th International Conference on integrated Formal Methods (iFM), LNCS, vol. 13274, pp. 149-167. doi: 10.1007/978-3-031-07727-2_9, arXiv: 2111.01454. """ -module ExponentiationModule +module Exponentiation using StaticArrays using MathematicalSystems @@ -140,7 +140,7 @@ _exp(::SparseMatrixCSC, ::PadeExpAlg) = require(@__MODULE__, Expokit) function load_expokit_pade() return quote - @inline ExponentiationModule._exp(A::SparseMatrixCSC, ::PadeExpAlg) = Expokit.padm(A) + @inline Exponentiation._exp(A::SparseMatrixCSC, ::PadeExpAlg) = Expokit.padm(A) end end # quote / load_expokit_pade @@ -283,8 +283,8 @@ function Φ₁(A::AbstractMatrix, δ::Real, alg::AbstractExpAlg=BaseExp, isinv=f end # dispatch for discretization methods -Φ₁(A, δ, alg, isinv::Val{:false}, Φ) = _Φ₁_blk(A, δ, alg) -Φ₁(A, δ, alg, isinv::Val{:true}, Φ) = _Φ₁_inv(A, δ, alg, Φ) +Φ₁(A::AbstractMatrix, δ::Real, alg::AbstractExpAlg, isinv::Val{:false}, Φ) = _Φ₁_blk(A, δ, alg) +Φ₁(A::AbstractMatrix, δ::Real, alg::AbstractExpAlg, isinv::Val{:true}, Φ) = _Φ₁_inv(A, δ, alg, Φ) # evaluate the series Φ₁(A, δ) = ∑_{i=0}^∞ \\dfrac{δ^{i+1}}{(i+1)!}A^i # without assuming invertibility of A, by taking the exponential of a 2n x 2n matrix @@ -402,13 +402,13 @@ It can be shown that where ``Φ(A, δ) = e^{Aδ}``. In particular, `Φ₂ = P_{3n}[1:n, (2*n+1):3*n]`. This method can be found in [[FRE11]](@ref). """ -function Φ₂(A::AbstractMatrix, δ, alg::AbstractExpAlg=BaseExp, isinv=false, Φ=nothing) +function Φ₂(A::AbstractMatrix, δ::Real, alg::AbstractExpAlg=BaseExp, isinv=false, Φ=nothing) return Φ₂(A, δ, alg, Val(isinv), Φ) end # dispatch for discretization methods -Φ₂(A, δ, alg, isinv::Val{:false}, Φ) = _Φ₂_blk(A, δ, alg) -Φ₂(A, δ, alg, isinv::Val{:true}, Φ) = _Φ₂_inv(A, δ, alg, Φ) +Φ₂(A::AbstractMatrix, δ::Real, alg::AbstractExpAlg, isinv::Val{:false}, Φ) = _Φ₂_blk(A, δ, alg) +Φ₂(A::AbstractMatrix, δ::Real, alg::AbstractExpAlg, isinv::Val{:true}, Φ) = _Φ₂_inv(A, δ, alg, Φ) @inline _P₂_blk(P, n, ::AbstractExpAlg) = P[1:n, (2 * n + 1):(3 * n)] @inline _P₂_blk(P, n, ::LazyExpAlg) = sparse(get_columns(P, (2 * n + 1):(3 * n))[1:n, :]) diff --git a/src/Discretization/FirstOrder.jl b/src/Discretization/FirstOrder.jl index 9eeb9c5416..c42c8dc235 100644 --- a/src/Discretization/FirstOrder.jl +++ b/src/Discretization/FirstOrder.jl @@ -2,7 +2,7 @@ # First-order approximation model # =============================== -using ..ExponentiationModule: _exp +using ..Exponentiation: _exp """ FirstOrder{EM} <: AbstractApproximationModel diff --git a/src/Discretization/Forward.jl b/src/Discretization/Forward.jl index cdabe671df..bdbe37a0d3 100644 --- a/src/Discretization/Forward.jl +++ b/src/Discretization/Forward.jl @@ -2,7 +2,7 @@ # Forward approximation # ================================== -using ..ExponentiationModule: _exp, _alias +using ..Exponentiation: _exp, _alias """ Forward{EM, SO, SI, IT, BT} <: AbstractApproximationModel diff --git a/src/Discretization/ForwardBackward.jl b/src/Discretization/ForwardBackward.jl index f8291edb9a..cffadfd740 100644 --- a/src/Discretization/ForwardBackward.jl +++ b/src/Discretization/ForwardBackward.jl @@ -1,4 +1,4 @@ -using ..ExponentiationModule: _exp, _alias +using ..Exponentiation: _exp, _alias # obs: S should normally be <:JuMP.MOI.AbstractOptimizer struct ForwardBackward{EM,SO,SI,IT,BT,S} <: AbstractApproximationModel diff --git a/src/Discretization/NoBloating.jl b/src/Discretization/NoBloating.jl index 72c7433c13..ef4667ebe9 100644 --- a/src/Discretization/NoBloating.jl +++ b/src/Discretization/NoBloating.jl @@ -2,7 +2,7 @@ # Approximation model in discrete time, i.e. without bloating # ============================================================ -using ..ExponentiationModule: _exp, _alias +using ..Exponentiation: _exp, _alias """ NoBloating{EM, SO, IT} <: AbstractApproximationModel diff --git a/src/Discretization/SecondOrderddt.jl b/src/Discretization/SecondOrderddt.jl index 1aa913a20d..a65755625f 100644 --- a/src/Discretization/SecondOrderddt.jl +++ b/src/Discretization/SecondOrderddt.jl @@ -2,7 +2,7 @@ # Second-order approximation from d/dt # =================================== -using ..ExponentiationModule: _exp, _alias +using ..Exponentiation: _exp, _alias """ SecondOrderddt{EM, SO, SI, IT, BT} <: AbstractApproximationModel diff --git a/src/Discretization/StepIntersect.jl b/src/Discretization/StepIntersect.jl index 40719443db..e09ab31bd2 100644 --- a/src/Discretization/StepIntersect.jl +++ b/src/Discretization/StepIntersect.jl @@ -3,7 +3,7 @@ # backward of the same model # ============================================================================ -using ..ExponentiationModule: _exp, _alias +using ..Exponentiation: _exp, _alias """ StepIntersect{DM<:AbstractApproximationModel} <: AbstractApproximationModel diff --git a/src/Discretization/discretization.jl b/src/Discretization/discretization.jl index d65114fe61..5ffeba45ad 100644 --- a/src/Discretization/discretization.jl +++ b/src/Discretization/discretization.jl @@ -1,7 +1,8 @@ using IntervalMatrices: correction_hull, input_correction using Reexport -@reexport import ..ExponentiationModule: _alias +using .Exponentiation +import ..Exponentiation: _alias # ================================== # Abstract interface diff --git a/src/Initialization/init_Expokit.jl b/src/Initialization/init_Expokit.jl index b243c39bf6..66cac6cd81 100644 --- a/src/Initialization/init_Expokit.jl +++ b/src/Initialization/init_Expokit.jl @@ -1 +1 @@ -eval(ExponentiationModule.load_expokit_pade()) +eval(Exponentiation.load_expokit_pade()) diff --git a/src/ReachabilityAnalysis.jl b/src/ReachabilityAnalysis.jl index 59c4ae034b..5045ee5243 100644 --- a/src/ReachabilityAnalysis.jl +++ b/src/ReachabilityAnalysis.jl @@ -24,8 +24,7 @@ include("Continuous/fields.jl") include("Continuous/normalization.jl") include("Continuous/homogeneization.jl") include("Continuous/linearization.jl") -include("Discretization/ExponentiationModule.jl") -using .ExponentiationModule +include("Discretization/Exponentiation.jl") include("Discretization/discretization.jl") # =========================================================== diff --git a/test/discretization/exponentiation_module.jl b/test/discretization/exponentiation.jl similarity index 86% rename from test/discretization/exponentiation_module.jl rename to test/discretization/exponentiation.jl index ed85e38545..1834422496 100644 --- a/test/discretization/exponentiation_module.jl +++ b/test/discretization/exponentiation.jl @@ -1,6 +1,7 @@ # using SparseArrays using Expokit -using ReachabilityAnalysis.ExponentiationModule +using ReachabilityAnalysis.Exponentiation +using ReachabilityAnalysis.Exponentiation: _exp @testset "Exponentiation" begin A = [10.0 0; 0 20] diff --git a/test/runtests.jl b/test/runtests.jl index ef7bb37ff7..b0f66419fa 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -17,7 +17,7 @@ TEST_MODELS = ["models/harmonic_oscillator.jl", TEST_FILES = ["continuous/solve.jl", "continuous/symbolics.jl", "continuous/traces.jl", - "discretization/exponentiation_module.jl", + "discretization/exponentiation.jl", "flowpipes/flowpipes.jl", "flowpipes/setops.jl", "reachsets/reachsets.jl", From 8187b5ca195225d96b23afaa36481d3772f9404a Mon Sep 17 00:00:00 2001 From: schillic Date: Fri, 20 Oct 2023 18:25:34 +0200 Subject: [PATCH 7/7] fix missing renaming --- docs/src/lib/discretize.md | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/docs/src/lib/discretize.md b/docs/src/lib/discretize.md index b352c08d04..e70fe406a3 100644 --- a/docs/src/lib/discretize.md +++ b/docs/src/lib/discretize.md @@ -44,15 +44,15 @@ over vectors when needed, that is, $e^{δA} v$ for each $v$. This method is part well-suited if `A` is vert sparse. Use the option `exp=:krylov` (or `exp=:lazy`) for this purpose. ```@docs -ReachabilityAnalysis.ExponentiationModule -ReachabilityAnalysis.ExponentiationModule._exp -ReachabilityAnalysis.ExponentiationModule.Φ₁ -ReachabilityAnalysis.ExponentiationModule.Φ₂ -ReachabilityAnalysis.ExponentiationModule.AbstractExpAlg -ReachabilityAnalysis.ExponentiationModule.BaseExpAlg -ReachabilityAnalysis.ExponentiationModule.LazyExpAlg -ReachabilityAnalysis.ExponentiationModule.IntervalExpAlg -ReachabilityAnalysis.ExponentiationModule.PadeExpAlg +ReachabilityAnalysis.Exponentiation +ReachabilityAnalysis.Exponentiation._exp +ReachabilityAnalysis.Exponentiation.Φ₁ +ReachabilityAnalysis.Exponentiation.Φ₂ +ReachabilityAnalysis.Exponentiation.AbstractExpAlg +ReachabilityAnalysis.Exponentiation.BaseExpAlg +ReachabilityAnalysis.Exponentiation.LazyExpAlg +ReachabilityAnalysis.Exponentiation.IntervalExpAlg +ReachabilityAnalysis.Exponentiation.PadeExpAlg ``` ## References