Previous: The “vat model” of computation, Up: A high level view of Goblins [Contents][Index]
An individual vat is not parallel, i.e. only one object within a vat can be working at any given time, however vats can work with one another cooperatively to multitask. This is done by asynchronously messages to objects on other vats, possibly waiting for their reply and then continuing work.
In this way it’s important to not block a vat (e.g. by calling sleep or reading/writing to a port, etc.) as that will block the event loop and prevent the vat doing work. Instead we should be using Goblins’ promise system in places where we’d block.
Lets take the following example which has two actors ^sleeper
,
which sleeps for 10 seconds then returns 'awake
, and
^hello
which just returns the symbol 'hello
. Both actors
are spawned within the same vat, we then use another vat to invoke
both of them at the same time ascynhronously:
(use-modules (goblins) (fibers)) (define-actor (^sleeper _bcom) (lambda () (sleep 10) 'awake)) (define-actor (^hello _bcom) (lambda () 'hello)) (define a-vat (spawn-vat)) (define sleeper (with-vat a-vat (spawn ^sleeper))) (define hello (with-vat a-vat (spawn ^hello))) (define b-vat (spawn-vat)) (with-vat b-vat (on (<- sleeper) (lambda (response) (pk 'sleeper-said response))) (on (<- hello) (lambda (response) (pk 'hello-said response))))
When we run this, the first thing we might notice is nothing happens for 10 seconds, then this is printed:
;;; (sleeper-said awake) ;;; (hello-said hello)
During those initial 10 seconds, the entire a-vat
is blocked
because of the (sleep 10)
call, this stops other objects like
our ^hello
object from doing anything. Instead, we should rely
on promises. The following code fixes the blocking issue:
(use-modules (goblins) (goblins vat) (fibers)) (define-actor (^sleeper _bcom) (lambda () (define sleep-vow (spawn-fibrous-vow (lambda () (sleep 10) #t))) (on sleep-vow (lambda _ 'awake) #:promise? #t))) (define-actor (^hello _bcom) (lambda () 'hello)) (define a-vat (spawn-vat)) (define sleeper (with-vat a-vat (spawn ^sleeper))) (define hello (with-vat a-vat (spawn ^hello))) (define b-vat (spawn-vat)) (with-vat b-vat (on (<- sleeper) (lambda (response) (pk 'sleeper-said response))) (on (<- hello) (lambda (response) (pk 'hello-said response))))
This time we get:
;;; (hello-said hello) ;;; (sleeper-said awake)
The initial response from ^hello
occurs immediately when we run
the code, then 10 seconds later we see ^sleeper
’s response.
This is because the vat is not blocked, we’re instead using
spawn-fibrous-vow
. Vats are built on top of the
Fibers concurrency library.
This procedure calls a thunk (a procedure of zero arguments) in a new
fiber. By spawning a new fiber, the vat’s event loop fiber is not
suspended when sleep
is called. spawn-fibrous-vow
returns a promise that will eventually resolve with the value returned
by the thunk.
This sleep example could be made even easier by using Goblins’ built
in timeout
procedure provided by Timers module. Other
tasks likely to block (e.g. reading or writing from ports or working
with other blocking resources) could be written with IO module,
which wraps the resource in a Goblins actor that ensures the vat is
never blocked.
Previous: The “vat model” of computation, Up: A high level view of Goblins [Contents][Index]