Skip to content

Commit 23b6a1f

Browse files
committedNov 17, 2014
start adding tests to juliatypes.jl
the subtype algorithm is currently tweaked to reproduce the behavior of F_<:. running non_terminating() demonstrates execution on the non-terminating instance from the paper.
1 parent cdc4225 commit 23b6a1f

File tree

1 file changed

+112
-17
lines changed

1 file changed

+112
-17
lines changed
 

‎examples/juliatypes.jl

+112-17
Original file line numberDiff line numberDiff line change
@@ -27,30 +27,42 @@ end
2727

2828
type Var
2929
name::Symbol
30-
lb::Ty
31-
ub::Ty
30+
lb
31+
ub
3232
Var(n, lb=BottomT, ub=AnyT) = new(n, lb, ub)
3333
end
3434

35-
function show(io::IO, v::Var)
35+
function show_var_bounds(io::IO, v::Var)
3636
if v.lb !== BottomT
3737
show(io, v.lb)
3838
print(io, "<:")
3939
end
40-
print(io, v.name, "<:")
41-
show(io, v.ub)
40+
print(io, v.name)
41+
if v.ub !== AnyT
42+
print(io, "<:")
43+
show(io, v.ub)
44+
end
4245
end
4346

47+
show(io::IO, v::Var) = print(io, v.name)
48+
4449
type ForAllT <: Ty
4550
var::Var
46-
T::Ty
51+
T
52+
ForAllT(v::Var, t) = new(v, t)
53+
ForAllT(v::Var, t::Type) = new(v, convert(Ty, t))
4754
end
4855

4956
function show(io::IO, x::ForAllT)
50-
print(io, "@UnionAll ")
51-
show(io, x.var)
52-
print(io, " ")
57+
if x.T === x.var && x.var.ub !== AnyT
58+
print(io, "¬", x.var.ub)
59+
return
60+
end
61+
print(io, "(∀ ")
62+
show_var_bounds(io, x.var)
63+
print(io, ". ")
5364
show(io, x.T)
65+
print(io, ")")
5466
end
5567

5668
AnyT = TagT(TypeName(:Any), ())
@@ -62,7 +74,7 @@ inst(typename::TypeName, params...) = TagT(typename, params)
6274

6375
inst(t::TagT) = t
6476

65-
inst(t::ForAllT, param) = subst(t.T, Dict(t.var => param))
77+
inst(t::ForAllT, param) = subst(t.T, Dict{Any,Any}(t.var => param))
6678
inst(t::ForAllT, param, rest...) = inst(inst(t,param), rest...)
6779

6880
super(t::TagT) = inst(t.name.super, t.params...)
@@ -85,6 +97,8 @@ tupletype(xs...) = inst(TupleName, xs...)
8597

8698
isequal_type(x::Ty, y::Ty) = issub(x, y, true)
8799

100+
issub(x, y, env, inv) = (x === y)
101+
88102
function issub(x::Ty, y::Ty, inv::Bool = false)
89103
env = Dict()
90104
ans = issub(x, y, env, inv)
@@ -200,7 +214,22 @@ function add_constraint!(env, var, c::SupConstraint)
200214
end
201215
end
202216

217+
ctr = 1
218+
rename(x) = x
219+
function rename(x::ForAllT)
220+
global ctr
221+
v = Var(symbol(string("a",ctr)), x.var.lb, rename(x.var.ub))
222+
ctr += 1
223+
ForAllT(v, rename(inst(x, v)))
224+
end
225+
203226
function issub_var(a::Var, b, env, invariant)
227+
if b === AnyT
228+
return true
229+
end
230+
if haskey(env, a) && !invariant
231+
return issub(rename(env[a][1].rhs), b, env, invariant)
232+
end
204233
add_constraint!(env, a, invariant ? EqConstraint(b) : SubConstraint(b))
205234
return true
206235
end
@@ -220,17 +249,30 @@ function issub(a::Ty, b::Var, env, invariant)
220249
end
221250

