Skip to content

Latest commit

 

History

History
208 lines (160 loc) · 7.07 KB

README.md

File metadata and controls

208 lines (160 loc) · 7.07 KB

miniVerse (and microVerse)

Note: this code has been extracted from the gregr/experiments GitHub repository to make it into one single self-contained repository.

This is an implementation of VC based on the paper: The Verse Calculus: a Core Calculus for Functional Logic Programming

This implementation uses a naive sequence-concurrent evaluation strategy that should be complete with respect to the rewrite semantics from the paper. Runnable examples from the paper are included.

micro.rkt provides a small-step interpreter for a low-level VC notation. example-micro.rkt contains examples written with the micro notation.

mini.rkt builds on micro, providing some syntactic sugar for writing VC programs, as well as a small base library. example-mini.rkt contains examples written with the mini notation.

Getting started

  • Download Racket

  • Verify that everything is working (no obvious errors or crashes) by running the examples found in this directory. You can run them from the command line like this:

    • racket example-micro.rkt
    • racket example-mini.rkt

Running miniVerse programs

While writing a Racket program, or while running Racket interactively, (require "mini.rkt") to import procedures for running miniVerse programs.

The most convenient procedures to use are these:

  • (run* PROGRAM) will run PROGRAM until it produces a value or gets stuck. If PROGRAM diverges or loops forever, run* may not terminate.
  • (run STEP-COUNT PROGRAM) is like (run* PROGRAM), except it only performs STEP-COUNT steps of execution. If this is not enough to produce a value, the current execution state will be returned. run should always terminate.
  • (trace-run* PROGRAM) is like (run* PROGRAM), except it produces a trace of all the intermediate execution states of the program being run. These states may be quite verbose, even for small programs.
  • (trace-run STEP-COUNT PROGRAM) is the tracing variant of (run STEP-COUNT PROGRAM).

Here is an example of using run* to run miniVerse programs within Racket's interactive environment:

> racket
Welcome to Racket v8.4 [cs].

> (require "mini.rkt")

> (run* #t)
(value #t)

> (run* '(cons 'a 'b))
(value (a . b))

> (run* '(all (exist (X)
                (== X (alt 1 2))
                (cons X X))))
(value #((1 . 1) (2 . 2)))

> (run* '(exist (first)
           (== first (lambda (x)
                       (exist (a b) (== x (cons a b)) a)))
           (exist (x y)
             (== x (cons y 5))
             (== (first x) 2)
             y)))
(value 2)

> (run* '(all (exist (x y)
                (== x (alt 3 4))
                (== y (alt 20 30))
                (cons x y))))
(value #((3 . 20) (3 . 30) (4 . 20) (4 . 30)))

> (run* '(all (exist (x)
                (== (vector-ref (vector 2 3 2 7 9) x) 2)
                x)))
(value #(0 2))

> (run* '(all (exist (append)
                (== append
                    (lambda (xs ys)
                      (alt
                        (begin (== xs '()) ys)
                        (exist (x xrest)
                          (== xs (cons x xrest))
                          (cons x (append xrest ys))))))
                (exist (as bs)
                  (== (append as bs) (list 1 2 3))
                  (list as bs)))))
(value #((() (1 2 3)) ((1) (2 3)) ((1 2) (3)) ((1 2 3) ())))

Running microVerse programs

There is no convenient interface provided for running microVerse programs yet. For now, take a look at the scaffolding in example-micro.rkt to see one possible way to run them.

One reason you might want to run microVerse programs directly, at least while experimenting, is because their execution states are currently much less verbose than those of the corresponding miniVerse programs, making them easier to follow. (This can be fixed for miniVerse programs. See the TODO list.)

miniVerse grammar

CONSTANT ::= <scheme-value>
NAME     ::= <symbol>

;; NOTE: operator keywords may be shadowed by lambda and exist bindings
E ::= (quote CONSTANT)
    | (lambda (NAME ...) E E ...)  ; body is an implicit begin
    | (exist (NAME ...) E E ...)   ; body is an implicit begin
    | (== E E)
    | (begin E E ...) ; n-ary sequence
    | (alt E ...)     ; empty alt produces a "fail"
    | (one E E ...)   ; body is an implicit begin
    | (all E E ...)   ; body is an implicit begin
    | (if/exist (NAME ...) E E E)  ; wraps an exist scope around the condition and consequent
    | (for/exist (NAME ...) E E)   ; wraps an exist scope around the iterator and body
    | (if E E E)
    | (for E E)
    | (E.proc E ...)  ; n-ary application (if E.proc is not an unshadowed operator keyword)
    | NAME            ; variable reference

The initial miniVerse environment also defines these procedures:

;; unary predicates
null? boolean? pair? number? symbol? string? vector? procedure?

(vector E ...)
(vector-ref E.vector E.index)
(vector-length E.vector)

(list E ...)
(cons E E)
(car E)
(cdr E)

;; 2-ary predicates
< <= > >=

;; 2-ary arithmetic
+ - * /

;; These correspond to tuple operator definitions from the VC paper:
vhead vtail vcons vmap

microVerse grammar

CONSTANT ::= <scheme-value>
NAME     ::= <scheme-value>

OPNAME ::= number? | symbol? | string? | vector? | procedure?
         | vector-lengtho  ; (op vector-lengtho E.vector E.output)
         | vector-refo     ; (op vector-refo E.vector E.index E.output)
         | cons            ; 2-ary pair constructor
         | < | <=          ; 2-ary operators
         | +o | *o         ; 3-ary arithmetic

E ::= (value CONSTANT)
    | (ref NAME)
    | (lam NAME E)
    | (exist (NAME ...) E)
    | (== E E)
    | (op OPNAME E ...)
    | (app E E)
    | (seq E E)
    | (alt E E)
    | (one E)
    | (all E)

Miscellaneous notes

Scheme vectors assume the role of VC tuples in this implementation. VC tuple indexing corresponds to miniVerse's vector-ref, which can also be run "backwards".

Existential scope is tree-structured in this implementation, so if and for have explicit scope-introducing counterparts if/exist and for/exist that can be used to recover the scoping behavior described in the VC paper.

No evaluation will be performed under lambda.

Arithmetic operators can be run "backwards" in some special cases, but there is no general arithmetic solver.

TODO

  • Make tracing less verbose by eliminating unused dependencies in states

    • garbage collection (we currently do none)
    • safe-for-space closures that only capture actual dependencies, rather than everything they see
  • Experiment with different kinds of IO

    • immediate read/write on a channel: sort of like unsafePerformIO
    • scoped IO: an IO handler that receives and services IO request descriptions, sort of like the IO monad
    • transactional IO: a (mozart/oz-style) single-assignment stream of IO requests is gradually assigned to by concurrent processes
  • Try more sophisticated evaluation strategies for better performance