Internals
The macro @resumable
transform a function definition into a finite state-machine, i.e. a callable type holding the state and references to the internal variables of the function and a constructor for this new type respecting the method signature of the original function definition. When calling the new type a modified version of the body of the original function definition is executed:
- a dispatch mechanism is inserted at the start to allow a non local jump to a label inside the body;
- the
@yield
statement is replaced by areturn
statement and a label placeholder as endpoint of a non local jump; for
loops are transformed inwhile
loops andtry
-catch
-finally
-end
expressions are converted in a sequence oftry
-catch
-end
expressions with at the end of thecatch
part a non local jump to a label that marks the beginning of the expressions in thefinally
part.
The two last transformations are needed to overcome the limitations of the non local jump macros @goto
and @label
.
The complete procedure is explained using the following example:
@resumable function fibonacci(n::Int)
a = 0
b = 1
for i in 1:n-1
@yield a
a, b = b, a + b
end
a
end
Split the definition
The function definition is split by MacroTools.splitdef
in different parts, eg. :name
, :body
, :args
, ...
For loops
for
loops in the body of the function definition are transformed in equivalent while loops:
begin
a = 0
b = 1
_iter = 1:n-1
_iterstate = start(_iter)
while !done(_iter, _iterstate)
i, _iterstate = next(_iter, _iterstate)
@yield a
a, b = b, a + b
end
a
end
Slots
The slots and their types in the function definition are recovered by running the code_typed
function for the locally evaluated function definition:
n :: Int64
a :: Int64
b :: Int64
_iter :: UnitRange{Int64}
_iterstate :: Int64
i :: Int64
Type definition
A mutable struct
is defined containing the state and the slots:
mutable struct ##123 <: ResumableFunctions.FiniteStateMachineIterator
_state :: UInt8
n :: Int64
a :: Int64
b :: Int64
_iter :: UnitRange{Int64}
_iterstate :: Int64
i :: Int64
function ##123()
fsmi = new()
fsmi._state = 0x00
fsmi
end
end
Call definition
A call function is constructed that creates the previously defined composite type. This function satisfy the calling convention of the original function definition and is returned from the macro:
function fibonacci(n::Int)
fsmi = ##123(n)
fsmi.n = n
fsmi
end
Transformation of the body
In 6 steps the body of the function definition is transformed into a finite state-machine.
Renaming of slots
The slots are replaced by references to the fields of the composite type:
begin
_fsmi.a = 0
_fsmi.b = 1
_fsmi._iter = 1:n-1
_fsmi._iterstate = start(_fsmi._iter)
while !done(_fsmi._iter, _fsmi._iterstate)
_fsmi.i, _fsmi._iterstate = next(_fsmi._iter, _fsmi._iterstate)
@yield _fsmi.a
_fsmi.a, _fsmi.b = _fsmi.b, _fsmi.a + _fsmi.b
end
_fsmi.a
end
Two-way communication
All expressions of the form _fsmi.arg = @yield
are transformed into:
@yield
_fsmi.arg = _arg
Exception handling
Exception handling is added to @yield
:
begin
_fsmi.a = 0
_fsmi.b = 1
_fsmi._iter = 1:n-1
_fsmi._iterstate = start(_fsmi._iter)
while !done(_fsmi._iter, _fsmi._iterstate)
_fsmi.i, _fsmi._iterstate = next(_fsmi._iter, _fsmi._iterstate)
@yield _fsmi.a
_arg isa Exception && throw(_arg)
_fsmi.a, _fsmi.b = _fsmi.b, _fsmi.a + _fsmi.b
end
_fsmi.a
end
Try catch finally end block handling
try
-catch
-finally
-end
expressions are converted in a sequence of try
-catch
-end
expressions with at the end of the catch
part a non local jump to a label that marks the beginning of the expressions in the finally
part.
Yield transformation
The @yield
statement is replaced by a return
statement and a label placeholder as endpoint of a non local jump:
begin
_fsmi.a = 0
_fsmi.b = 1
_fsmi._iter = 1:n-1
_fsmi._iterstate = start(_fsmi._iter)
while !done(_fsmi._iter, _fsmi._iterstate)
_fsmi.i, _fsmi._iterstate = next(_fsmi._iter, _fsmi._iterstate)
_fsmi._state = 0x01
return _fsmi.a
@label _STATE_1
_fsmi._state = 0xff
_arg isa Exception && throw(_arg)
_fsmi.a, _fsmi.b = _fsmi.b, _fsmi.a + _fsmi.b
end
_fsmi.a
end
Dispatch mechanism
A dispatch mechanism is inserted at the start of the body to allow a non local jump to a label inside the body:
begin
_fsmi_state == 0x00 && @goto _STATE_0
_fsmi_state == 0x01 && @goto _STATE_1
error("@resumable function has stopped!")
@label _STATE_0
_fsmi._state = 0xff
_arg isa Exception && throw(_arg)
_fsmi.a = 0
_fsmi.b = 1
_fsmi._iter = 1:n-1
_fsmi._iterstate = start(_fsmi._iter)
while !done(_fsmi._iter, _fsmi._iterstate)
_fsmi.i, _fsmi._iterstate = next(_fsmi._iter, _fsmi._iterstate)
_fsmi._state = 0x01
return _fsmi.a
@label _STATE_1
_fsmi._state = 0xff
_arg isa Exception && throw(_arg)
_fsmi.a, _fsmi.b = _fsmi.b, _fsmi.a + _fsmi.b
end
_fsmi.a
end
Making the type callable
A function is defined with one argument _arg
:
function (_fsmi::##123)(_arg::Any=nothing)
_fsmi_state == 0x00 && @goto _STATE_0
_fsmi_state == 0x01 && @goto _STATE_1
error("@resumable function has stopped!")
@label _STATE_0
_fsmi._state = 0xff
_arg isa Exception && throw(_arg)
_fsmi.a = 0
_fsmi.b = 1
_fsmi._iter = 1:n-1
_fsmi._iterstate = start(_fsmi._iter)
while !done(_fsmi._iter, _fsmi._iterstate)
_fsmi.i, _fsmi._iterstate = next(_fsmi._iter, _fsmi._iterstate)
_fsmi._state = 0x01
return _fsmi.a
@label _STATE_1
_fsmi._state = 0xff
_arg isa Exception && throw(_arg)
_fsmi.a, _fsmi.b = _fsmi.b, _fsmi.a + _fsmi.b
end
_fsmi.a
end