-
Notifications
You must be signed in to change notification settings - Fork 115
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Adding page about eval & differences to expect in interactive & compiled mode. Motivated by #1253 Maybe we could gather `eval` related practices on this page?
- Loading branch information
Showing
2 changed files
with
94 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
# Eval | ||
|
||
## Overview | ||
|
||
The `eval` function in Gerbil Scheme is a powerful tool for dynamic code execution. However, its behavior and usage differ between runtime and compile-time, and it has some key differences compared to Gambit Scheme. This document outlines how to use `eval` effectively in various scenarios. | ||
|
||
## Runtime Behavior | ||
|
||
### Interactive Environment | ||
|
||
In the Gerbil interpreter (`gxi`), `eval` works similarly to other Scheme implementations: | ||
|
||
```scheme | ||
> (def greeting "hello world") | ||
> (eval '(displayln greeting)) | ||
hello world | ||
``` | ||
|
||
### Compiled Binaries | ||
|
||
When using `eval` in compiled binaries, additional steps are necessary. | ||
To demonstrate this let's slightly modify code from previous example, save it into the file: | ||
|
||
> `eval-example.ss` | ||
```scheme | ||
(export main) | ||
(def greeting "hello world") | ||
(def (main . args) | ||
(eval '(displayln greeting))) | ||
``` | ||
|
||
Then compile into executable and run: | ||
|
||
```console | ||
$ gxc -exe eval-example.ss | ||
... | ||
$ ./eval-example | ||
*** ERROR -- Unbound variable: greeting | ||
--- continuation backtrace: | ||
[0] ##primordial-exception-handler-hook | ||
``` | ||
|
||
An error is reported because eval operates on top-level namespace while `greeting` is defined a module namespace which is added implicitly. | ||
|
||
This may be fixed by adding a namespace to `greeting`: | ||
|
||
```scheme | ||
(export main) | ||
(def greeting "hello world") | ||
(def (main . args) | ||
(eval '(displayln eval-example#greeting)) | ||
``` | ||
|
||
```console | ||
$ gxc -exe eval-example.ss | ||
... | ||
$ ./eval-example | ||
hello world | ||
``` | ||
|
||
Adding namespace to each symbol referenced under `eval` may be a bit tedious, there is a better way to do this with `(extern ...)` form. | ||
Eval expression may be modified to look like this: | ||
|
||
```scheme | ||
(begin | ||
(extern namespace: eval-example | ||
greeting) | ||
(displayln greeting)) | ||
``` | ||
|
||
Using `extern` will make `eval-example#greeting` available as `greeting` at runtime. But there is one more thing to do before it will work without errors, because `extern` relies on `:gerbil/expander` which should be loaded. In interactive mode this happens implicitly, while inside compiled binaries expander need to be loaded manually. Here is the final version: | ||
|
||
```scheme | ||
(import :gerbil/expander) | ||
(export main) | ||
(def greeting "hello world") | ||
(def (main . args) | ||
(gerbil-load-expander!) | ||
(eval '(begin | ||
(extern namespace: eval-example | ||
greeting) | ||
(displayln greeting)))) | ||
``` | ||
|
||
Which works as expected: | ||
|
||
```console | ||
$ gxc -exe eval-example.ss | ||
... | ||
$ ./eval-example | ||
hello world | ||
``` |