Skip to content

Commit 01e48a4

Browse files
authored
Merge pull request #165 from JuliaControl/new_jacbuffer
added: `linearize!` is allocation-free once again
2 parents 2e7ab0e + 896a35d commit 01e48a4

7 files changed

+125
-31
lines changed

Project.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
name = "ModelPredictiveControl"
22
uuid = "61f9bdb8-6ae4-484a-811f-bbf86720c31c"
33
authors = ["Francis Gagnon"]
4-
version = "1.3.4"
4+
version = "1.3.5"
55

66
[deps]
77
ControlSystemsBase = "aaaaaaaa-a6ca-5380-bf3e-84a91bcd477e"

src/general.jl

+23
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,29 @@ const DEFAULT_LWT = 0.0
66
const DEFAULT_CWT = 1e5
77
const DEFAULT_EWT = 0.0
88

9+
"Abstract type for all differentiation buffers."
10+
abstract type DifferentiationBuffer end
11+
12+
"Struct with both function and configuration for ForwardDiff differentiation."
13+
struct JacobianBuffer{FT<:Function, CT<:ForwardDiff.JacobianConfig} <: DifferentiationBuffer
14+
f!::FT
15+
config::CT
16+
end
17+
18+
function Base.show(io::IO, buffer::DifferentiationBuffer)
19+
return print(io, "DifferentiationBuffer with a $(typeof(buffer.config).name.name)")
20+
end
21+
22+
"Create a JacobianBuffer with function `f!`, output `y` and input `x`."
23+
JacobianBuffer(f!, y, x) = JacobianBuffer(f!, ForwardDiff.JacobianConfig(f!, y, x))
24+
25+
"Compute in-place and return the Jacobian matrix of `buffer.f!` at `x`."
26+
function jacobian!(
27+
A, buffer::JacobianBuffer, y, x
28+
)
29+
return ForwardDiff.jacobian!(A, buffer.f!, y, x, buffer.config)
30+
end
31+
932
"Termination status that means 'no solution available'."
1033
const ERROR_STATUSES = (
1134
JuMP.INFEASIBLE, JuMP.DUAL_INFEASIBLE, JuMP.LOCALLY_INFEASIBLE,

src/model/linearization.jl

+78-20
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,71 @@
1-
"""
2-
LinModel(model::NonLinModel; x=model.x0+model.xop, u=model.uop, d=model.dop)
1+
"A linearization buffer for the [`linearize`](@ref) function."
2+
struct LinearizationBuffer{
3+
NT <:Real,
4+
JB_FUD <:JacobianBuffer,
5+
JB_FXD <:JacobianBuffer,
6+
JB_FXU <:JacobianBuffer,
7+
JB_HD <:JacobianBuffer,
8+
JB_HX <:JacobianBuffer
9+
}
10+
x::Vector{NT}
11+
u::Vector{NT}
12+
d::Vector{NT}
13+
buffer_f_at_u_d::JB_FUD
14+
buffer_f_at_x_d::JB_FXD
15+
buffer_f_at_x_u::JB_FXU
16+
buffer_h_at_d ::JB_HD
17+
buffer_h_at_x ::JB_HX
18+
function LinearizationBuffer(
19+
x::Vector{NT},
20+
u::Vector{NT},
21+
d::Vector{NT},
22+
buffer_f_at_u_d::JB_FUD,
23+
buffer_f_at_x_d::JB_FXD,
24+
buffer_f_at_x_u::JB_FXU,
25+
buffer_h_at_d::JB_HD,
26+
buffer_h_at_x::JB_HX
27+
) where {NT<:Real, JB_FUD, JB_FXD, JB_FXU, JB_HD, JB_HX}
28+
return new{NT, JB_FUD, JB_FXD, JB_FXU, JB_HD, JB_HX}(
29+
x, u, d,
30+
buffer_f_at_u_d,
31+
buffer_f_at_x_d,
32+
buffer_f_at_x_u,
33+
buffer_h_at_d,
34+
buffer_h_at_x
35+
)
36+
end
37+
38+
end
339

4-
Call [`linearize(model; x, u, d)`](@ref) and return the resulting linear model.
5-
"""
6-
LinModel(model::NonLinModel; kwargs...) = linearize(model; kwargs...)
40+
Base.show(io::IO, buffer::LinearizationBuffer) = print(io, "LinearizationBuffer object")
41+
42+
function LinearizationBuffer(NT, f!, h!, nu, nx, ny, nd, p)
43+
x, u, d, f_at_u_d!, f_at_x_d!, f_at_x_u!, h_at_d!, h_at_x! = get_linearize_funcs(
44+
NT, f!, h!, nu, nx, ny, nd, p
45+
)
46+
xnext, y = Vector{NT}(undef, nx), Vector{NT}(undef, ny) # TODO: to replace ?
47+
return LinearizationBuffer(
48+
x, u, d,
49+
JacobianBuffer(f_at_u_d!, xnext, x),
50+
JacobianBuffer(f_at_x_d!, xnext, u),
51+
JacobianBuffer(f_at_x_u!, xnext, d),
52+
JacobianBuffer(h_at_d!, y, x),
53+
JacobianBuffer(h_at_x!, y, d)
54+
)
55+
end
56+
57+
"Get the linearization functions for [`NonLinModel`](@ref) `f!` and `h!` functions."
58+
function get_linearize_funcs(NT, f!, h!, nu, nx, ny, nd, p)
59+
x = Vector{NT}(undef, nx)
60+
u = Vector{NT}(undef, nu)
61+
d = Vector{NT}(undef, nd)
62+
f_at_u_d!(xnext, x) = f!(xnext, x, u, d, p)
63+
f_at_x_d!(xnext, u) = f!(xnext, x, u, d, p)
64+
f_at_x_u!(xnext, d) = f!(xnext, x, u, d, p)
65+
h_at_d!(y, x) = h!(y, x, d, p)
66+
h_at_x!(y, d) = h!(y, x, d, p)
67+
return x, u, d, f_at_u_d!, f_at_x_d!, f_at_x_u!, h_at_d!, h_at_x!
68+
end
769

870
@doc raw"""
971
linearize(model::SimModel; x=model.x0+model.xop, u=model.uop, d=model.dop) -> linmodel
@@ -92,8 +154,8 @@ end
92154
93155
Linearize `model` and store the result in `linmodel` (in-place).
94156
95-
The keyword arguments are identical to [`linearize`](@ref). The code allocates a small
96-
amount of memory to compute the Jacobians.
157+
The keyword arguments are identical to [`linearize`](@ref). The code is allocation-free if
158+
`model` simulations does not allocate.
97159
98160
# Examples
99161
```jldoctest
@@ -112,26 +174,22 @@ function linearize!(
112174
linmodel::LinModel{NT}, model::SimModel; x=model.x0+model.xop, u=model.uop, d=model.dop
113175
) where NT<:Real
114176
nonlinmodel = model
115-
buffer = nonlinmodel.buffer
177+
buffer, linbuffer = nonlinmodel.buffer, nonlinmodel.linbuffer
116178
# --- remove the operating points of the nonlinear model (typically zeros) ---
117179
x0, u0, d0 = buffer.x, buffer.u, buffer.d
118180
u0 .= u .- nonlinmodel.uop
119181
d0 .= d .- nonlinmodel.dop
120182
x0 .= x .- nonlinmodel.xop
121183
# --- compute the Jacobians at linearization points ---
122-
A::Matrix{NT}, Bu::Matrix{NT}, Bd::Matrix{NT} = linmodel.A, linmodel.Bu, linmodel.Bd
123-
C::Matrix{NT}, Dd::Matrix{NT} = linmodel.C, linmodel.Dd
124184
xnext0::Vector{NT}, y0::Vector{NT} = linmodel.buffer.x, linmodel.buffer.y
125-
myf_x0!(xnext0, x0) = f!(xnext0, nonlinmodel, x0, u0, d0, model.p)
126-
myf_u0!(xnext0, u0) = f!(xnext0, nonlinmodel, x0, u0, d0, model.p)
127-
myf_d0!(xnext0, d0) = f!(xnext0, nonlinmodel, x0, u0, d0, model.p)
128-
myh_x0!(y0, x0) = h!(y0, nonlinmodel, x0, d0, model.p)
129-
myh_d0!(y0, d0) = h!(y0, nonlinmodel, x0, d0, model.p)
130-
ForwardDiff.jacobian!(A, myf_x0!, xnext0, x0)
131-
ForwardDiff.jacobian!(Bu, myf_u0!, xnext0, u0)
132-
ForwardDiff.jacobian!(Bd, myf_d0!, xnext0, d0)
133-
ForwardDiff.jacobian!(C, myh_x0!, y0, x0)
134-
ForwardDiff.jacobian!(Dd, myh_d0!, y0, d0)
185+
linbuffer.x .= x0
186+
linbuffer.u .= u0
187+
linbuffer.d .= d0
188+
jacobian!(linmodel.A, linbuffer.buffer_f_at_u_d, xnext0, x0)
189+
jacobian!(linmodel.Bu, linbuffer.buffer_f_at_x_d, xnext0, u0)
190+
jacobian!(linmodel.Bd, linbuffer.buffer_f_at_x_u, xnext0, d0)
191+
jacobian!(linmodel.C, linbuffer.buffer_h_at_d, y0, x0)
192+
jacobian!(linmodel.Dd, linbuffer.buffer_h_at_x, y0, d0)
135193
# --- compute the nonlinear model output at operating points ---
136194
xnext0, y0 = linmodel.buffer.x, linmodel.buffer.y
137195
h!(y0, nonlinmodel, x0, d0, model.p)

src/model/nonlinmodel.jl

+15-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
struct NonLinModel{
2-
NT<:Real, F<:Function, H<:Function, P<:Any, DS<:DiffSolver
2+
NT<:Real, F<:Function, H<:Function, P<:Any, DS<:DiffSolver, LB<:LinearizationBuffer
33
} <: SimModel{NT}
44
x0::Vector{NT}
55
f!::F
@@ -21,10 +21,11 @@ struct NonLinModel{
2121
yname::Vector{String}
2222
dname::Vector{String}
2323
xname::Vector{String}
24+
linbuffer::LB
2425
buffer::SimModelBuffer{NT}
2526
function NonLinModel{NT}(
26-
f!::F, h!::H, Ts, nu, nx, ny, nd, p::P, solver::DS
27-
) where {NT<:Real, F<:Function, H<:Function, P<:Any, DS<:DiffSolver}
27+
f!::F, h!::H, Ts, nu, nx, ny, nd, p::P, solver::DS, linbuffer::LB
28+
) where {NT<:Real, F<:Function, H<:Function, P<:Any, DS<:DiffSolver, LB<:LinearizationBuffer}
2829
Ts > 0 || error("Sampling time Ts must be positive")
2930
uop = zeros(NT, nu)
3031
yop = zeros(NT, ny)
@@ -38,7 +39,7 @@ struct NonLinModel{
3839
x0 = zeros(NT, nx)
3940
t = zeros(NT, 1)
4041
buffer = SimModelBuffer{NT}(nu, nx, ny, nd)
41-
return new{NT, F, H, P, DS}(
42+
return new{NT, F, H, P, DS, LB}(
4243
x0,
4344
f!, h!,
4445
p,
@@ -47,6 +48,7 @@ struct NonLinModel{
4748
nu, nx, ny, nd,
4849
uop, yop, dop, xop, fop,
4950
uname, yname, dname, xname,
51+
linbuffer,
5052
buffer
5153
)
5254
end
@@ -145,7 +147,8 @@ function NonLinModel{NT}(
145147
isnothing(solver) && (solver=EmptySolver())
146148
f!, h! = get_mutating_functions(NT, f, h)
147149
f!, h! = get_solver_functions(NT, solver, f!, h!, Ts, nu, nx, ny, nd)
148-
return NonLinModel{NT}(f!, h!, Ts, nu, nx, ny, nd, p, solver)
150+
linbuffer = LinearizationBuffer(NT, f!, h!, nu, nx, ny, nd, p)
151+
return NonLinModel{NT}(f!, h!, Ts, nu, nx, ny, nd, p, solver, linbuffer)
149152
end
150153

151154
function NonLinModel(
@@ -226,6 +229,13 @@ end
226229
"Do nothing if `model` is a [`NonLinModel`](@ref)."
227230
steadystate!(::SimModel, _ , _ ) = nothing
228231

232+
"""
233+
LinModel(model::NonLinModel; x=model.x0+model.xop, u=model.uop, d=model.dop)
234+
235+
Call [`linearize(model; x, u, d)`](@ref) and return the resulting linear model.
236+
"""
237+
LinModel(model::NonLinModel; kwargs...) = linearize(model; kwargs...)
238+
229239
"Call `model.f!(xnext0, x0, u0, d0, p)` for [`NonLinModel`](@ref)."
230240
f!(xnext0, model::NonLinModel, x0, u0, d0, p) = model.f!(xnext0, x0, u0, d0, p)
231241

src/sim_model.jl

+4-4
Original file line numberDiff line numberDiff line change
@@ -29,13 +29,13 @@ struct SimModelBuffer{NT<:Real}
2929
end
3030

3131
@doc raw"""
32-
SimModelBuffer{NT}(nu::Int, nx::Int, ny::Int, nd::Int)
32+
SimModelBuffer{NT}(nu::Int, nx::Int, ny::Int, nd::Int, linearization=nothing)
3333
3434
Create a buffer for `SimModel` objects for inputs, states, outputs, and disturbances.
3535
3636
The buffer is used to store intermediate results during simulation without allocating.
3737
"""
38-
function SimModelBuffer{NT}(nu::Int, nx::Int, ny::Int, nd::Int) where {NT <: Real}
38+
function SimModelBuffer{NT}(nu::Int, nx::Int, ny::Int, nd::Int, ) where {NT<:Real}
3939
u = Vector{NT}(undef, nu)
4040
x = Vector{NT}(undef, nx)
4141
y = Vector{NT}(undef, ny)
@@ -371,5 +371,5 @@ to_mat(A::Real, dims...) = fill(A, dims)
371371

372372
include("model/linmodel.jl")
373373
include("model/solver.jl")
374-
include("model/nonlinmodel.jl")
375-
include("model/linearization.jl")
374+
include("model/linearization.jl")
375+
include("model/nonlinmodel.jl")

test/1_test_sim_model.jl

+2
Original file line numberDiff line numberDiff line change
@@ -285,6 +285,8 @@ end
285285
@test linmodel1.Bd linmodel2.Bd
286286
@test linmodel1.C linmodel2.C
287287
@test linmodel1.Dd linmodel2.Dd
288+
@test repr(nonlinmodel1.linbuffer) == "LinearizationBuffer object"
289+
@test repr(nonlinmodel1.linbuffer.buffer_f_at_u_d) == "DifferentiationBuffer with a JacobianConfig"
288290

289291
f1!(ẋ, x, u, d, _) = (ẋ .= x.^5 + u.^4 + d.^3; nothing)
290292
h1!(y, x, d, _) = (y .= x.^2 + d; nothing)

test/4_test_plot_sim.jl

+2-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,8 @@
22
using .SetupMPCtests, ControlSystemsBase, LinearAlgebra
33
model = LinModel(sys, Ts, i_d=[3])
44
res = sim!(model, 15)
5-
display(res)
5+
6+
@test repr(res) == "Simulation results of LinModel with 15 time steps."
67
@test isa(res.obj, LinModel)
78
@test length(res.T_data) == 15
89
@test res.U_data[:, 1] model.uop .+ 1

0 commit comments

Comments
 (0)