-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathinterp-Rfun.rkt
69 lines (61 loc) · 2.35 KB
/
interp-Rfun.rkt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
#lang racket
(require racket/fixnum)
(require "utilities.rkt")
(require "interp-Rvec.rkt")
(provide interp-Rfun interp-Rfun-class)
;; Note to maintainers of this code:
;; A copy of this interpreter is in the book and should be
;; kept in sync with this code.
(define interp-Rfun-class
(class interp-Rvec-class
(super-new)
(define/public (apply-fun fun-val arg-vals e)
(match fun-val
[`(function ,xs ,body ,fun-env)
(define params-args (for/list ([x xs] [arg arg-vals])
(cons x (box arg))))
(define new-env (append params-args fun-env))
((interp-exp new-env) body)]
[else (error 'interp-exp "expected function, not ~a\nin ~v"
fun-val e)]))
(define/override ((interp-exp env) e)
(define recur (interp-exp env))
(verbose "Rfun/interp-exp" e)
(match e
[(Var x) (unbox (dict-ref env x))]
[(Let x e body)
(define new-env (dict-set env x (box (recur e))))
((interp-exp new-env) body)]
[(Apply fun args)
(define fun-val (recur fun))
(define arg-vals (for/list ([e args]) (recur e)))
(apply-fun fun-val arg-vals e)]
[else ((super interp-exp env) e)]))
(define/public (interp-def d)
(match d
[(Def f (list `[,xs : ,ps] ...) rt _ body)
(cons f (box `(function ,xs ,body ())))]
))
(define/override (interp-program p)
(verbose "interp-Rfun" p)
(match p
[(ProgramDefsExp info ds body)
(let ([top-level (for/list ([d ds]) (interp-def d))])
(for/list ([f (in-dict-values top-level)])
(set-box! f (match (unbox f)
[`(function ,xs ,body ())
`(function ,xs ,body ,top-level)])))
((interp-exp top-level) body))]
;; For after the shrink pass.
[(ProgramDefs info ds)
(define top-level (for/list ([d ds]) (interp-def d)))
(for ([f (in-dict-values top-level)])
(set-box! f (match (unbox f)
[`(function ,xs ,body ())
`(function ,xs ,body ,top-level)])))
;; call the main function
((interp-exp top-level) (Apply (Var 'main) '()))]
))
))
(define (interp-Rfun p)
(send (new interp-Rfun-class) interp-program p))