Intermediate Language

BiwaScheme is a VM based compiler-interpreter. Scheme program is translated into IL (Intermediate Language) by BiwaScheme.Compiler, and then executed by BiwaScheme.Interpreter.

Most of the opecodes are taken from 3imp.pdf.

Registers:

You can see how these registers work on /test/tracer.html.

Opecodes

op_constant

Push val onto the stack. Often used with op_argument.

format

["constant", <value>, <opecodes_next>]

registers

example

program:

(display "hello")

compiled:

[frame
   [constant "hello"
   [argument
   [constant 1
   [argument
   [refer-global "display"
   [apply]]]]]]
[halt]]

op_argument

Push a value (typically an argument of function call) onto the stack.

format

["argument", <opecodes_next>]

registers

example

program:

(print "hi")

compiled:

[frame
   [constant "hi"
   [argument
   [constant 1
   [argument
   [refer-global "print"
   [apply]]]]]]
[halt]]

op_test

Execute either of two operations depending on the value of 'a' register.

format

["test", <opecodes_then>, <opecodes_else>]

example

program:

(if 7 8 9)

compiled:

[constant 7
[test
   [constant 8
      [halt]]
   [constant 9
      [halt]]]]

op_close

Create Scheme closure object (which is represented with JS Array).

format

["close", <args>, <n>, <body>, <opecodes_next>, <dotpos>]

TODO: <opecodes_next> should be the last in consistency with other opecodes

description

op_close creates closure object onto the 'a' register.

closure object

Scheme closures are represented by JS arrays with its attribute 'closure_p' assigned to true.

[body, freevar1, freevar2, ..., dotpos]

example

program:

(lambda () 1)

compiled:

[close 
  0
  [constant 1
    [return]]
  [halt]
  -1]

example of reassigned freevars

Usually closure object contains actual values of the freevars:

(define (f a b)
  (lambda () a b))
(f 11 22)

-> [ ["refer-free", 0, ["refer-free", 1, ["return"]]], 11, 22, -1]

But when the freevar may reassigned by set!, a box (JS array contains the actual value) is stored. It will be unboxed by op_indirect (see also: op_indirect, op_box)

(define (f a b)
  (set! a 99)
  (lambda () a b))
(f 11 22)

-> [ ["refer-free", 0, ["indirect", ["refer-free", 1, ["return"]]]], [99], 22, -1]

op_conti

Capture the current continuation.

Captured continuation is just a Scheme closure which includes whole copy of current stack and op_nuate1 to restore it (see also: op_nuate1).

format

["conti", <n>, <opecodes_next>]

n: number of args(for outer lambda) to remove (= 0 unless tail position)

example

program:

(call/cc func)

compiled:

[frame
   [conti 0
   [argument
   [constant 1
   [argument
   [refer-global "func"
   [apply]]]]]]
[halt]]

op_nuate1

On invoking a continuation, call winders defined by dynamic-wind.

format

["nuate1", <saved_stack>, <dynamic_winder>]

description

op_nuate1 takes saved stack as an argument. This is just passed to op_nuate2.

The second argument is the winder on this continuation is captured.

op_nuate1 is never generated by the Compiler. It is dynamically generated by Interpreter when processing op_conti.

example

program:

(define cc (call/cc identity))
(cc 123)

compiled (excerpt):

