-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathtyping.ml
382 lines (365 loc) · 10.5 KB
/
typing.ml
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
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
open Builtin
open Type
open Error
open Syntax
open Global
let te_vars = ref []
let reset_te_vars () = te_vars := []
let bind_te_vars vars =
te_vars := [];
List.map (fun v ->
if List.mem_assoc v !te_vars then
failwith "bind_te_vars"
else (
let ty = new_global_type_var() in
te_vars := (v,ty) :: !te_vars;
ty
)
) vars
let rec should_generate expr =
let rec go expr =
match expr.e_desc with
| Pexpr_array [] -> true
| Pexpr_constant _ -> true
| Pexpr_constr(c,arg) ->
begin match arg with
| None -> true
| Some arg -> go arg
end
| Pexpr_function _ -> true
| Pexpr_ident _ -> true
| Pexpr_if(cond,ifso,ifnot) ->
begin match ifnot with
| None -> go ifso
| Some ifnot -> go ifso && go ifnot
end
| Pexpr_let(isrec,binds,body) ->
List.for_all (fun (_,e) -> should_generate e) binds &&
should_generate body
| Pexpr_sequence(e1,e2) -> go e2
| Pexpr_tuple es -> List.for_all should_generate es
| _ -> false
in
go expr
(* type of *)
let type_of_constant = function
| Const_char _ -> type_char
| Const_int _ -> type_int
| Const_float _ -> type_float
| Const_string _ -> type_string
let type_of_type_expression strict te =
let rec go te =
match te.te_desc with
| Ptype_arrow(te1,te2) ->
type_arrow (go te1) (go te2)
| Ptype_constr(id,params) ->
let cd =
try
find_type_desc id
with Not_found ->
unbound_type_constr_err te.te_loc id in
if List.length params <> cd.info.ty_arity then
type_arity_err te.te_loc cd params;
{ typ_desc=Tconstr(cd.info.ty_constr, List.map go params)
; typ_level=notgeneric }
| Ptype_tuple tes ->
type_product (List.map go tes)
| Ptype_var v ->
begin try
List.assoc v !te_vars
with Not_found ->
if strict then
unbound_type_var_err v te
else (
let ty = new_global_type_var() in
te_vars := (v,ty):: !te_vars;
ty
)
end
in
go te
(* unify *)
let rec unify ty1 ty2 =
let ty1' = type_repr ty1
and ty2' = type_repr ty2 in
if ty1' == ty2' then
()
else (
begin match ty1'.typ_desc, ty2'.typ_desc with
| Tvar link1, Tvar link2 ->
if ty1'.typ_level < ty2'.typ_level then (
ty2'.typ_level <- ty1'.typ_level;
link2 := Tlink ty1'
) else (
ty1'.typ_level <- ty2'.typ_level;
link1 := Tlink ty2'
)
| Tvar link, _ when not (check_occur ty1'.typ_level ty1' ty2') ->
link := Tlink ty2';
| _, Tvar link when not (check_occur ty2'.typ_level ty2' ty1') ->
link := Tlink ty1';
| Tarrow(t1x,t1y), Tarrow(t2x,t2y) ->
unify t1x t2x;
unify t1y t2y
| Tconstr(c1,tys1), Tconstr(c2,tys2)
when c1.info.ty_stamp = c2.info.ty_stamp ->
unify_list tys1 tys2
| Tconstr({info={ty_abbr=Tabbrev(args,body)}}, params), _ ->
unify (expand_abbrev body args params) ty2'
| _, Tconstr({info={ty_abbr=Tabbrev(args,body)}}, params) ->
unify ty1' (expand_abbrev body args params)
| Tproduct t1s, Tproduct t2s ->
unify_list t1s t2s
| _ ->
raise Unify
end;
type_repr ty1 |> ignore;
type_repr ty2 |> ignore;
)
and unify_list t1s t2s =
match t1s, t2s with
| [], [] -> ()
| t1::t1s, t2::t2s ->
unify t1 t2;
unify_list t1s t2s
| _ -> raise Unify
let unify_expr expr expect_ty actual_ty =
try
unify expect_ty actual_ty
with Unify ->
expr_wrong_type_err expr expect_ty actual_ty
let unify_pat pat expect_ty actual_ty =
try
unify expect_ty actual_ty
with Unify ->
pat_wrong_type_err pat expect_ty actual_ty
let rec typing_expr env expr =
match expr.e_desc with
| Pexpr_apply(e,es) ->
let ty_f = typing_expr env e in
List.fold_left (fun ty arg ->
let ty1, ty2 =
try
filter_arrow ty
with Unify ->
application_of_non_function_err expr ty_f
in
typing_expect env arg ty1;
ty2
) ty_f es
| Pexpr_array es ->
let ty = new_type_var() in
List.iter (fun e -> typing_expect env e ty) es;
type_array ty
| Pexpr_constant c ->
type_of_constant c
| Pexpr_constr(id,arg) ->
let cd =
try find_constr_desc id
with Not_found -> unbound_constr_err expr.e_loc id
in
begin match arg with
| None ->
begin match cd.info.cs_kind with
| Constr_constant ->
type_instance cd.info.cs_res
| _ ->
let ty1, ty2 = type_pair_instance cd.info.cs_arg cd.info.cs_res in
type_arrow ty1 ty2
end
| Some arg ->
begin match cd.info.cs_kind with
| Constr_constant ->
constant_constr_err expr.e_loc cd
| _ ->
let ty1, ty2 = type_pair_instance cd.info.cs_arg cd.info.cs_res in
typing_expect env arg ty1;
ty2
end
end
| Pexpr_constraint(e,te) ->
let ty = type_of_type_expression false te in
typing_expect (env) e ty;
ty
| Pexpr_for(name,start,stop,_,body) ->
typing_expect env start type_int;
typing_expect env stop type_int;
typing_stmt ((name,type_int)::env) body;
type_unit
| Pexpr_function mat ->
begin match mat with
| [] -> failwith "empty matching"
| (p,e)::_ as mat ->
let ty1 = new_type_var()
and ty2 = new_type_var() in
List.iter (fun (p,e) ->
typing_expect (typing_pat [] p ty1 @ env) e ty2
) mat;
type_arrow ty1 ty2
end;
| Pexpr_ident id ->
begin match id with
| Lident name ->
let t = type_instance (try
List.assoc name env
with Not_found ->
try
(find_value_desc id).info.v_typ
with Not_found ->
unbound_value_err expr.e_loc id
) in
t
| Ldot _ ->
try
type_instance (find_value_desc id).info.v_typ
with Not_found ->
unbound_value_err expr.e_loc id
end
| Pexpr_if(cond,ifso,ifnot) ->
typing_expect env cond type_bool;
begin match ifnot with
| None ->
typing_expect env ifso type_unit;
type_unit
| Some ifnot ->
let ty = typing_expr env ifso in
typing_expect env ifnot ty;
ty
end
| Pexpr_let(isrec,binds,body) ->
typing_expr (typing_let env isrec binds) body
| Pexpr_sequence(e1,e2) ->
typing_stmt env e1;
typing_expr env e2
| Pexpr_try(body,pes) ->
let ty = typing_expr env body in
List.iter (fun (p,e) ->
typing_expect (typing_pat [] p type_exn @ env) e ty
) pes;
ty
| Pexpr_tuple es ->
type_product (List.map (typing_expr env) es)
and typing_expect env e expect_ty =
match e.e_desc with
| Pexpr_let(isrec,binds,body) ->
typing_expect (typing_let env isrec binds) body expect_ty
| Pexpr_sequence(e1,e2) ->
typing_stmt env e1;
typing_expect env e2 expect_ty
| Pexpr_if(cond,ifso,ifnot) ->
typing_expect env cond type_bool;
typing_expect env ifso expect_ty;
begin match ifnot with
| None -> ()
| Some ifnot -> typing_expect env ifnot expect_ty
end
| Pexpr_tuple es ->
begin try
List.iter2 (typing_expect env) es
(filter_product (List.length es) expect_ty)
with Unify ->
unify_expr e expect_ty (typing_expr env e)
end;
| _ ->
unify_expr e expect_ty (typing_expr env e)
and typing_stmt env e =
let ty = typing_expr env e in
match (type_repr ty).typ_desc with
| Tvar _ -> ()
| Tarrow _ -> partial_apply_warn e.e_loc
| _ ->
if not (same_base_type ty type_unit) then
not_unit_type_warn e ty
and typing_let env isrec pes =
push_level();
let tys = List.map (fun _ -> new_type_var()) pes in
let env' = typing_pat_list [] (List.map fst pes) tys @ env in
List.iter2 (fun (p,e) ty ->
typing_expect (if isrec then env' else env) e ty
) pes tys;
pop_level();
let gens = List.map (fun (p,e) -> should_generate e) pes in
List.iter2 (fun gen ty ->
if gen then (
gen_type ty;
) else
value_restrict ty
) gens tys;
env'
and typing_pat penv pat ty =
match pat.p_desc with
| Ppat_alias(pat,a) ->
if List.mem_assoc a penv then
nonlinear_pattern_err pat a
else
typing_pat ((a,ty)::penv) pat ty
| Ppat_any ->
penv
| Ppat_array ps ->
let arity = List.length ps in
begin try
let ty_elem = filter_array arity ty in
let rec go penv = function
| [] -> penv
| p::ps ->
go (typing_pat penv p ty_elem) ps
in
go penv ps
with Unify ->
pat_wrong_type_err pat ty (new_type_var())
end
| Ppat_constant c ->
unify_pat pat ty (type_of_constant c);
penv
| Ppat_constr(id,arg) ->
let cd =
try find_constr_desc id
with Not_found -> unbound_constr_err pat.p_loc id
in
begin match arg with
| None ->
begin match cd.info.cs_kind with
| Constr_constant ->
unify_pat pat ty (type_instance cd.info.cs_res);
penv
| _ ->
nonconstant_constr_err pat.p_loc cd
end
| Some arg ->
begin match cd.info.cs_kind with
| Constr_constant ->
constant_constr_err pat.p_loc cd
| _ ->
let ty1, ty2 = type_pair_instance cd.info.cs_arg cd.info.cs_res in
unify_pat pat ty ty2;
typing_pat penv arg ty1
end
end
| Ppat_constraint(pat, te) ->
let ty' = type_of_type_expression false te in
let penv = typing_pat penv pat ty' in
unify_pat pat ty ty';
penv
| Ppat_or(p1,p2) ->
let penv = typing_pat penv p1 ty in
typing_pat penv p2 ty
| Ppat_tuple ps ->
let arity = List.length ps in
begin try
typing_pat_list penv ps (filter_product arity ty)
with Unify ->
pat_wrong_type_err pat ty (type_product (new_type_var_list arity))
end
| Ppat_var v ->
if assoc_mem v penv then
nonlinear_pattern_err pat v
else
(v,ty)::penv
and typing_pat_list penv ps tys =
match ps, tys with
| [], [] ->
penv
| p::ps, ty::tys ->
typing_pat_list (typing_pat penv p ty) ps tys
| _, _ ->
failwith "arity unmatch"