# Robustness analysis of a MIMO system

In the following, we show an example of robustness analysis of a MIMO system, adapted from Sec. 8.11.3 in Skogestad, "Multivariable Feedback Control: Analysis and Design".

The process to be controlled is a spinning satellite.

$$$P(s) = \dfrac{1}{s^2+a^2} \begin{bmatrix} s - a^2 & a(s+1) \\ -a(s+1) & s-a^2 \end{bmatrix} , \quad a = 10$$$

We start by defining the nominal process model (a minimal realization of $P(s)$ rather than a transfer-function matrix)

using JuliaSimControls, LinearAlgebra, Plots
a = 10
P = ss([0 a; -a 0], I(2), [1 a; -a 1], 0)
w = exp10.(LinRange(-2, 3, 200))
sigmaplot(P, w)

As we can see, we have a large peak in the Bode plot, in fact, the system is only marginally stable with two eigenvalues on the imaginary axis (there is little air resistance in space after all).

We proceed to model the uncertainty we have about the system. We'll model a 20% actuator uncertainty at low frequencies, and let this uncertainty grow to 200% at large frequencies. This kind of uncertainty, diagonal, complex input uncertainty, is always present in physical systems.

W0 = makeweight(0.2, (1,1), 2) |> ss # Weight that goes from 0.2 at low frequencies to 2 at high frequencies
W = I(2) + W0*I(2) * uss([δc(), δc()]) # Create a diagonal complex uncertainty weighted in frequency by W0
Ps = P*W # Uncertain system
UncertainSS{Continuous}
A =
0.0  10.0  -1.590990257669732    0.0
-10.0   0.0   0.0                 -1.590990257669732
0.0   0.0  -1.7677669529663689   0.0
0.0   0.0   0.0                 -1.7677669529663689
B =
2.0  0.0  1.0  0.0
0.0  2.0  0.0  1.0
2.0  0.0  0.0  0.0
0.0  2.0  0.0  0.0
C =
0.0   0.0  0.0  0.0
0.0   0.0  0.0  0.0
1.0  10.0  0.0  0.0
-10.0   1.0  0.0  0.0
D =
0.0  0.0  1.0  0.0
0.0  0.0  0.0  1.0
0.0  0.0  0.0  0.0
0.0  0.0  0.0  0.0

Continuous-time state-space model

Ps is now represented as a upper linear fractional transform (upper LFT).

We can draw samples from this uncertainty representation (sampling of $\Delta$ and closing the loop starprod(Δ, Ps)) like so

Psamples = rand(Ps, 100)
sigmaplot(Psamples, w)

We can also extract the nominal model using

system_mapping(Ps)
StateSpace{Continuous, Float64}
A =
0.0  10.0
-10.0   0.0
B =
1.0  0.0
0.0  1.0
C =
1.0  10.0
-10.0   1.0
D =
0.0  0.0
0.0  0.0

Continuous-time state-space model

And obtain $M$ and $\Delta$ from the $M\Delta$ formulation when the loop is closed with $K$

K = ss(I(2))
lft(Ps, -K).M
StateSpace{Continuous, Float64}
A =
-1.0   0.0  -1.590990257669732    0.0
0.0  -1.0   0.0                 -1.590990257669732
0.0   0.0  -1.7677669529663689   0.0
0.0   0.0   0.0                 -1.7677669529663689
B =
2.0  0.0
0.0  2.0
2.0  0.0
0.0  2.0
C =
-1.0  -10.0  0.0  0.0
10.0   -1.0  0.0  0.0
D =
0.0  0.0
0.0  0.0

Continuous-time state-space model
Ps.Δ # Ps.delta also works
2-element Vector{Any}:
δ{ComplexF64, Float64}(0.0 + 0.0im, 1.0, Symbol("##δ#399"))
δ{ComplexF64, Float64}(0.0 + 0.0im, 1.0, Symbol("##δ#400"))

Most importantly, we can calculate the robustness margin of the system under the modeled uncertainty. For this, we calculate the structured singular value, $\mu$

μ = mussv(lft(Ps, -K))
7.3272705078125

The value of $\mu$ is very high, whenever $\mu > 1$, the system is not stable with respect to the modeled uncertainty. The tolerated uncertainty is only $\dfrac{1}{||\mu||_\infty}$

1/μ
0.13647646813827571

of the modeled uncertainty, this number is called the robust stability margin, and we would ideally want it to be $\geq 1$, indicating that we are robust with respect to at least 100% of the modeled uncertainty.

When we call mussv, an algorithm operating on the statespace representation is used to pinpoint the exact location of the peak structured singular value. Compare this to the grid-based method

plot(w, structured_singular_value(lft(Ps, -K), w), xscale=:log10, title="μ", xlabel="Frequency")

### Unmodeled actuator delay

If we had a known delay in the actuator, we could choose to either model it:

Pd = P*delay(0.1)

creates a model of the system with an input delay of 0.1 seconds. Another approach is to consider the delay as unmodeled dynamics and include an uncertainty description of it. To this purpose, we can create the system as

Pd = P * (I(2) + neglected_delay(0.1).*I(2)*uss([δc(), δc()]))
sigmaplot(rand(Pd, 100), w)

We see that neglecting a delay of no more than 0.1s has some effect on the dynamics around the resonance peak, but the uncertainty becomes rather large after that.

With such an uncertainty, the stability margin over frequencies is

μ = structured_singular_value(lft(Pd, -K), w)
plot(w, inv.(μ), scale=:log10, lab="Stability margin", xlabel="Frequency")
hline!(, lab="Stability boundary", l=(:red, :dash))

indicating that our primitive feedback controller K can just barely handle the uncertainty.