[nuate1 #<Object{stack,last_refer,call_stack,tco_counter}> #<DynamicWind>]

op_nuate2

On invoking a continuation, restore the stack saved by op_conti (see op_conti).

format

nuate2 is paired by refer-local which restores the register a.

["refer-local", 0,
["nuate2", <saved_stack>]]

description

op_nuate2 takes saved stack as an argument. When op_nuate2 is invoked, the interpreter stack is replaced by this saved stack.

op_nuate2 is never generated by the Compiler. It is dynamically generated by Interpreter when processing op_nuate1.

example

program:

(define cc (call/cc identity))
(cc 123)

compiled (excerpt):

[nuate2 #<Object{stack,last_refer,call_stack,tco_counter}>]

op_halt

Indicate end of program.

A program may have more than one op_halt (eg. if is compiled into op_test which contains then-clause and else-clause, both have op_halt on its end.)

example

program:

'()

compiled:

[constant ()
  [halt]]

op_frame

Push stack frame.

op_frame must be called before applying a function, except when it is a tail call (see also: op_shift.)

format

["frame", <opecodes_next>, <opecodes_after>]

description

op_frame pushes a frame (i.e. the following values) onto stack:

Usually this frame is popped by return.

op_frame is generated by:

example

program:

(print 11)

compiled:

[frame        // op_frame
   [constant 11          // push arguments to stack           
   [argument
   [constant 1           // push number of arguments 
   [argument
   [refer-global "print" // load function to 'a' register
   [apply]]]]]]          // invoke function
[halt]]

Note that op_frame itself does not touch 'f' register. Since function arguments are evaluated between op_frame and apply and these evaluations must be done in the current frame, 'f' register is updated by apply.

op_apply

Invoke a function-like object, i.e. Scheme closure or BiwaScheme library JS function.

(Use js-call for native JS functions.)

format

["apply"]

description

In contrast to other opecodes, op_apply takes no arguments. Instead it retrieves needed information from the stack.

Before calling op_apply, you must put these values onto stack:

applicable things

Note that you cannot invoke pure JS functions directly. You can use js-call or js-invoke instead.

example

program:

(+ 11 22)

compiled:

[frame
   [constant 22  
   [argument          // push 22 (second argument) to stack
   [constant 11  
   [argument          // push 11 (first argument) to stack
   [constant 2   
   [argument          // push 2 (number of arguments) to stack
   [refer-global "+"  // load function '+' to 'a' register
   [apply]]]]]]]]     // invoke the function with arguments 11 and 22
[halt]]

op_tco_hinted_apply

Same as op_apply, but used for tail calls. Added for stack trace.

format

["tco_hinted_apply"]

description

See op_apply for how it retrieves the function and its arguments.

Stack trace

BiwaScheme.Interpreter has three variables to handle stack trace:

Here is the difference between normal call and tail call.

Length of call_stack is capped to Interpreter#max_trace_size (which defaults to BiwaScheme.max_trace_size). We do not cap length of tco_counter because it does not grow with tail calls.

example

program:

(define (f) (print 7))

compiled:

[close 0                      // Create a Scheme closure
   [constant 7                // Body of f
   [argument
   [constant 1
   [argument
   [refer-global "print"
   [shift 1
   [tco_hinted_apply]]]]]]]   // Invoke print with tco_hinted_apply
   [assign-global "f"
   [halt]]
-1]

op_return

Pop stack frame.

op_return is generated by (lambda ...) and must be called as the last opecode of function execution.

format

["return"]

op_return does not have because it pops next opecodes from stack.

description

op_return pops the following values from stack:

op_return also sets x, f, c to register 'x', 'f', 'c'.

example

program:

((lambda () 11))

compiled:

[frame
   [constant 0
   [argument
   [close 0
      [constant 11
      [return]]
      [apply]
   -1]]]
[halt]]

op_shift

Discard the arguments for parent call from stack.

Since stack frame is not pushed by op_frame and popped by op_return for tail calls, we need to remove old arguments with op_shift.

format

["shift", <n_args>, <opecodes_next>]

description

op_shift removes arguments for parent call from stack. Suppose the following code:

(define (foo x y)
  (bar "a" "b" "c"))
(foo "x" "y")

Before applying bar, op_shift removes the following values. Note that first four elements will not be popped - that's why this operation is named "shift".

Also note that stack frame is not pushed by op_frame for tail calls.

example

program:

((lambda () (print 1))

Here (print 1) is a tail call.

compiled:

[frame
   [constant 0
   [argument
   [close 0  // make a closure, and
      [constant 1
      [argument
      [constant 1
      [argument
      [refer-global "print"
      [shift 1
      [apply]]]]]]]
      [apply]  // immediately invoke it
   -1]]]
[halt]]

Without tail call optimization, this program is compiled to:

[frame
   [constant 0
   [argument
   [close 0
      [frame  // push stack frame
         [constant 1
         [argument
         [constant 1
         [argument
         [refer-global "print"
         [apply]]]]]]
      [return]]  // pop stack frame
      [apply]
   -1]]]
[halt]]

op_refer-local

Load the value of a local variable.

format

["refer-local", <variable_pos>, <opecodes_next>]

example

program:

(define a 99)

compiled:

[frame
   [constant 7
   [argument
   [constant 1
   [argument
   [close 0
      [refer-local 0
         [return]]
      [apply]
      -1]]]]]
[halt]]

op_refer-free

Load the value of a free variable.

format

["refer-free", <variable_pos>, <opecodes_next>]

example

program:

(let ((a "a")(b "b"))
  (let ()
    a)

compiled:

[frame
   [constant "a"
   [argument
   [constant "b"
   [argument
   [constant 2
   [argument
   [close 0
      [constant 0
      [argument
      [refer-local 1
      [argument
      [close 1
         [refer-free 0
         [return]]
         [shift 0
         [apply]]
      -1]]]]]
      [apply]
   -1]]]]]]]
halt]

op_refer-global

Load the value of a certain global variable.

example

program:

map

compiled:

[refer-global "map"
[halt]]

op_assign-local

Assign value to a variable.

format

["assign-local", <FIXME>, <opecodes_next>]

example

See example of op_indirect.

op_assign-free

Assign a value to a free variable.

format

["assign-free", <FIXME>, <opecodes_next>]

example

program:

(let ((a "a"))
  (let ()
    (set! a "aa")))

compiled:

[frame
   [constant "a"
   [argument
   [constant 1
   [argument
   [close 0
      [box 0
      [constant 0
      [argument
      [refer-local 0
      [argument
      [close 1
         [constant "aa"
         [assign-free 0
         [return]]]
         [shift 0
         [apply]]
      -1]]]]]]
      [apply]
   -1]]]]]
halt]

op_assign-global

Assign a value to a global variable.

format

["assign-global", <str_variable_name>, <opecodes_next>]

example

program:

(define a 99)

compiled:

[constant 99
[assign-global "a"
[halt]]]

op_box

Make a box for a variable which may be reassigned by set! (see op_indirect).

format

["box", <FIXME>, <opecodes_next>]

example

See example of op_indirect.

op_indirect

is used to load the value of a variable which may reassigned by 'set!'.

format

["indirect", <opecodes_next>]

example

program:

(let ((a "a"))
  (set! a "b")
  a)

compiled:

[frame
   [constant "a"
   [argument
   [constant 1
   [argument
   [close 0
      [box 0
      [constant "b"
      [assign-local 0
      [refer-local 0
      [indirect
      [return]]]]]]
      [apply]
   -1]]]]]
[halt]]