-
Notifications
You must be signed in to change notification settings - Fork 1
/
type-check-Cif.rkt
214 lines (191 loc) · 7.36 KB
/
type-check-Cif.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
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
#lang racket
(require graph)
(require "multigraph.rkt")
(require "utilities.rkt")
(require "type-check-Cvar.rkt")
(require "type-check-Lif.rkt")
(provide type-check-Cif type-check-Cif-class type-check-Cif-mixin)
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; type-check-Cif
(define (type-check-Cif-mixin super-class)
(class super-class
(super-new)
(inherit check-type-equal?)
(define/override (type-equal? t1 t2)
(debug 'type-equal? "lenient" t1 t2)
(match* (t1 t2)
[('_ t2) #t]
[(t1 '_) #t]
[(other wise) (super type-equal? t1 t2)]))
(define/public (combine-types t1 t2)
(match (list t1 t2)
[(list '_ t2) t2]
[(list t1 '_) t1]
[else
t1]))
;; TODO: move some things from here to later type checkers
(define/public (free-vars-exp e)
(define (recur e) (send this free-vars-exp e))
(match e
[(Var x) (set x)]
[(Int n) (set)]
[(Bool b) (set)]
[(Let x e body)
(set-union (recur e) (set-remove (recur body) x))]
[(If cnd thn els)
(set-union (recur cnd) (recur thn) (recur els))]
[(Prim op es)
(apply set-union (cons (set) (map recur es)))]
[else (error 'free-vars-exp "unmatched ~a" e)]))
(field [type-changed #t])
(define/public (exp-ready? exp env)
(for/and ([x (free-vars-exp exp)])
(dict-has-key? env x)))
(define/public (update-type x t env)
(debug 'update-type x t)
(cond [(dict-has-key? env x)
(define old-t (dict-ref env x))
(unless (type-equal? t old-t)
(error 'update-type "old type ~a and new type ~ are inconsistent"
old-t t))
(define new-t (combine-types old-t t))
(cond [(not (equal? new-t old-t))
(dict-set! env x new-t)
(set! type-changed #t)])]
[(eq? t '_)
(void)]
[else
(set! type-changed #t)
(dict-set! env x t)]))
(define/override ((type-check-atm env) e)
(match e
[(Bool b) (values (Bool b) 'Boolean)]
[else
((super type-check-atm env) e)]
))
(define/override ((type-check-exp env) e)
(debug 'type-check-exp "Cif ~a" e)
(match e
[(Bool b) (values (Bool b) 'Boolean)]
[(Prim 'eq? (list e1 e2))
(define-values (e1^ T1) ((type-check-exp env) e1))
(define-values (e2^ T2) ((type-check-exp env) e2))
(check-type-equal? T1 T2 e)
(values (Prim 'eq? (list e1^ e2^)) 'Boolean)]
[else
((super type-check-exp env) e)]
))
(define/override (type-check-stmt env)
(lambda (s)
(debug 'type-check-stmt "Cwhile" s)
(match s
[(Assign (Var x) e)
#:when (exp-ready? e env)
(define-values (e^ t) ((type-check-exp env) e))
(update-type x t env)]
[(Assign (Var x) e) (void)]
[(Prim 'read '()) (void)]
[else (void)]
)))
(define/override ((type-check-tail env block-env blocks) t)
(debug 'type-check-tail "Cif" t)
(match t
[(Return e)
#:when (exp-ready? e env)
(define-values (e^ t) ((type-check-exp env) e))
t]
[(Return e) '_]
[(Seq s t)
((type-check-stmt env) s)
((type-check-tail env block-env blocks) t)]
[(Goto label)
(cond [(dict-has-key? block-env label)
(dict-ref block-env label)]
[else '_])]
[(IfStmt cnd tail1 tail2)
(cond [(exp-ready? cnd env)
(define-values (c Tc) ((type-check-exp env) cnd))
(unless (type-equal? Tc 'Boolean)
(error "type error: condition should be Boolean, not" Tc))
])
(define T1 ((type-check-tail env block-env blocks) tail1))
(define T2 ((type-check-tail env block-env blocks) tail2))
(unless (type-equal? T1 T2)
(error "type error: branches of if should have same type, not"
T1 T2))
(combine-types T1 T2)]
[else ((super type-check-tail env block-env blocks) t)]))
(define/public (adjacent-tail t)
(match t
[(Goto label) (set label)]
[(IfStmt cnd t1 t2) (set-union (adjacent-tail t1) (adjacent-tail t2))]
[(Seq s t) (adjacent-tail t)]
[else (set)]))
(define/public (C-blocks->CFG blocks)
(define G (make-multigraph '()))
(for ([label (in-dict-keys blocks)])
(add-vertex! G label))
(for ([(src b) (in-dict blocks)])
(for ([tgt (adjacent-tail b)])
(add-directed-edge! G src tgt)))
G)
;; Do the iterative dataflow analysis because of deadcode
;; in the un-optimized version of the compiler. -Jeremy
(define/public (type-check-blocks info blocks env start)
(define block-env (make-hash))
(set! type-changed #t)
(define (iterate)
(cond [type-changed
(set! type-changed #f)
(for ([(label tail) (in-dict blocks)])
(define t ((type-check-tail env block-env blocks) tail))
(update-type label t block-env)
)
(verbose "type-check-blocks" env block-env)
(iterate)]
[else (void)]))
(iterate)
(unless (dict-has-key? block-env start)
(error 'type-check-blocks "failed to infer type for ~a" start))
(define t (dict-ref block-env start))
(values env t))
(define/override (type-check-program p)
(match p
[(CProgram info blocks)
(define empty-env (make-hash))
(define-values (env t)
(type-check-blocks info blocks empty-env 'start))
(unless (type-equal? t 'Integer)
(error "return type of program must be Integer, not" t))
(define locals-types
(for/list ([(x t) (in-dict env)])
(cons x t)))
(define new-info (dict-set info 'locals-types locals-types))
(CProgram new-info blocks)]
[else (super type-check-program p)]))
#;(define/override (type-check-program p)
(match p
[(CProgram info blocks)
;; We override Cvar for this case to type check all the
;; blocks, even the unreachable ones. -Jeremy
(define env (make-hash))
(define block-env (make-hash))
;; First, type check the start block and make sure it's Integer
(define t
((type-check-tail env block-env blocks) (dict-ref blocks 'start)))
(unless (type-equal? t 'Integer)
(error "return type of program must be Integer, not" t))
;; Type check the unreachable blocks
(for ([(label block) (in-dict blocks)])
((type-check-tail env block-env blocks) block))
(define locals-types (for/list ([(x t) (in-dict env)])
(cons x t)))
(define new-info (dict-set info 'locals-types locals-types))
(CProgram new-info blocks)]
[else (error 'type-check-program "expected a C program, not ~a" p)]))
))
(define type-check-Cif-class (type-check-Cif-mixin
(type-check-Cvar-mixin
type-check-Lif-class)))
(define (type-check-Cif p)
(send (new type-check-Cif-class) type-check-program p))