Skip to content

Commit bd283f0

Browse files
authored
add parameter conversion helper (#16)
* add parameter conversion helper * simplify 1/0 is Inf anyway * qualify CSBase
1 parent 9094109 commit bd283f0

File tree

4 files changed

+58
-2
lines changed

4 files changed

+58
-2
lines changed

README.md

+11
Original file line numberDiff line numberDiff line change
@@ -201,6 +201,17 @@ plot(t, [Ym Um], layout=(2,1), ylabel = ["y" "u"], legend=false)
201201
```
202202
Once again, the output looks identical and is therefore omitted here.
203203

204+
## Parameter conversion
205+
The form of the PID controller used in this package is often referred to as ["standard form"](https://en.wikipedia.org/wiki/Proportional%E2%80%93integral%E2%80%93derivative_controller#Standard_form). If you have PID parameters on "parallel form"
206+
```math
207+
K_p (br-y) + K_i (r-y)/s - K_d s y/(Tf s + 1)
208+
```
209+
you may convert these to the standard form
210+
```math
211+
K (b r - y + 1/T_i (r - y) - s T_d y/(1 + s T_d / N))
212+
```
213+
using the function `K, Ti, Td = parallel2standard(kp, ki, kd)` or, if a filter parameter is included, `K, Ti, Td, N = parallel2standard(kp, ki, kd, Tf)`. This function also accepts a vector of parameters in the same order, in which case a vector is returned.
214+
204215
## Details
205216
- The derivative term only acts on the (filtered) measurement and not the command signal. It is thus safe to pass step changes in the reference to the controller. The parameter $b$ can further be set to zero to avoid step changes in the control signal in response to step changes in the reference.
206217
- Bumpless transfer when updating `K` is realized by updating the state `I`. See the docs for `set_K!` for more details.

examples/juliac/juliac_pid.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -46,5 +46,5 @@ end
4646

4747
# compile using something like this, modified to suit your local paths
4848
# cd(@__DIR__)
49-
# run(`/home/fredrikb/repos/julia/julia --project --experimental /home/fredrikb/repos/julia/contrib/juliac.jl --output-lib juliac_pid --experimental --trim=unsafe-warn --compile-ccallable juliac_pid.jl`)
49+
# run(`/home/fredrikb/repos/julia/julia --project --experimental /home/fredrikb/repos/julia/contrib/juliac.jl --output-lib juliac_pid --trim=unsafe-warn --experimental --compile-ccallable juliac_pid.jl`)
5050
# run(`ls -ltrh`)

src/DiscretePIDs.jl

+38-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
module DiscretePIDs
22

3-
export DiscretePID, calculate_control!, set_K!, set_Td!, set_Ti!, reset_state!
3+
export DiscretePID, calculate_control!, set_K!, set_Td!, set_Ti!, reset_state!, parallel2standard
44

55
using Printf
66

@@ -201,4 +201,41 @@ function reset_state!(pid::DiscretePID)
201201
nothing
202202
end
203203

204+
"""
205+
K, Ti, Td = parallel2standard(Kp, Ki, Kd)
206+
207+
Convert parameters from form "parallel" form
208+
``K_p + K_i/s + K_d s``
209+
210+
to "standard" form used in DiscretePID:
211+
``K(1 + 1/(T_i s) + T_d s)``
212+
213+
You may provide either three arguments or an array with three elements in the same order.
214+
"""
215+
function parallel2standard(Kp, Ki, Kd)
216+
Kp == 0 && throw(DomainError("Cannot convert to standard form when Kp=0"))
217+
return (Kp, Kp / Ki, Kd / Kp)
218+
end
219+
220+
"""
221+
K, Ti, Td, N = parallel2standard(Kp, Ki, Kd, Tf)
222+
223+
Convert parameters from form "parallel" form with first-order filter
224+
``K_p (br-y) + K_i (r-y)/s - K_d s y/(Tf s + 1)``
225+
226+
to "standard" form used in DiscretePID:
227+
``K (br-y + (r-y)/(T_i s) - T_d s y/(T_d / N s + 1))``
228+
229+
You may provide either four arguments or an array with four elements in the same order.
230+
"""
231+
function parallel2standard(Kp, Ki, Kd, Tf)
232+
Kp, Ti, Td = parallel2standard(Kp, Ki, Kd)
233+
N = Td / Tf
234+
return (Kp, Ti, Td, N)
235+
end
236+
237+
function parallel2standard(p)
238+
return [parallel2standard(p...)...]
239+
end
240+
204241
end

test/runtests.jl

+8
Original file line numberDiff line numberDiff line change
@@ -183,4 +183,12 @@ res2 = lsim(P, ctrl, 3)
183183
@test DiscretePID(Ts=1f0) isa DiscretePID{Float32}
184184
@test DiscretePID(Ts=1.0) isa DiscretePID{Float64}
185185

186+
kpkikdTf = rand(4)
187+
kp, ki, kd, Tf = kpkikdTf
188+
ps = parallel2standard(kpkikdTf)
189+
K,Ti,Td,N = ps
190+
191+
@test ControlSystemsBase.pid(kp, ki, kd; Tf, form = :parallel, filter_order=1) ControlSystemsBase.pid(K, Ti, Td; Tf=Td/N, form = :standard, filter_order=1)
192+
193+
186194
end

0 commit comments

Comments
 (0)