## Robust Control Toolbox |

This example uses `systune`

to explore trade-offs between setpoint tracking and disturbance rejection when tuning PID controllers.

When tuning 1-DOF PID controllers, it is often impossible to achieve good tracking and fast disturbance rejection at the same time. Assuming the control bandwidth is fixed, faster disturbance rejection requires more gain inside the bandwidth, which can only be achieved by increasing the slope near the crossover frequency. Because a larger slope means a smaller phase margin, this typically comes at the expense of more overshoot in the response to setpoint changes.

**Figure 1: Trade-off in 1-DOF PID Tuning.**

This example uses `systune`

to explore this trade-off and find the right compromise for your application. See also `pidtool`

for a more direct way to make such trade-off (see "Design Focus" under Controller Options).

Consider the PID loop of Figure 2 with a load disturbance at the plant input.

**Figure 2: PID Control Loop.**

For this example we use the plant model

The target control bandwidth is 10 rad/s. Create a tunable PID controller and fix its derivative filter time constant to (10 times the bandwidth) so that there are only three gains to tune (proportional, integral, and derivative gains).

G = zpk(-5,[-1 -2 -10],10); C = ltiblock.pid('C','pid'); C.Tf.Value = 0.01; C.Tf.Free = false; % fix Tf=0.01

Construct a tunable model `T0`

of the closed-loop transfer from `r`

to `y`

. Use an "analysis point" block to mark the location `u`

where the disturbance enters.

LS = AnalysisPoint('u'); T0 = feedback(G*LS*C,1); T0.u = 'r'; T0.y = 'y';

The gain of the open-loop response is a key indicator of the feedback loop behavior. The open-loop gain should be high (greater than one) inside the control bandwidth to ensure good disturbance rejection, and should be low (less than one) outside the control bandwidth to be insensitive to measurement noise and unmodeled plant dynamics. Accordingly, use three requirements to express the control objectives:

"Tracking" requirement to specify a response time of about 0.2 seconds to step changes in

`r`

"MaxLoopGain" requirement to force a roll-off of -20 dB/decade past the crossover frequency 10 rad/s

"MinLoopGain" requirement to adjust the integral gain at frequencies below 0.1 rad/s.

s = tf('s'); wc = 10; % target crossover frequency % Tracking R1 = TuningGoal.Tracking('r','y',2/wc); % Bandwidth and roll-off R2 = TuningGoal.MaxLoopGain('u',wc/s); % Disturbance rejection R3 = TuningGoal.MinLoopGain('u',wc/s); R3.Focus = [0 0.1];

Use `systune`

to tune the PID gains to meet these requirements. Treat the bandwidth and disturbance rejection goals as hard constraints and optimize tracking subject to these constraints.

T1 = systune(T0,R1,[R2 R3]);

Final: Soft = 1.12, Hard = 0.99971, Iterations = 112

Verify that all three requirements are nearly met. The blue curves are the achieved values and the yellow patches highlight regions where the requirements are violated.

```
figure('Position',[100,100,560,580])
viewSpec([R1 R2 R3],T1)
```

To gain insight into the trade-off between tracking and disturbance rejection, increase the minimum loop gain in the frequency band [0,0.1] rad/s by a factor . Re-tune the PID gains for the values .

```
% Increase loop gain by factor 2
alpha = 2;
R3.MinGain = alpha*wc/s;
T2 = systune(T0,R1,[R2 R3]);
```

Final: Soft = 1.17, Hard = 0.99948, Iterations = 181

```
% Increase loop gain by factor 4
alpha = 4;
R3.MinGain = alpha*wc/s;
T3 = systune(T0,R1,[R2 R3]);
```

Final: Soft = 1.27, Hard = 0.99962, Iterations = 135

Compare the responses to a step command `r`

and to a step disturbance `d`

entering at the plant input `u`

.

figure, step(T1,T2,T3,3) title('Setpoint tracking') legend('\alpha = 1','\alpha = 2','\alpha = 4')

% Compute closed-loop transfer from u to y D1 = getIOTransfer(T1,'u','y'); D2 = getIOTransfer(T2,'u','y'); D3 = getIOTransfer(T3,'u','y'); step(D1,D2,D3,10) title('Disturbance rejection') legend('\alpha = 1','\alpha = 2','\alpha = 4')

Note how disturbance rejection improves as `alpha`

increases, but at the expense of increased overshoot in setpoint tracking. Plot the open-loop responses for the three designs, and note how the slope before crossover (0dB) increases with `alpha`

.

L1 = getLoopTransfer(T1,'u'); L2 = getLoopTransfer(T2,'u'); L3 = getLoopTransfer(T3,'u'); bodemag(L1,L2,L3,{1e-2,1e2}), grid title('Open-loop response') legend('\alpha = 1','\alpha = 2','\alpha = 4')

Which design is most suitable depends on the primary purpose of the feedback loop you are tuning.

If you cannot compromise tracking to improve disturbance rejection, consider using a 2-DOF architecture instead. A 2-DOF PID controller is capable of fast disturbance rejection without significant increase of overshoot in setpoint tracking.

**Figure 3: 2-DOF PID Control Loop.**

Use the `ltiblock.pid2`

object to parameterize the 2-DOF PID controller and construct a tunable model `T0`

of the closed-loop system in Figure 3.

C = ltiblock.pid2('C','pid'); C.Tf.Value = 0.01; C.Tf.Free = false; % fix Tf=0.01 T0 = feedback(G*LS*C,1,2,1,+1); T0 = T0(:,1); T0.u = 'r'; T0.y = 'y';

Next tune the 2-DOF PI controller for the largest loop gain tried earlier ( ).

% Minimum loop gain inside bandwidth (for disturbance rejection) alpha = 4; R3.MinGain = alpha*wc/s; % Tune 2-DOF PI controller T4 = systune(T0,R1,[R2 R3]);

Final: Soft = 1.09, Hard = 0.99976, Iterations = 139

Compare the setpoint tracking and disturbance rejection properties of the 1-DOF and 2-DOF designs for .

clf, step(T3,'b',T4,'g--',4) title('Setpoint tracking') legend('1-DOF','2-DOF')

D4 = getIOTransfer(T4,'u','y'); step(D3,'b',D4,'g--',4) title('Disturbance rejection') legend('1-DOF','2-DOF')

The responses to a step disturbance are similar but the 2-DOF controller eliminates the overshoot in the response to a setpoint change. You can use `showTunable`

to compare the tuned gains in the 1-DOF and 2-DOF controllers.

```
showTunable(T3) % 1-DOF PI
```

C = 1 s Kp + Ki * --- + Kd * -------- s Tf*s+1 with Kp = 9.64, Ki = 16.1, Kd = 0.885, Tf = 0.01 Name: C Continuous-time PIDF controller in parallel form.

```
showTunable(T4) % 2-DOF PI
```

C = 1 s u = Kp (b*r-y) + Ki --- (r-y) + Kd -------- (c*r-y) s Tf*s+1 with Kp = 6.8, Ki = 19.9, Kd = 0.97, Tf = 0.01, b = 0.739, c = 1.12 Name: C Continuous-time 2-DOF PIDF controller in parallel form.