RODESystem
Construction of RODESystem
A RODESystem
is represented by the state function
\[\begin{array}{l} dx = f(x, u, t, W) \end{array}\]
and the output function
\[ y = g(x, u, t)\]
where $t$ is the time, $x \in R^n$ is the state, $u \in R^p$ and $y \in R^m$ is output of the system. Therefore to construct a RODESystem
, we need to define statefunc
and outputfunc
with the corresponding syntax,
function statefunc(dx, x, u, t)
dx .= ... # Update dx
end
and
function outputfunc(x, u, t)
y = ... # Compute y
return y
end
As an example, consider the system with the state function
\[ \begin{array}{l} dx_1 = 2 x_1 sin(W_1 - W_2) \\ dx_2 = -2 x_2 cos(W_1 + W_2) \end{array}\]
and with the output function
\[ y = x\]
That is, all the state variable are taken as output. The statefunc
and the outputfunc
is defined as,
julia> function statefunc(dx, x, u, t, W)
dx[1] = 2x[1]*sin(W[1] - W[2])
dx[2] = -2x[2]*cos(W[1] + W[2])
end
statefunc (generic function with 1 method)
julia> outputfunc(x, u, t) = x
outputfunc (generic function with 1 method)
To construct the RODESystem
, we need to specify the initial condition and time.
julia> x0 = [1., 1.]
2-element Vector{Float64}:
1.0
1.0
julia> t = 0.
0.0
Note from statefunc
, the system has not any input, i.e. input is nothing, and has an output with a dimension of 1.
julia> input = nothing
julia> output = Outport(2)
2-element Outport{Outpin{Float64}}:
Outpin(eltype:Float64, isbound:false)
Outpin(eltype:Float64, isbound:false)
We are ready to construct the system
julia> ds = RODESystem(righthandside=statefunc, readout=outputfunc, state=x0, input=input, output=output, solverkwargs=(dt=0.01,))
ERROR: MethodError: no method matching FastBroadcast.BroadcastCharacteristics()
The applicable method may be too new: running in world age 47354, while current world is 95809.
Closest candidates are:
FastBroadcast.BroadcastCharacteristics() at /juliateam/.julia/packages/FastBroadcast/2JngL/src/FastBroadcast.jl:111 (method too new to be called from this world context.)
FastBroadcast.BroadcastCharacteristics(!Matched::Expr, !Matched::Vector{Symbol}, !Matched::Bool, !Matched::Bool) at /juliateam/.julia/packages/FastBroadcast/2JngL/src/FastBroadcast.jl:106 (method too new to be called from this world context.)
FastBroadcast.BroadcastCharacteristics(!Matched::Any, !Matched::Any, !Matched::Any, !Matched::Any) at /juliateam/.julia/packages/FastBroadcast/2JngL/src/FastBroadcast.jl:106 (method too new to be called from this world context.)
Note that ds
has a solver to solve its state function statefunc
which is random differential equation. To solve its statefunc
, the step size of the solver must be specified. See Random Differential Equtions
of DifferentialEquations
package.
Basic Operation of RODESystem
When a RODESystem
is triggered from its trigger
link, it read the current time from its trigger
link, reads its input (if available, i.e. its input is not nothing), solves its state function, computes its output value and writes its output value its output
bus (again, if available, i.e., its output bus is not nothing). To drive a RODESystem
, it must be launched
. Let us continue with ds
constructed in the previous section.
julia> iport, trg, hnd = Inport(2), Outpin(), Inpin{Bool}()
(Inport(numpins:2, eltype:Inpin{Float64}), Outpin(eltype:Float64, isbound:false), Inpin(eltype:Bool, isbound:false))
julia> connect!(ds.output, iport)
ERROR: UndefVarError: ds not defined
julia> connect!(trg, ds.trigger)
ERROR: UndefVarError: ds not defined
julia> connect!(ds.handshake, hnd)
ERROR: UndefVarError: ds not defined
julia> task = launch(ds)
ERROR: UndefVarError: ds not defined
julia> task2 = @async while true
all(take!(iport) .=== NaN) && break
end
Task (failed) @0x00007fedebfffc70
MethodError: no method matching take!(::Missing)
Closest candidates are:
take!(!Matched::IOBuffer) at iobuffer.jl:385
take!(!Matched::Base.GenericIOBuffer) at iobuffer.jl:370
take!(!Matched::Channel) at channels.jl:383
...
Stacktrace:
[1] take!(pin::Inpin{Float64})
@ Causal ~/.julia/packages/Causal/vOCIT/src/connections/pin.jl:111
[2] _broadcast_getindex_evalf
@ ./broadcast.jl:648 [inlined]
[3] _broadcast_getindex
@ ./broadcast.jl:621 [inlined]
[4] getindex
@ ./broadcast.jl:575 [inlined]
[5] macro expansion
@ ./broadcast.jl:984 [inlined]
[6] macro expansion
@ ./simdloop.jl:77 [inlined]
[7] copyto!
@ ./broadcast.jl:983 [inlined]
[8] copyto!
@ ./broadcast.jl:936 [inlined]
[9] copy
@ ./broadcast.jl:908 [inlined]
[10] materialize
@ ./broadcast.jl:883 [inlined]
[11] take!(inport::Inport{Inpin{Float64}})
@ Causal ~/.julia/packages/Causal/vOCIT/src/connections/port.jl:186
[12] macro expansion
@ ./none:2 [inlined]
[13] (::Main.ex-rode_system_ex.var"#1#2")()
@ Main.ex-rode_system_ex ./task.jl:406
When launched, ds
is ready to be driven. We can drive ds
by drive(ds, t)
or put!(ds.trigger, t)
where t
is the time until which we will drive ds
.
julia> put!(trg, 1.)
ERROR: MethodError: no method matching iterate(::Missing)
Closest candidates are:
iterate(!Matched::Union{LinRange, StepRangeLen}) at range.jl:664
iterate(!Matched::Union{LinRange, StepRangeLen}, !Matched::Int64) at range.jl:664
iterate(!Matched::T) where T<:Union{Base.KeySet{var"#s79", var"#s78"} where {var"#s79", var"#s78"<:Dict}, Base.ValueIterator{var"#s77"} where var"#s77"<:Dict} at dict.jl:693
...
When triggered, ds
read the time t
from its trigger
link, solved its differential equation, computed its value and writes its output value to its output
bus. To signal that, the evolution is succeeded, ds
writes true
to its handshake
link which must be taken to further drive ds
. (approve!(ds)
) can also be used.
julia> take!(hnd)
ERROR: MethodError: no method matching take!(::Missing)
Closest candidates are:
take!(!Matched::IOBuffer) at iobuffer.jl:385
take!(!Matched::Base.GenericIOBuffer) at iobuffer.jl:370
take!(!Matched::Channel) at channels.jl:383
...
We can continue to drive ds
.
julia> for t in 2. : 10.
put!(trg, t)
take!(hnd)
end
ERROR: MethodError: no method matching iterate(::Missing)
Closest candidates are:
iterate(!Matched::Union{LinRange, StepRangeLen}) at range.jl:664
iterate(!Matched::Union{LinRange, StepRangeLen}, !Matched::Int64) at range.jl:664
iterate(!Matched::T) where T<:Union{Base.KeySet{var"#s79", var"#s78"} where {var"#s79", var"#s78"<:Dict}, Base.ValueIterator{var"#s77"} where var"#s77"<:Dict} at dict.jl:693
...
After each evolution, ds
writes its current output value to its output
bus.
julia> [outbuf(pin.link.buffer) for pin in iport]
ERROR: type Missing has no field buffer
When launched, a task
was constructed which still running. As long as no exception is thrown during the evolution of ds
, the state of task
is running which implies ds
can be driven.
julia> task
ERROR: UndefVarError: task not defined
julia> task2
Task (failed) @0x00007fedebfffc70
MethodError: no method matching take!(::Missing)
Closest candidates are:
take!(!Matched::IOBuffer) at iobuffer.jl:385
take!(!Matched::Base.GenericIOBuffer) at iobuffer.jl:370
take!(!Matched::Channel) at channels.jl:383
...
Stacktrace:
[1] take!(pin::Inpin{Float64})
@ Causal ~/.julia/packages/Causal/vOCIT/src/connections/pin.jl:111
[2] _broadcast_getindex_evalf
@ ./broadcast.jl:648 [inlined]
[3] _broadcast_getindex
@ ./broadcast.jl:621 [inlined]
[4] getindex
@ ./broadcast.jl:575 [inlined]
[5] macro expansion
@ ./broadcast.jl:984 [inlined]
[6] macro expansion
@ ./simdloop.jl:77 [inlined]
[7] copyto!
@ ./broadcast.jl:983 [inlined]
[8] copyto!
@ ./broadcast.jl:936 [inlined]
[9] copy
@ ./broadcast.jl:908 [inlined]
[10] materialize
@ ./broadcast.jl:883 [inlined]
[11] take!(inport::Inport{Inpin{Float64}})
@ Causal ~/.julia/packages/Causal/vOCIT/src/connections/port.jl:186
[12] macro expansion
@ ./none:2 [inlined]
[13] (::Main.ex-rode_system_ex.var"#1#2")()
@ Main.ex-rode_system_ex ./task.jl:406
To terminate the task
safely, ds
should be terminated safely.
julia> put!(trg, NaN)
ERROR: MethodError: no method matching iterate(::Missing)
Closest candidates are:
iterate(!Matched::Union{LinRange, StepRangeLen}) at range.jl:664
iterate(!Matched::Union{LinRange, StepRangeLen}, !Matched::Int64) at range.jl:664
iterate(!Matched::T) where T<:Union{Base.KeySet{var"#s79", var"#s78"} where {var"#s79", var"#s78"<:Dict}, Base.ValueIterator{var"#s77"} where var"#s77"<:Dict} at dict.jl:693
...
julia> put!(ds.output, [NaN, NaN])
ERROR: UndefVarError: ds not defined
Note that the state of task
is done
which implies the task
has been terminated safely.
julia> task
ERROR: UndefVarError: task not defined
julia> task2
Task (failed) @0x00007fedebfffc70
MethodError: no method matching take!(::Missing)
Closest candidates are:
take!(!Matched::IOBuffer) at iobuffer.jl:385
take!(!Matched::Base.GenericIOBuffer) at iobuffer.jl:370
take!(!Matched::Channel) at channels.jl:383
...
Stacktrace:
[1] take!(pin::Inpin{Float64})
@ Causal ~/.julia/packages/Causal/vOCIT/src/connections/pin.jl:111
[2] _broadcast_getindex_evalf
@ ./broadcast.jl:648 [inlined]
[3] _broadcast_getindex
@ ./broadcast.jl:621 [inlined]
[4] getindex
@ ./broadcast.jl:575 [inlined]
[5] macro expansion
@ ./broadcast.jl:984 [inlined]
[6] macro expansion
@ ./simdloop.jl:77 [inlined]
[7] copyto!
@ ./broadcast.jl:983 [inlined]
[8] copyto!
@ ./broadcast.jl:936 [inlined]
[9] copy
@ ./broadcast.jl:908 [inlined]
[10] materialize
@ ./broadcast.jl:883 [inlined]
[11] take!(inport::Inport{Inpin{Float64}})
@ Causal ~/.julia/packages/Causal/vOCIT/src/connections/port.jl:186
[12] macro expansion
@ ./none:2 [inlined]
[13] (::Main.ex-rode_system_ex.var"#1#2")()
@ Main.ex-rode_system_ex ./task.jl:406
Full API
Causal.@def_rode_system
— Macro@def_rode_system ex
where ex
is the expression to define to define a new AbstractRODESystem component type. The usage is as follows:
@def_rode_system mutable struct MyRODESystem{T1,T2,T3,...,TN,OP,RH,RO,ST,IP,OP} <: AbstractRODESystem
param1::T1 = param1_default # optional field
param2::T2 = param2_default # optional field
param3::T3 = param3_default # optional field
⋮
paramN::TN = paramN_default # optional field
righthandside::RH = righthandside_function # mandatory field
readout::RO = readout_function # mandatory field
state::ST = state_default # mandatory field
input::IP = input_default # mandatory field
output::OP = output_default # mandatory field
end
Here, MyRODESystem
has N
parameters. MyRODESystem
is represented by the righthandside
and readout
function. state
, input
and output
is the initial state, input port and output port of MyRODESystem
.
righthandside
must have the signature
function righthandside((dx, x, u, t, W, args...; kwargs...)
dx .= .... # update dx
end
and readout
must have the signature
function readout(x, u, t)
y = ...
return y
end
New RODE system must be a subtype of AbstractRODESystem
to function properly.
Example
julia> @def_rode_system mutable struct MySystem{RH, RO, IP, OP} <: AbstractRODESystem
A::Matrix{Float64} = [2. 0.; 0 -2]
righthandside::RH = (dx, x, u, t, W) -> (dx .= A * x * W)
readout::RO = (x, u, t) -> x
state::Vector{Float64} = rand(2)
input::IP = nothing
output::OP = Outport(2)
end
julia> ds = MySystem();
Causal.RODESystem
— Typemutable struct RODESystem{RH, RO, ST<:(AbstractVector{var"#s16875"} where var"#s16875"<:Real), IP<:(Union{var"#s16874", var"#s16873"} where {var"#s16874"<:Inport, var"#s16873"<:Nothing}), OP<:(Union{var"#s16872", var"#s16632"} where {var"#s16872"<:Outport, var"#s16632"<:Nothing}), var"4004", var"4005", var"4006", Symbol, var"4007", Float64, var"4008", var"4009", var"4010", var"4011", var"4012", var"4013"} <: AbstractRODESystem
Constructs a generic RODE system
Fields
righthandside::Any
Right-hand-side function
readout::Any
Readout function
state::AbstractVector{var"#s16875"} where var"#s16875"<:Real
State
input::Union{var"#s16874", var"#s16873"} where {var"#s16874"<:Inport, var"#s16873"<:Nothing}
Input. Expected to be an
Inport
orNothing
output::Union{var"#s16872", var"#s16632"} where {var"#s16872"<:Outport, var"#s16632"<:Nothing}
Output port
trigger::Any
handshake::Any
callbacks::Any
name::Any
id::Any
t::Any
modelargs::Any
modelkwargs::Any
solverargs::Any
solverkwargs::Any
alg::Any
integrator::Any
Causal.MultiplicativeNoiseLinearSystem
— Typemutable struct MultiplicativeNoiseLinearSystem{T1<:(AbstractMatrix{var"#s16875"} where var"#s16875"<:Real), RH, RO, ST<:(AbstractVector{var"#s16874"} where var"#s16874"<:Real), IP<:(Union{var"#s16873", var"#s16872"} where {var"#s16873"<:Inport, var"#s16872"<:Nothing}), OP<:(Union{var"#s16632", var"#s16631"} where {var"#s16632"<:Outport, var"#s16631"<:Nothing}), var"4004", var"4005", var"4006", Symbol, var"4007", Float64, var"4008", var"4009", var"4010", var"4011", var"4012", var"4013"} <: AbstractRODESystem
Constructs a MultiplicativeNoiseLinearSystem
with the dynamics
\[\begin{array}{l} \dot{x} = A x W \end{array} where `W` is the noise process.\]
Fields
A::AbstractMatrix{var"#s16875"} where var"#s16875"<:Real
A
righthandside::Any
Right-hand-side function
readout::Any
Readout function
state::AbstractVector{var"#s16874"} where var"#s16874"<:Real
State
input::Union{var"#s16873", var"#s16872"} where {var"#s16873"<:Inport, var"#s16872"<:Nothing}
Input. Expected to be an
Inport
orNothing
output::Union{var"#s16632", var"#s16631"} where {var"#s16632"<:Outport, var"#s16631"<:Nothing}
Output port
trigger::Any
handshake::Any
callbacks::Any
name::Any
id::Any
t::Any
modelargs::Any
modelkwargs::Any
solverargs::Any
solverkwargs::Any
alg::Any
integrator::Any