Skip to content

Commit 216d19a

Browse files
committed
implement comprehensions by calling collect on a Generator
fix #4470 Generalized comprehension syntax fix #7258 type-inference-independent comprehensions [ci skip]
1 parent e139f30 commit 216d19a

10 files changed

+79
-260
lines changed

base/essentials.jl

+14
Original file line numberDiff line numberDiff line change
@@ -193,3 +193,17 @@ const (:) = Colon()
193193
# For passing constants through type inference
194194
immutable Val{T}
195195
end
196+
197+
immutable Generator{F,I}
198+
f::F
199+
iter::I
200+
end
201+
202+
start(g::Generator) = start(g.iter)
203+
done(g::Generator, s) = done(g.iter, s)
204+
function next(g::Generator, s)
205+
v, s2 = next(g.iter, s)
206+
g.f(v), s2
207+
end
208+
209+
collect(g::Generator) = map(g.f, g.iter)

base/inference.jl

+3-74
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,6 @@ type StaticVarInfo
2525
gensym_types::Array{Any,1} # types of the GenSym's in this function
2626
vinfo::Array{Any,1} # variable properties
2727
label_counter::Int # index of the current highest label for this function
28-
fedbackvars::ObjectIdDict
2928
mod::Module
3029
end
3130

@@ -53,7 +52,7 @@ function StaticVarInfo(linfo::LambdaStaticData, ast=linfo.ast)
5352
else
5453
sp = svec()
5554
end
56-
StaticVarInfo(sp, vars, gensym_types, vinflist, nl, ObjectIdDict(), linfo.module)
55+
StaticVarInfo(sp, vars, gensym_types, vinflist, nl, linfo.module)
5756
end
5857

