-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathinvariance.jl
392 lines (338 loc) · 12.7 KB
/
invariance.jl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
using LinearAlgebra, MatrixEquations, JuMP, Convex, SCS, COSMO, GLPK, Distributions
# Return true if x ∈ {x | Ax ≤ b}, otherwise return false
function in_polytope(x, A, b)
for i in 1:length(b)
A[i,:]⋅x > b[i] + ϵ ? (return false) : nothing
end
return true
end
# Return trajectory of discrete time system where each column is the state at step i
# Julia arrays are stored in column-major order so it's faster to do matrix[:,i] = data rather than matrix[i,:] = data
function compute_traj(init, steps::Int64, weights; type="normal")
dim = size(weights[1], 2) - 1
state_traj = Matrix{Float64}(undef, dim, steps+1)
state_traj[:,1] = init
for i in 1:steps
state_traj[:,i+1] = eval_net(state_traj[:,i], weights, 1, type=type)
end
return state_traj
end
# Specific for the pendulum
function plot_traj(state_traj)
plt = plot(reuse = false)
plot!(plt, t, rad2deg.(state_traj[1,:]), linewidth=3, legend=false, xlabel="Time (s.)", ylabel="Angle (deg.)", fontfamily=font(14, "Computer Modern"), yguidefont=(14) , xguidefont=(14), tickfont = (12))
return plt
end
# Iterate through each cell cᵢ in a PWA function and find fixed points if they exist.
function find_fixed_points(state2map, state2input, weights)
dim = size(weights[1], 2) - 1
fixed_points = Vector{Vector{Float64}}(undef, 0)
fp_dict = Dict{ Vector{Float64}, Vector{Tuple{Matrix{Float64},Vector{Float64}}} }() # stores [(A,b), (C,d)]
for ap in keys(state2map)
C, d = state2map[ap]
A, b = state2input[ap]
if rank(I - C) == dim
fp = inv(I - C) * d
if in_polytope(fp, A, b) && (eval_net(fp, weights, 1) ≈ fp) # second condition is sanity check
push!(fixed_points, fp)
fp_dict[fp] = [(A, b), (C, d)]
println("Found fixed point.")
if all(norm.(eigen(C).values) .< 1)
println("Fixed point is a stable equilibrium")
else
println("Fixed point is an unstable equilibrium")
end
end
else
error("Non-unique fixed point! Make more general")
end
end
return fixed_points, fp_dict
end
# Find local quadratic Lyapunov function of the form: (x - c)'Q⁻¹(x - c)
function local_stability(fp, fp_dict)
region = fp_dict[fp]
C, d = region[2]
dim = length(d)
F = eigen(C)
if all(norm.(F.values) .< 1) # then stable
# Transform affine sytem to linear system:
# xₜ₊₁ = Cxₜ + d, x_ₜ = xₜ - p ⟹ x_ₜ₊₁ = Cx_ₜ
# local Lyapunov function: x_'*Q*x_ = (x - p)'*Q*(x - p)
Q = lyapd(C', Matrix{Float64}(I, dim, dim)) # 2nd arg is symmetric. Where does it come from?
else
error("Unstable local system.")
end
return Q
end
# Compute Q̄ such that (x₊ - p)'*Q̄*(x₊ - p) ≤ 1 = {x₊ | (x - p)'*Q*(x - p) ≤ 1, x₊ = C*x + d}
function forward_reach_ellipse(Q, fp, fp_dict)
region = fp_dict[fp]
C, d = region[2]
C_inv = inv(C)
return C_inv'*Q*C_inv
end
# Solve the optimization problem:
# maximize a'x
# subject to (x - fp)'P(x - fp) ≤ 1
function lin_quad_opt(a, P_inv, fp)
x_opt = P_inv*a ./ sqrt(a⋅(P_inv*a)) + fp
opt_val = a⋅x_opt
return x_opt, opt_val
end
# Checks whether a given polyhedron is bounded (i.e. is a polyotpe)
# Can be made faster by solving Cheby LP
is_polytope(A, b) = isbounded(HPolyhedron(constraints_list(A, b)))
# Checks whether a given polytope is a subset of a given ellipsoid
function check_P_in_E(A, b, Q, α, fp)
is_polytope(A, b) ? nothing : (return false)
P_inv = inv(Q/α)
for i in 1:length(b)
x_opt, opt_val = lin_quad_opt(A[i,:], P_inv, fp)
opt_val < b[i] ? (return false) : nothing
end
return true
end
# find polytope P s.t. E_innner ⊂ P ⊂ E_outer
# E_inner = {x | (x - x_f)'(1\α)*Q̄*(x - x_f) ≤ 1}
# E_outer = {x | (x - x_f)'(1\α)*Q*(x - x_f) ≤ 1}
function intermediate_polytope(Q̄, α, fp, Aₓ, bₓ, C; max_constraints=100)
dim = length(fp)
# get constraints for linear system (x̄ frame)
A = deepcopy(Aₓ)
b = deepcopy(bₓ) - A*fp
P_inv = inv(Q̄/α)
for k in 1:max_constraints
A = vcat(A, reshape([bound_r(-1,1) for _ in 1:dim], 1,2))
x_opt, opt_val = lin_quad_opt(A[end,:], P_inv, zeros(dim)) # want constraints in the x̄ frame
b = vcat(b, [opt_val])
end
return A, b + A*fp # return constraints in the x frame
end
# generate random polytope that contains the fixed point
# we have that x̄ = x - fp
function random_polytope(fp, num_constraints)
F = [bound_r(-1, 1) for i in 1:num_constraints, j in 1:length(fp)]
b = ones(num_constraints)
return F, b + F*fp # return constraints in the x frame
end
# Plot convergence over time of some points to a fixed point
function convergence(fp, state2backward, weights, traj_length)
plt = plot(title="Distance to Fixed Point vs Time-Step", xlabel="Time-Step", ylabel="Distance")
for state in keys(state2backward)
Aᵢ, bᵢ = state2backward[state]
xₒ, nothing, nothing = cheby_lp([], [], Aᵢ, bᵢ, [])
state_traj = compute_traj(xₒ, traj_length, weights)
distances = [norm(state_traj[:,k] - fp) for k in 1:traj_length+1]
plot!(plt, 1:traj_length+1, distances, label=false)
end
return plt
end
# Solve polytope ROA SDP using JuMP
function solve_sdp_jump(Fₒ, C)
n_cons, dim = size(Fₒ)
model = Model(COSMO.Optimizer)
set_optimizer_attribute(model, "max_iter", 4000)
set_optimizer_attribute(model, "verbose", false)
@variable(model, Q_diag[1:n_cons])
Q = Matrix(Diagonal(Q_diag))
@variable(model, R[1:n_cons, 1:n_cons])
# @objective(model, Min, -sum(Q_diag) + sum(R))
@objective(model, Min, 0)
@constraint(model, Q*Fₒ*C - R'*Fₒ .== zeros(n_cons, dim))
@constraint(model, [Q R; R' Q] ≥ 0, PSDCone())
@constraint(model, Q_diag .≥ 10.0*ones(n_cons))
# @constraint(model, Q_diag .≤ 1e6*ones(n_cons))
@constraint(model, R .≥ zeros(n_cons, n_cons))
optimize!(model)
if termination_status(model) == MOI.OPTIMAL
return Diagonal(value.(Q_diag)), value.(R)
else
@show termination_status(model)
error("Suboptimal SDP!")
end
end
# Solve polytope ROA SDP using CVX
function solve_sdp_cvx(Fₒ, C)
n_cons, dim = size(Fₒ)
Q_diag = Variable(n_cons, Positive())
R = Variable(n_cons, n_cons, Positive())
# problem = minimize(-sum(Q_diag) + sum(R))
problem = minimize(0)
problem.constraints += diagm(Q_diag)*Fₒ*C - R'*Fₒ == zeros(n_cons, dim)
problem.constraints += ([diagm(Q_diag) R; R' diagm(Q_diag)] in :SDP)
problem.constraints += Q_diag ≥ ones(n_cons)
solve!(problem, () -> SCS.Optimizer())
if problem.status == MOI.OPTIMAL
return Diagonal(evaluate(Q_diag)), evaluate(R)
else
@show problem.status
error("Suboptimal SDP!")
end
end
# Solve polytope ROA LP using JuMP
function solve_lp(Pₒ, n_cons, n_consₓ)
model = Model(GLPK.Optimizer)
@variable(model, λ[1:n_cons+n_consₓ])
@objective(model, Max, sum(λ[1:n_cons]))
@constraint(model, (Pₒ' - I)*λ .≤ zeros(n_cons+n_consₓ))
@constraint(model, λ[n_cons+1:end] .≤ ones(n_consₓ))
# @constraint(model, λ[1:n_cons] .≤ 1e2*ones(n_cons)) # extra
@constraint(model, λ .≥ 1e-6*ones(n_cons+n_consₓ)) # extra
optimize!(model)
if termination_status(model) == MOI.OPTIMAL
return value.(λ)
else
@show termination_status(model)
error("Suboptimal LP!")
end
end
# Compute polytopic ROA #
function invariant_polytope(Aₓ, bₓ, Aₒ, bₒ, C)
n_consₓ, dim = size(Aₓ)
n_cons = length(bₒ)
Fₓ = vcat([reshape(Aₓ[i,:] ./ bₓ[i], (1, :)) for i in 1:length(bₓ)]...)
Fₒ = vcat([reshape(Aₒ[i,:] ./ bₒ[i], (1, :)) for i in 1:length(bₒ)]...)
Fₒ = vcat(Fₒ, Fₓ)
# solve SDP
Q_jmp, R_jmp = solve_sdp_jump(Fₒ, C)
Pₒ_jmp = R_jmp*inv(Q_jmp)
λ = solve_lp(Pₒ_jmp, n_cons, n_consₓ)
F_res = inv(Diagonal(λ))*Fₒ
# Verify that found polytope is invariant
if is_invariant(F_res, C)
println("Found invariant polytope!")
return F_res, ones(size(F_res,1))
else
println("Polytope not invariant.")
return F_res, ones(size(F_res,1))
end
end
# check if polytope Fx≤1 is invariant under dynamics xₜ₊₁ = Cxₜ
# The polytope is invariant iff ∃ P ∈ PSD s.t.
# FA = PF && P1 ≤ 1
function is_invariant(F, C)
n_cons, dim = size(F)
model = Model(GLPK.Optimizer)
@variable(model, P[1:n_cons, 1:n_cons])
@objective(model, Min, 0) # feasibility
@constraint(model, P.≥ zeros(n_cons, n_cons))
@constraint(model, P*F .== F*C )
@constraint(model, P*ones(n_cons).≤ ones(n_cons))
optimize!(model)
if termination_status(model) == MOI.OPTIMAL
return true
else
@show termination_status(model)
return false
end
end
# Attempt to find a polytopic ROA via the SDP method
function polytope_roa_sdp(Aₓ, bₓ, C, fp, num_constraints)
# Generate initial polytope
Aₛ, bₛ = random_polytope(fp, num_constraints)
# Put constraints in x̄ frame
Aₓ_ = Aₓ
bₓ_ = bₓ - Aₓ_*fp
Aₛ_ = Aₛ
bₛ_ = bₛ - Aₛ_*fp
A_roa_, b_roa_ = invariant_polytope(Aₓ_, bₓ_, Aₛ_, bₛ_, C)
return A_roa_, b_roa_ + A_roa_*fp # return constraints in the x frame
end
# Given fixed points and fp_dict find one that is a local attractor
function find_attractor(fixed_points, fp_dict)
for (i, fp) in enumerate(fixed_points)
region = fp_dict[fp]
Aₓ, bₓ = region[1]
C, d = region[2]
if all(norm.(eigen(C).values) .< 1)
println("Verified fixed point is a local attractor.")
return fp, C, d, Aₓ, bₓ
elseif i == length(fixed_points)
error("Unstable local system")
end
end
return fp, C, d, Aₓ, bₓ
end
# Checks whether the PWA function given by state2map is a homeomorphism
function is_homeomorphism(state2map, dim)
signs = Vector{Int64}(undef, dim)
for (i, key) in enumerate(keys(state2map))
C, d = state2map[key]
signs_i = [logabsdet(C)[2] for n in 1:dim]
if i == 1
signs = signs_i
elseif signs != signs_i || 0 ∈ signs_i
return false
end
end
return true
end
# Given a NN model, number of seed polytope ROA constraints, and number of backward reachability steps, compute ROA
function find_roa(dynamics::String, nn_weights, num_constraints, num_steps; nn_params="")
if dynamics == "pendulum"
weights = pendulum_net(nn_file, 1)
Aᵢ, bᵢ = input_constraints_pendulum(weights, "pendulum")
Aₒ, bₒ = output_constraints_pendulum(weights, "origin")
println("Input set: pendulum")
elseif dynamics == "vanderpol"
weights = pytorch_net(nn_weights, nn_params, 1)
Aᵢ, bᵢ = input_constraints_vanderpol(weights, "box")
Aₒ, bₒ = output_constraints_vanderpol(weights, "origin")
println("Input set: van der Pol box")
elseif dynamics == "mpc"
weights = pytorch_mpc_net("mpc", 1)
Aᵢ, bᵢ = input_constraints_mpc(weights, "box")
Aₒ, bₒ = output_constraints_mpc(weights, "origin")
println("Input set: MPC box")
elseif dynamics == "taxinet"
weights = taxinet_cl()
Aᵢ, bᵢ = input_constraints_taxinet(weights)
Aₒ, bₒ = output_constraints_taxinet(weights)
println("Input set: Taxinet box")
else
error("Unrecognized dynamics!")
end
# Run algorithm on one-step dynamics
@time begin
state2input, state2output, state2map, state2backward = compute_reach(weights, Aᵢ, bᵢ, [Aₒ], [bₒ])
end
println("Dynamics function has ", length(state2input), " affine regions.")
# Check if PWA function is a homeomorphism
homeomorph = is_homeomorphism(state2map, size(Aᵢ,2))
println("PWA function is a homeomorphism: ", homeomorph)
# Find fixed point(s) #
fixed_points, fp_dict = find_fixed_points(state2map, state2input, weights)
fp, C, d, Aₓ, bₓ = find_attractor(fixed_points, fp_dict)
# Find Polytopic ROA via SDP method #
A_roa, b_roa = polytope_roa_sdp(Aₓ, bₓ, C, fp, num_constraints)
println("Found seed ROA!")
# Perform backward reachability on the seed invariant set
if dynamics == "pendulum"
weights_chain = pendulum_net(model, num_steps)
elseif dynamics == "vanderpol"
weights_chain = pytorch_net(nn_weights, nn_params, num_steps)
elseif dynamics == "mpc"
weights_chain = pytorch_mpc_net("mpc", num_steps)
elseif dynamics == "taxinet"
num_layers = length(weights)
layer_sizes = [size(weights[i],2) for i in 1:num_layers]
push!(layer_sizes, size(weights[end],1))
weights_chain = chain_net(weights, num_steps+1, num_layers, layer_sizes)
end
state2input_chain, state2output_chain, state2map_chain, state2backward_chain = compute_reach(weights_chain, Aᵢ, bᵢ, [A_roa], [b_roa], fp=fp, back=true, connected=true)
if dynamics == "pendulum"
plt_in2 = plot_hrep_pendulum(state2backward_chain[1])
plot!(plt_in2, title=string(num_steps, "-Step BRS"), xlims=(-90, 90), ylims=(-90, 90))
elseif dynamics == "vanderpol"
plt_in2 = plot_hrep_vanderpol(state2backward_chain[1])
plot!(plt_in2, title=string(num_steps, "-Step BRS"), xlims=(-3, 3), ylims=(-3, 3))
elseif dynamics == "mpc"
plt_in2 = plot_hrep_mpc(state2backward_chain[1])
plot!(plt_in2, title=string(num_steps, "-Step BRS"), xlims=(-5, 5), ylims=(-5, 5))
elseif dynamics == "taxinet"
plt_in2 = polytopes(state2backward_chain[1])
plot!(plt_in2, title=string(num_steps, "-Step BRS"), xlims=(-5, 5), ylims=(-5, 5))
end
return A_roa, b_roa, fp, state2backward_chain[1], plt_in2
end