在本页中:
5.1 Building Mutators for GC2
allocator-setup
5.2 Mutator API for GC2
if
and
or
cond
case
define-values
let
let-values
let*
set!
quote
begin
define
lambda
λ
error
add1
sub1
zero?
+
-
*
/
even?
odd?
=
<
>
<=
>=
symbol?
symbol=?
number?
boolean?
empty?
eq?
cons
cons?
first
rest
set-first!
set-rest!
empty
print-only-errors
halt-on-errors
import-primitives
5.3 Testing Mutators with GC2
test/  location=?
test/  value=?
printf

5 GC Mutator Language, 2

 #lang plai/gc2/mutator package: plai-lib

The GC Mutator Scheme language is used to test garbage collectors written with the GC Collector Language, 2 language. Since collectors support a subset of Racket’s values, the GC Mutator Scheme language supports a subset of procedures and syntax. In addition, many procedures that can be written in the mutator are omitted as they make good test cases. Therefore, the mutator language provides only primitive procedures, such as +, cons, etc.

5.1 Building Mutators for GC2

The first expression of a mutator must be:

语法

(allocator-setup collector-module
                 heap-size)
 
heap-size = exact-nonnegative-integer
collector-module specifies the path to the garbage collector that the mutator should use. The collector must be written in the GC Collector Scheme language.

The rest of a mutator module is a sequence of definitions, expressions and test cases. The GC Mutator Scheme language transforms these definitions and statements to use the collector specified in allocator-setup. In particular, many of the primitive forms, such as cons map directly to procedures such as gc:cons, written in the collector.

5.2 Mutator API for GC2

The GC Mutator Scheme language supports the following procedures and syntactic forms:

语法

if

Just like Racket’s if.

语法

and

Just like Racket’s and.

语法

or

Just like Racket’s or.

语法

cond

Just like Racket’s cond.

语法

case

Just like Racket’s case.
Just like Racket’s define-values.

语法

let

Just like Racket’s let.
Just like Racket’s let-values.

语法

let*

Just like Racket’s let*.

语法

set!

Similar to Racket’s set!. Unlike Racket’s set!, this set! is syntactically allowed only in positions that discard its result, e.g., at the top-level or in a begin expression (although not as the last expression in a begin).

语法

quote

Just like Racket’s quote.

语法

begin

Just like Racket’s begin.

语法

(define (id arg-id ...) body-expression ...+)

Just like Racket’s define, except restricted to the simpler form above.

语法

(lambda (arg-id ...) body-expression ...+)

语法

(λ (arg-id ...) body-expression ...+)

Just like Racket’s lambda and λ, except restricted to the simpler form above.

Just like Racket’s error.
Just like Racket’s add1.
Just like Racket’s sub1.
Just like Racket’s zero?.

+ : procedure?

Just like Racket’s +.

- : procedure?

Just like Racket’s -.

* : procedure?

Just like Racket’s *.

/ : procedure?

Just like Racket’s /.
Just like Racket’s even?.
Just like Racket’s odd?.

= : procedure?

Just like Racket’s =.

< : procedure?

Just like Racket’s <.

> : procedure?

Just like Racket’s >.

<= : procedure?

Just like Racket’s <=.

>= : procedure?

Just like Racket’s >=.
Just like Racket’s symbol?.
Just like Racket’s symbol=?.
Just like Racket’s number?.
Just like Racket’s boolean?.
Just like Racket’s empty?.
Just like Racket’s eq?.

函数

(cons hd tl)  cons?

  hd : any/c
  tl : any/c
Constructs a (mutable) pair.

函数

(cons? v)  boolean?

  v : any/c
Returns #t when given a value created by cons, #f otherwise.

函数

(first c)  any/c

  c : cons?
Extracts the first component of c.

函数

(rest c)  any/c

  c : cons?
Extracts the rest component of c.

函数

(set-first! c v)  void

  c : cons?
  v : any/c
Sets the first of the cons cell c.

Calls to this function are allowed only in syntactic positions that would discard its result, e.g., at the top-level or inside a begin expression (but not in the last expression in a begin). Also, this function appear only in the function position of an application expression.

So, in order to pass around a version of this function, you must write something like this (λ (c v) (begin (set-first! c v) 42)), perhaps picking a different value than 42 as the result.

函数

(set-rest! c v)  void

  c : cons?
  v : any/c
Sets the rest of the cons cell c, with the same syntactic restrictions as set-first!.

语法

empty

The identifier empty is defined to invoke (gc:alloc-flat '()) wherever it is used.

Behaves like PLAI’s print-only-errors.

Behaves like PLAI’s halt-on-errors.

Other common procedures are left undefined as they can be defined in terms of the primitives and may be used to test collectors.

Additional procedures from scheme may be imported with:

语法

(import-primitives id ...)

Imports the procedures id ... from scheme. Each procedure is transformed to correctly interface with the mutator. That is, its arguments are dereferenced from the mutator’s heap and the result is allocated on the mutator’s heap. The arguments and result must be heap-value?s, even if the imported procedure accepts or produces structured data.

For example, the GC Mutator Scheme language does not define modulo:

(import-primitives modulo)
 
(test/value=? (modulo 5 3) 2)

5.3 Testing Mutators with GC2

GC Mutator Scheme provides two forms for testing mutators:

语法

(test/location=? mutator-expr1 mutator-expr2)

test/location=? succeeds if mutator-expr1 and mutator-expr2 reference the same location on the heap.

语法

(test/value=? mutator-expr datum/quoted)

test/value=? succeeds if mutator-expr and datum/expr are structurally equal. datum/quoted is not allocated on the mutator’s heap. Futhermore, it must either be a quoted value or a literal value.

语法

(printf format mutator-expr ...)

 
format = literal-string
In GC Mutator Scheme, printf is a syntactic form and not a procedure. The format string, format is not allocated on the mutator’s heap.