5958
type VarState
@@ -1070,8 +1069,7 @@ function abstract_eval(e::ANY, vtypes, sv::StaticVarInfo)
10701069
return abstract_eval_constant(e)
10711070
end
10721071
e = e::Expr
1073-
# handle:
1074-
# call null new & static_typeof
1072+
# handle: call null new &
10751073
if is(e.head,:call)
10761074
t = abstract_eval_call(e, vtypes, sv)
10771075
elseif is(e.head,:null)
@@ -1089,40 +1087,6 @@ function abstract_eval(e::ANY, vtypes, sv::StaticVarInfo)
10891087
elseif is(e.head,:&)
10901088
abstract_eval(e.args[1], vtypes, sv)
10911089
t = Any
1092-
elseif is(e.head,:static_typeof)
1093-
var = e.args[1]
1094-
t = abstract_eval(var, vtypes, sv)
1095-
if isa(t,DataType) && typeseq(t,t.name.primary)
1096-
# remove unnecessary typevars
1097-
t = t.name.primary
1098-
end
1099-
if is(t,Bottom)
1100-
# if we haven't gotten fed-back type info yet, return Bottom. otherwise
1101-
# Bottom is the actual type of the variable, so return Type{Bottom}.
1102-
if haskey(sv.fedbackvars, var)
1103-
t = Type{Bottom}
1104-
end
1105-
elseif isleaftype(t)
1106-
t = Type{t}
1107-
elseif isleaftype(inference_stack.types)
1108-
if isa(t,TypeVar)
1109-
t = Type{t.ub}
1110-
else
1111-
t = Type{t}
1112-
end
1113-
else
1114-
# if there is any type uncertainty in the arguments, we are
1115-
# effectively predicting what static_typeof will say when
1116-
# the function is compiled with actual arguments. in that case
1117-
# abstract types yield Type{<:T} instead of Type{T}.
1118-
# this doesn't really model the situation perfectly, but
1119-
# "isleaftype(inference_stack.types)" should be good enough.
1120-
if isa(t,TypeVar)
1121-
t = Type{t}
1122-
else
1123-
t = Type{TypeVar(:_,t)}
1124-
end
1125-
end
11261090
elseif is(e.head,:method)
11271091
t = (length(e.args) == 1) ? Any : Void
11281092
elseif is(e.head,:copyast)
@@ -1678,10 +1642,6 @@ function typeinf_uncached(linfo::LambdaStaticData, atypes::ANY, sparams::SimpleV
16781642
recpts = IntSet() # statements that depend recursively on our value
16791643
W = IntSet()
16801644

1681-
@label typeinf_top
1682-
1683-
typegotoredo = false
1684-
16851645
# exception handlers
16861646
cur_hand = ()
16871647
handler_at = Any[ () for i=1:n ]
@@ -1753,24 +1713,6 @@ function typeinf_uncached(linfo::LambdaStaticData, atypes::ANY, sparams::SimpleV
17531713
s[l] = stupdate(s[l], changes, vars)
17541714
end
17551715
end
1756-
elseif is(hd,:type_goto)
1757-
for i = 2:length(stmt.args)
1758-
var = stmt.args[i]::GenSym
1759-
# Store types that need to be fed back via type_goto
1760-
# in gensym_init. After finishing inference, if any
1761-
# of these types changed, start over with the fed-back
1762-
# types known from the beginning.
1763-
# See issue #3821 (using !typeseq instead of !subtype),
1764-
# and issue #7810.
1765-
id = var.id+1
1766-
vt = gensym_types[id]
1767-
ot = gensym_init[id]
1768-
if ot===NF || !typeseq(vt,ot)
1769-
gensym_init[id] = vt
1770-
typegotoredo = true
1771-
end
1772-
sv.fedbackvars[var] = true
1773-
end
17741716
elseif is(hd,:return)
17751717
pc´ = n+1
17761718
rt = abstract_eval(stmt.args[1], s[pc], sv)
@@ -1835,16 +1777,6 @@ function typeinf_uncached(linfo::LambdaStaticData, atypes::ANY, sparams::SimpleV
18351777
end
18361778
end
18371779

1838-
if typegotoredo
1839-
# if any type_gotos changed, clear state and restart.
1840-
for ll = 2:length(s)
1841-
s[ll] = ()
1842-
end
1843-
empty!(W)
1844-
gensym_types[:] = gensym_init
1845-
frame.result = curtype
1846-
@goto typeinf_top
1847-
end
18481780
for i = 1:length(gensym_types)
18491781
if gensym_types[i] === NF
18501782
gensym_types[i] = Union{}
@@ -1931,7 +1863,7 @@ function eval_annotate(e::ANY, vtypes::ANY, sv::StaticVarInfo, decls, undefs)
19311863

19321864
e = e::Expr
19331865
head = e.head
1934-
if is(head,:static_typeof) || is(head,:line) || is(head,:const)
1866+
if is(head,:line) || is(head,:const)
19351867
return e
19361868
#elseif is(head,:gotoifnot) || is(head,:return)
19371869
# e.typ = Any
@@ -2168,9 +2100,6 @@ function effect_free(e::ANY, sv, allow_volatile::Bool)
21682100
end
21692101
if isa(e,Expr)
21702102
e = e::Expr
2171-
if e.head === :static_typeof
2172-
return true
2173-
end
21742103
ea = e.args
21752104
if e.head === :call
21762105
if is_known_call_p(e, is_pure_builtin, sv)

src/alloc.c

+1-2
Original file line numberDiff line numberDiff line change
@@ -87,12 +87,11 @@ jl_sym_t *null_sym; jl_sym_t *body_sym;
8787
jl_sym_t *method_sym;
8888
jl_sym_t *enter_sym; jl_sym_t *leave_sym;
8989
jl_sym_t *exc_sym; jl_sym_t *error_sym;
90-
jl_sym_t *static_typeof_sym;
9190
jl_sym_t *new_sym; jl_sym_t *using_sym;
9291
jl_sym_t *const_sym; jl_sym_t *thunk_sym;
9392
jl_sym_t *anonymous_sym; jl_sym_t *underscore_sym;
9493
jl_sym_t *abstracttype_sym; jl_sym_t *bitstype_sym;
95-
jl_sym_t *compositetype_sym; jl_sym_t *type_goto_sym;
94+
jl_sym_t *compositetype_sym;
9695
jl_sym_t *global_sym; jl_sym_t *list_sym;
9796
jl_sym_t *dot_sym; jl_sym_t *newvar_sym;
9897
jl_sym_t *boundscheck_sym; jl_sym_t *inbounds_sym;

src/codegen.cpp

+1-15
Original file line numberDiff line numberDiff line change
@@ -3552,20 +3552,6 @@ static jl_cgval_t emit_expr(jl_value_t *expr, jl_codectx_t *ctx, bool isboxed, b
35523552
else if (head == null_sym) {
35533553
return ghostValue(jl_void_type);
35543554
}
3555-
else if (head == static_typeof_sym) {
3556-
jl_value_t *extype = expr_type((jl_value_t*)ex, ctx);
3557-
if (jl_is_type_type(extype)) {
3558-
extype = jl_tparam0(extype);
3559-
if (jl_is_typevar(extype))
3560-
extype = ((jl_tvar_t*)extype)->ub;
3561-
}
3562-
else {
3563-
extype = (jl_value_t*)jl_any_type;
3564-
}
3565-
if (jl_is_tuple_type(extype))
3566-
jl_add_linfo_root(ctx->linfo, extype);
3567-
return mark_julia_const(extype);
3568-
}
35693555
else if (head == new_sym) {
35703556
jl_value_t *ty = expr_type(args[0], ctx);
35713557
size_t nargs = jl_array_len(ex->args);
@@ -3705,7 +3691,7 @@ static jl_cgval_t emit_expr(jl_value_t *expr, jl_codectx_t *ctx, bool isboxed, b
37053691
return ghostValue(jl_void_type);
37063692
}
37073693
// some expression types are metadata and can be ignored
3708-
if (valuepos || !(head == line_sym || head == type_goto_sym)) {
3694+
if (valuepos || !(head == line_sym)) {
37093695
if (head == abstracttype_sym || head == compositetype_sym ||
37103696
head == bitstype_sym) {
37113697
jl_errorf("type definition not allowed inside a local scope");

src/interpreter.c

-3
Original file line numberDiff line numberDiff line change
@@ -219,9 +219,6 @@ static jl_value_t *eval(jl_value_t *e, jl_value_t **locals, size_t nl, size_t ng
219219
else if (ex->head == exc_sym) {
220220
return jl_exception_in_transit;
221221
}
222-
else if (ex->head == static_typeof_sym) {
223-
return (jl_value_t*)jl_any_type;
224-
}
225222
else if (ex->head == method_sym) {
226223
jl_sym_t *fname = (jl_sym_t*)args[0];
227224
assert(jl_expr_nargs(ex) != 1 || jl_is_symbol(fname));

src/jltypes.c

-2
Original file line numberDiff line numberDiff line change
@@ -3590,7 +3590,6 @@ void jl_init_types(void)
35903590
exc_sym = jl_symbol("the_exception");
35913591
enter_sym = jl_symbol("enter");
35923592
leave_sym = jl_symbol("leave");
3593-
static_typeof_sym = jl_symbol("static_typeof");
35943593
new_sym = jl_symbol("new");
35953594
const_sym = jl_symbol("const");
35963595
global_sym = jl_symbol("global");
@@ -3601,7 +3600,6 @@ void jl_init_types(void)
36013600
abstracttype_sym = jl_symbol("abstract_type");
36023601
bitstype_sym = jl_symbol("bits_type");
36033602
compositetype_sym = jl_symbol("composite_type");
3604-
type_goto_sym = jl_symbol("type_goto");
36053603
toplevel_sym = jl_symbol("toplevel");
36063604
dot_sym = jl_symbol(".");
36073605
boundscheck_sym = jl_symbol("boundscheck");

src/julia-parser.scm

+33-21
Original file line numberDiff line numberDiff line change
@@ -1397,22 +1397,23 @@
13971397
(define (parse-comma-separated-assignments s)
13981398
(parse-comma-separated s parse-eq*))
13991399

1400+
(define (parse-iteration-spec s)
1401+
(let ((r (parse-eq* s)))
1402+
(cond ((and (pair? r) (eq? (car r) '=)) r)
1403+
((eq? r ':) r)
1404+
((and (length= r 4) (eq? (car r) 'comparison)
1405+
(or (eq? (caddr r) 'in) (eq? (caddr r) '∈)))
1406+
`(= ,(cadr r) ,(cadddr r)))
1407+
(else
1408+
(error "invalid iteration specification")))))
1409+
14001410
; as above, but allows both "i=r" and "i in r"
14011411
(define (parse-comma-separated-iters s)
14021412
(let loop ((ranges '()))
1403-
(let ((r (parse-eq* s)))
1404-
(let ((r (cond ((and (pair? r) (eq? (car r) '=))
1405-
r)
1406-
((eq? r ':)
1407-
r)
1408-
((and (length= r 4) (eq? (car r) 'comparison)
1409-
(or (eq? (caddr r) 'in) (eq? (caddr r) '∈)))
1410-
`(= ,(cadr r) ,(cadddr r)))
1411-
(else
1412-
(error "invalid iteration specification")))))
1413-
(case (peek-token s)
1414-
((#\,) (take-token s) (loop (cons r ranges)))
1415-
(else (reverse! (cons r ranges))))))))
1413+
(let ((r (parse-iteration-spec s)))
1414+
(case (peek-token s)
1415+
((#\,) (take-token s) (loop (cons r ranges)))
1416+
(else (reverse! (cons r ranges)))))))
14161417

14171418
(define (parse-space-separated-exprs s)
14181419
(with-space-sensitive
@@ -1469,7 +1470,12 @@
14691470
(begin (take-token s) (loop (cons nxt lst))))
14701471
((eqv? c #\;) (loop (cons nxt lst)))
14711472
((equal? c closer) (loop (cons nxt lst)))
1472-
((eq? c 'for) (take-token s) (parse-generator s t closer))
1473+
((eq? c 'for)
1474+
(take-token s)
1475+
(let ((gen (parse-generator s nxt #f)))
1476+
(if (eqv? (require-token s) #\,)
1477+
(take-token s))
1478+
(loop (cons gen lst))))
14731479
;; newline character isn't detectable here
14741480
#;((eqv? c #\newline)
14751481
(error "unexpected line break in argument list"))
@@ -1514,7 +1520,7 @@
15141520
(define (parse-comprehension s first closer)
15151521
(let ((r (parse-comma-separated-iters s)))
15161522
(if (not (eqv? (require-token s) closer))
1517-
(error (string "expected " closer))
1523+
(error (string "expected \"" closer "\""))
15181524
(take-token s))
15191525
`(comprehension ,first ,@r)))
15201526

@@ -1524,12 +1530,11 @@
15241530
`(dict_comprehension ,@(cdr c))
15251531
(error "invalid dict comprehension"))))
15261532

1527-
(define (parse-generator s first closer)
1528-
(let ((r (parse-comma-separated-iters s)))
1529-
(if (not (eqv? (require-token s) closer))
1530-
(error (string "expected " closer))
1531-
(take-token s))
1532-
`(macrocall @generator ,first ,@r)))
1533+
(define (parse-generator s first allow-comma)
1534+
(let ((r (if allow-comma
1535+
(parse-comma-separated-iters s)
1536+
(list (parse-iteration-spec s)))))
1537+
`(generator ,first ,@r)))
15331538

15341539
(define (parse-matrix s first closer gotnewline)
15351540
(define (fix head v) (cons head (reverse v)))
@@ -1959,6 +1964,13 @@
19591964
`(tuple ,ex)
19601965
;; value in parentheses (x)
19611966
ex))
1967+
((eq? t 'for)
1968+
(take-token s)
1969+
(let ((gen (parse-generator s ex #t)))
1970+
(if (eqv? (require-token s) #\) )
1971+
(take-token s)
1972+
(error "expected \")\""))
1973+
gen))
19621974
(else
19631975
;; tuple (x,) (x,y) (x...) etc.
19641976
(if (eqv? t #\, )

0 commit comments

Comments
 (0)