222251
function issub(a::ForAllT, b::ForAllT, env, invariant)
252+
println(a, " <: ", b); sleep(.5)
253+
223254
# 1. handle bounds
224-
# this could be the crucial bit: we make the bounds covariant
225-
if !(issub(a.var.ub, b.var.ub, env, false) &&
226-
issub(b.var.lb, a.var.lb, env, false))
255+
if !(issub(b.var.ub, a.var.ub, copy(env), false) &&
256+
issub(a.var.lb, b.var.lb, copy(env), false))
227257
return false
228258
end
259+
229260
# 2. handle expression
230-
fresh = Var(symbol(string("_",a.var.name,"_",b.var.name)), a.var.lb, a.var.ub)
231-
add_constraint!(env, fresh, SupConstraint(a.var.lb))
232-
add_constraint!(env, fresh, SubConstraint(a.var.ub))
233-
inner = issub(inst(a, fresh), inst(b, fresh), env, invariant)
261+
# for contravariance, fresh<:b.var.ub, for covariance fresh<:a.var.ub
262+
lb = b.var.lb
263+
ub = b.var.ub
264+
#var = Var(symbol(string("a",ctr)), lb, ub)
265+
#global ctr += 1
266+
var = b.var
267+
268+
env = copy(env)
269+
#add_constraint!(env, var, SupConstraint(lb))
270+
add_constraint!(env, var, SubConstraint(ub))
271+
272+
a = inst(a, var)
273+
b = b.T
274+
275+
inner = issub(a, b, env, invariant)
234276
return inner
235277
end
236278

@@ -302,6 +344,8 @@ const tndict = ObjectIdDict()
302344

303345
xlate(t) = xlate(t, ObjectIdDict())
304346

347+
xlate(t, env) = t
348+
305349
function xlate(t::UnionType, env)
306350
if t === Union()
307351
return BottomT
@@ -365,3 +409,54 @@ ArrayT =
365409
let ArrayName = TypeName(:Array, @UnionAll T @UnionAll N inst(AbstractArrayT, T, N))
366410
@UnionAll T @UnionAll N inst(ArrayName, T, N)
367411
end
412+
413+
function non_terminating()
414+
# undecidable F_<: instance
415+
¬T = @UnionAll α<:T α
416+
417+
θ = @UnionAll α ¬(@UnionAll β< ¬β)
Has conversations. Original line has conversations.
418+
419+
a0 = Var(:a0)
420+
env = Dict{Any,Any}(a0 => Any[SubConstraint(θ)])
421+
422+
issub(a0, (@UnionAll a1<:a0 ¬a1), env, false)
423+
end
424+
425+
using Base.Test
426+
427+
issub_strict(x,y) = issub(x,y) && !issub(y,x)
428+
429+
# level 1: no varags, union, UnionAll
430+
function test_1()
431+
@test issub_strict(Int, Integer)
432+
@test issub_strict(Array{Int,1}, AbstractArray{Int,1})
433+
434+
@test isequal_type(Int, Int)
435+
@test isequal_type(Integer, Integer)
436+
@test isequal_type(Array{Int,1}, Array{Int,1})
437+
@test isequal_type(AbstractArray{Int,1}, AbstractArray{Int,1})
438+
439+
@test issub_strict((Int,Int), (Integer,Integer))
440+
@test issub_strict((Array{Int,1},), (AbstractArray{Int,1},))
441+
442+
@test isequal_type((Integer,Integer), (Integer,Integer))
443+
444+
@test !issub((Int,Int), (Int,))
445+
@test !issub((Int,), (Integer,Integer))
446+
end
447+
448+
# level 2: varargs
449+
function test_2()
450+
@test issub_strict((Int,Int), (Int...,))
451+
@test issub_strict((Int,Int), (Int,Int...,))
452+
@test issub_strict((Int,Int), (Int,Integer...,))
453+
@test issub_strict((Int,Int), (Int,Int,Integer...,))
454+
@test issub_strict((Int,Int...), (Int...,))
455+
@test issub_strict((Int,Int,Int...), (Int...,))
456+
@test issub_strict((Int,Int,Int...), (Integer,Int...,))
457+
@test issub_strict((Int...,), (Any...,))
458+
@test issub_strict((), (Any...,))
459+
460+
@test isequal_type((Int...,), (Int...,))
461+
@test isequal_type((Integer...,), (Integer...,))
462+
end

0 commit comments

Comments
 (0)
Please sign in to comment.