annotate dmd/ForeachStatement.d @ 179:cd48cb899aee

Updated to dmd2.040
author korDen
date Sun, 17 Oct 2010 20:56:07 +0400
parents e3afd1303184
children b0d41ff5e0df
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1 module dmd.ForeachStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
2
114
e28b18c23469 added a module dmd.common for commonly used stuff
Trass3r
parents: 96
diff changeset
3 import dmd.common;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
4 import dmd.Statement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
5 import dmd.TOK;
66
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
6 import dmd.Token;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
7 import dmd.Loc;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
8 import dmd.LINK;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
9 import dmd.ArrayTypes;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
10 import dmd.Expression;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
11 import dmd.VarDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
12 import dmd.FuncDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
13 import dmd.Array;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
14 import dmd.Scope;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
15 import dmd.InterState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
16 import dmd.InlineScanState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
17 import dmd.OutBuffer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
18 import dmd.HdrGenState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
19 import dmd.IRState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
20 import dmd.BE;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
21 import dmd.ScopeDsymbol;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
22 import dmd.TypeAArray;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
23 import dmd.Type;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
24 import dmd.CallExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
25 import dmd.WANT;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
26 import dmd.TY;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
27 import dmd.TypeTuple;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
28 import dmd.TupleExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
29 import dmd.Global;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
30 import dmd.Initializer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
31 import dmd.ExpInitializer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
32 import dmd.IntegerExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
33 import dmd.ExpStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
34 import dmd.DeclarationExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
35 import dmd.Dsymbol;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
36 import dmd.BreakStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
37 import dmd.DefaultStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
38 import dmd.CaseStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
39 import dmd.SwitchStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
40 import dmd.VarExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
41 import dmd.AliasDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
42 import dmd.CompoundStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
43 import dmd.ScopeStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
44 import dmd.UnrolledLoopStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
45 import dmd.Identifier;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
46 import dmd.Lexer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
47 import dmd.DeclarationStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
48 import dmd.CompoundDeclarationStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
49 import dmd.AggregateDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
50 import dmd.TypeClass;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
51 import dmd.NotExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
52 import dmd.TypeStruct;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
53 import dmd.FuncLiteralDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
54 import dmd.IdentifierExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
55 import dmd.TypeFunction;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
56 import dmd.GotoStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
57 import dmd.FuncExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
58 import dmd.ReturnStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
59 import dmd.IndexExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
60 import dmd.ForStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
61 import dmd.SliceExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
62 import dmd.DotIdExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
63 import dmd.PostExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
64 import dmd.AddAssignExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
65 import dmd.CmpExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
66 import dmd.Id;
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
67 import dmd.Parameter;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
68 import dmd.STC;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
69
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
70 import dmd.expression.Util;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
71
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
72 import core.stdc.stdio;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
73
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
74 class ForeachStatement : Statement
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
75 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
76 TOK op; // TOKforeach or TOKforeach_reverse
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
77 Parameters arguments; // array of Argument*'s
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
78 Expression aggr;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
79 Statement body_;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
80
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
81 VarDeclaration key;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
82 VarDeclaration value;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
83
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
84 FuncDeclaration func; // function we're lexically in
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
85
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
86 Array cases; // put breaks, continues, gotos and returns here
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
87 Array gotos; // forward referenced goto's go here
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
88
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
89 this(Loc loc, TOK op, Parameters arguments, Expression aggr, Statement body_)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
90 {
178
e3afd1303184 Many small bugs fixed
korDen
parents: 175
diff changeset
91 register();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
92 super(loc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
93
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
94 this.op = op;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
95 this.arguments = arguments;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
96 this.aggr = aggr;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
97 this.body_ = body_;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
98
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
99 gotos = new Array();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
100 cases = new Array();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
101 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
102
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
103 override Statement syntaxCopy()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
104 {
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
105 auto args = Parameter.arraySyntaxCopy(arguments);
53
a8b50ff7f201 ForeachStatement.syntaxCopy
korDen
parents: 12
diff changeset
106 Expression exp = aggr.syntaxCopy();
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
107 auto s = new ForeachStatement(loc, op, args, exp,
53
a8b50ff7f201 ForeachStatement.syntaxCopy
korDen
parents: 12
diff changeset
108 body_ ? body_.syntaxCopy() : null);
a8b50ff7f201 ForeachStatement.syntaxCopy
korDen
parents: 12
diff changeset
109 return s;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
110 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
111
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
112 override Statement semantic(Scope sc)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
113 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
114 //printf("ForeachStatement.semantic() %p\n", this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
115 ScopeDsymbol sym;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
116 Statement s = this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
117 size_t dim = arguments.dim;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
118 TypeAArray taa = null;
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
119 Dsymbol sapply = null;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
120
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
121 Type tn = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
122 Type tnv = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
123
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
124 func = sc.func;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
125 if (func.fes)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
126 func = func.fes.func;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
127
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
128 aggr = aggr.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
129 aggr = resolveProperties(sc, aggr);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
130 aggr = aggr.optimize(WANT.WANTvalue);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
131 if (!aggr.type)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
132 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
133 error("invalid foreach aggregate %s", aggr.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
134 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
135 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
136
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
137 inferApplyArgTypes(op, arguments, aggr);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
138
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
139 /* Check for inference errors
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
140 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
141 if (dim != arguments.dim)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
142 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
143 //printf("dim = %d, arguments.dim = %d\n", dim, arguments.dim);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
144 error("cannot uniquely infer foreach argument types");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
145 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
146 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
147
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
148 Type tab = aggr.type.toBasetype();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
149
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
150 if (tab.ty == TY.Ttuple) // don't generate new scope for tuple loops
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
151 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
152 if (dim < 1 || dim > 2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
153 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
154 error("only one (value) or two (key,value) arguments for tuple foreach");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
155 return s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
156 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
157
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
158 auto tuple = cast(TypeTuple)tab;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
159 Statements statements = new Statements();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
160 //printf("aggr: op = %d, %s\n", aggr.op, aggr.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
161 size_t n;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
162 TupleExp te = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
163 if (aggr.op == TOK.TOKtuple) // expression tuple
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
164 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
165 te = cast(TupleExp)aggr;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
166 n = te.exps.dim;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
167 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
168 else if (aggr.op == TOK.TOKtype) // type tuple
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
169 {
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
170 n = Parameter.dim(tuple.arguments);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
171 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
172 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
173 assert(0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
174
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
175 for (size_t j = 0; j < n; j++)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
176 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
177 size_t k = (op == TOK.TOKforeach) ? j : n - 1 - j;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
178 Expression e;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
179 Type t;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
180 if (te)
90
39648eb578f6 more Expressions work
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 84
diff changeset
181 e = te.exps[k];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
182 else
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
183 t = Parameter.getNth(tuple.arguments, k).type;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
184
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
185 auto arg = arguments[0];
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
186 auto st = new Statements();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
187
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
188 if (dim == 2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
189 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
190 // Declare key
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
191 if (arg.storageClass & (STC.STCout | STC.STCref | STC.STClazy))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
192 error("no storage class for key %s", arg.ident.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
193 TY keyty = arg.type.ty;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
194 if (keyty != TY.Tint32 && keyty != TY.Tuns32)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
195 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
196 if (global.params.isX86_64)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
197 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
198 if (keyty != TY.Tint64 && keyty != TY.Tuns64)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
199 error("foreach: key type must be int or uint, long or ulong, not %s", arg.type.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
200 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
201 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
202 error("foreach: key type must be int or uint, not %s", arg.type.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
203 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
204 Initializer ie = new ExpInitializer(Loc(0), new IntegerExp(k));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
205 VarDeclaration var = new VarDeclaration(loc, arg.type, arg.ident, ie);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
206 var.storage_class |= STC.STCmanifest;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
207 DeclarationExp de = new DeclarationExp(loc, var);
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
208 st.push(new ExpStatement(loc, de));
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
209 arg = arguments[1]; // value
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
210 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
211 // Declare value
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
212 if (arg.storageClass & (STC.STCout | STC.STCref | STC.STClazy))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
213 error("no storage class for value %s", arg.ident.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
214 Dsymbol var;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
215 if (te)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
216 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
217 Type tb = e.type.toBasetype();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
218 if ((tb.ty == TY.Tfunction || tb.ty == TY.Tsarray) && e.op == TOK.TOKvar)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
219 {
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
220 auto ve = cast(VarExp)e;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
221 var = new AliasDeclaration(loc, arg.ident, ve.var);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
222 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
223 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
224 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
225 arg.type = e.type;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
226 Initializer ie = new ExpInitializer(Loc(0), e);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
227 VarDeclaration v = new VarDeclaration(loc, arg.type, arg.ident, ie);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
228 if (e.isConst())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
229 v.storage_class |= STC.STCconst;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
230
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
231 var = v;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
232 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
233 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
234 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
235 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
236 var = new AliasDeclaration(loc, arg.ident, t);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
237 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
238
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
239 auto de = new DeclarationExp(loc, var);
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
240 st.push(new ExpStatement(loc, de));
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
241
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
242 st.push(body_.syntaxCopy());
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
243 s = new CompoundStatement(loc, st);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
244 s = new ScopeStatement(loc, s);
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
245 statements.push(s);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
246 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
247
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
248 s = new UnrolledLoopStatement(loc, statements);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
249 s = s.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
250 return s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
251 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
252
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
253 sym = new ScopeDsymbol();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
254 sym.parent = sc.scopesym;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
255 sc = sc.push(sym);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
256
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
257 sc.noctor++;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
258
96
acd69f84627e further work
Trass3r
parents: 90
diff changeset
259 Lagain:
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
260 Identifier idapply = (op == TOK.TOKforeach_reverse)
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
261 ? Id.applyReverse : Id.apply;
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
262 sapply = null;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
263 switch (tab.ty)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
264 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
265 case TY.Tarray:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
266 case TY.Tsarray:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
267 if (!checkForArgTypes())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
268 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
269
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
270 if (dim < 1 || dim > 2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
271 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
272 error("only one or two arguments for array foreach");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
273 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
274 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
275
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
276 /* Look for special case of parsing char types out of char type
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
277 * array.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
278 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
279 tn = tab.nextOf().toBasetype();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
280 if (tn.ty == TY.Tchar || tn.ty == TY.Twchar || tn.ty == TY.Tdchar)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
281 {
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
282 Parameter arg;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
283
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
284 int i = (dim == 1) ? 0 : 1; // index of value
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
285 arg = arguments[i];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
286 arg.type = arg.type.semantic(loc, sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
287 tnv = arg.type.toBasetype();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
288 if (tnv.ty != tn.ty && (tnv.ty == TY.Tchar || tnv.ty == TY.Twchar || tnv.ty == TY.Tdchar))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
289 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
290 if (arg.storageClass & STC.STCref)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
291 error("foreach: value of UTF conversion cannot be ref");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
292 if (dim == 2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
293 {
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
294 arg = arguments[0];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
295 if (arg.storageClass & STC.STCref)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
296 error("foreach: key cannot be ref");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
297 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
298 goto Lapply;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
299 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
300 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
301
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
302 for (size_t i = 0; i < dim; i++)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
303 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
304 // Declare args
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
305 auto arg = arguments[i];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
306 Type argtype = arg.type.semantic(loc, sc);
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
307 auto var = new VarDeclaration(loc, argtype, arg.ident, null);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
308 var.storage_class |= STC.STCforeach;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
309 var.storage_class |= arg.storageClass & (STC.STCin | STC.STCout | STC.STCref | STC.STC_TYPECTOR);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
310 if (var.storage_class & (STC.STCref | STC.STCout))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
311 var.storage_class |= STC.STCnodtor;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
312
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
313 if (dim == 2 && i == 0)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
314 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
315 key = var;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
316 //var.storage_class |= STCfinal;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
317 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
318 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
319 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
320 value = var;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
321 /* Reference to immutable data should be marked as const
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
322 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
323 if (var.storage_class & STC.STCref && !tn.isMutable())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
324 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
325 var.storage_class |= STC.STCconst;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
326 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
327 }
96
acd69f84627e further work
Trass3r
parents: 90
diff changeset
328 static if (false)
acd69f84627e further work
Trass3r
parents: 90
diff changeset
329 {
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
330 DeclarationExp de = new DeclarationExp(loc, var);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
331 de.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
332 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
333 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
334
96
acd69f84627e further work
Trass3r
parents: 90
diff changeset
335 static if (true)
acd69f84627e further work
Trass3r
parents: 90
diff changeset
336 {
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
337 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
338 /* Convert to a ForStatement
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
339 * foreach (key, value; a) body =>
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
340 * for (T[] tmp = a[], size_t key; key < tmp.length; ++key)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
341 * { T value = tmp[k]; body }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
342 *
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
343 * foreach_reverse (key, value; a) body =>
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
344 * for (T[] tmp = a[], size_t key = tmp.length; key--; )
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
345 * { T value = tmp[k]; body }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
346 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
347 Identifier id = Lexer.uniqueId("__aggr");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
348 ExpInitializer ie = new ExpInitializer(loc, new SliceExp(loc, aggr, null, null));
179
cd48cb899aee Updated to dmd2.040
korDen
parents: 178
diff changeset
349 VarDeclaration tmp = new VarDeclaration(loc, tab.nextOf().arrayOf(), id, ie);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
350
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
351 Expression tmp_length = new DotIdExp(loc, new VarExp(loc, tmp), Id.length);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
352
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
353 if (!key)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
354 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
355 Identifier id2 = Lexer.uniqueId("__key");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
356 key = new VarDeclaration(loc, Type.tsize_t, id2, null);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
357 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
358
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
359 if (op == TOK.TOKforeach_reverse)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
360 key.init = new ExpInitializer(loc, tmp_length);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
361 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
362 key.init = new ExpInitializer(loc, new IntegerExp(0));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
363
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
364 auto cs = new Statements();
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
365 cs.push(new DeclarationStatement(loc, new DeclarationExp(loc, tmp)));
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
366 cs.push(new DeclarationStatement(loc, new DeclarationExp(loc, key)));
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
367 Statement forinit = new CompoundDeclarationStatement(loc, cs);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
368
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
369 Expression cond;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
370 if (op == TOK.TOKforeach_reverse)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
371 // key--
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
372 cond = new PostExp(TOK.TOKminusminus, loc, new VarExp(loc, key));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
373 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
374 // key < tmp.length
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
375 cond = new CmpExp(TOK.TOKlt, loc, new VarExp(loc, key), tmp_length);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
376
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
377 Expression increment = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
378 if (op == TOK.TOKforeach)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
379 // key += 1
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
380 increment = new AddAssignExp(loc, new VarExp(loc, key), new IntegerExp(1));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
381
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
382 // T value = tmp[key];
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
383 value.init = new ExpInitializer(loc, new IndexExp(loc, new VarExp(loc, tmp), new VarExp(loc, key)));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
384 Statement ds = new DeclarationStatement(loc, new DeclarationExp(loc, value));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
385
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
386 body_ = new CompoundStatement(loc, ds, body_);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
387
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
388 ForStatement fs = new ForStatement(loc, forinit, cond, increment, body_);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
389 s = fs.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
390 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
391 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
392 } else {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
393 if (tab.nextOf().implicitConvTo(value.type) < MATCH.MATCHconst)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
394 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
395 if (aggr.op == TOK.TOKstring)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
396 aggr = aggr.implicitCastTo(sc, value.type.arrayOf());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
397 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
398 error("foreach: %s is not an array of %s",
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
399 tab.toChars(), value.type.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
400 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
401
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
402 if (key)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
403 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
404 if (key.type.ty != Tint32 && key.type.ty != Tuns32)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
405 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
406 if (global.params.isX86_64)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
407 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
408 if (key.type.ty != Tint64 && key.type.ty != Tuns64)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
409 error("foreach: key type must be int or uint, long or ulong, not %s", key.type.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
410 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
411 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
412 error("foreach: key type must be int or uint, not %s", key.type.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
413 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
414
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
415 if (key.storage_class & (STCout | STCref))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
416 error("foreach: key cannot be out or ref");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
417 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
418
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
419 sc.sbreak = this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
420 sc.scontinue = this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
421 body_ = body_.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
422 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
423 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
424
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
425 case TY.Taarray:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
426 if (!checkForArgTypes())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
427 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
428
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
429 taa = cast(TypeAArray)tab;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
430 if (dim < 1 || dim > 2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
431 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
432 error("only one or two arguments for associative array foreach");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
433 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
434 }
96
acd69f84627e further work
Trass3r
parents: 90
diff changeset
435 version(SARRAYVALUE)
acd69f84627e further work
Trass3r
parents: 90
diff changeset
436 {
acd69f84627e further work
Trass3r
parents: 90
diff changeset
437 /* This only works if Key or Value is a static array.
acd69f84627e further work
Trass3r
parents: 90
diff changeset
438 */
acd69f84627e further work
Trass3r
parents: 90
diff changeset
439 tab = taa.getImpl().type;
acd69f84627e further work
Trass3r
parents: 90
diff changeset
440 goto Lagain;
acd69f84627e further work
Trass3r
parents: 90
diff changeset
441 }
acd69f84627e further work
Trass3r
parents: 90
diff changeset
442 else
acd69f84627e further work
Trass3r
parents: 90
diff changeset
443 {
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
444 if (op == TOK.TOKforeach_reverse)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
445 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
446 error("no reverse iteration on associative arrays");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
447 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
448 goto Lapply;
96
acd69f84627e further work
Trass3r
parents: 90
diff changeset
449 }
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
450 case TY.Tclass:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
451 case TY.Tstruct:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
452 version (DMDV2) {
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
453 /* Prefer using opApply, if it exists
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
454 */
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
455 if (dim != 1) // only one argument allowed with ranges
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
456 goto Lapply;
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
457
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
458 sapply = search_function(cast(AggregateDeclaration)tab.toDsymbol(sc), idapply);
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
459 if (sapply)
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
460 goto Lapply;
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
461
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
462 { /* Look for range iteration, i.e. the properties
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
463 * .empty, .next, .retreat, .head and .rear
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
464 * foreach (e; aggr) { ... }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
465 * translates to:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
466 * for (auto __r = aggr[]; !__r.empty; __r.next)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
467 * { auto e = __r.head;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
468 * ...
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
469 * }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
470 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
471 AggregateDeclaration ad = (tab.ty == TY.Tclass)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
472 ? cast(AggregateDeclaration)(cast(TypeClass)tab).sym
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
473 : cast(AggregateDeclaration)(cast(TypeStruct)tab).sym;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
474
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
475 Identifier idhead;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
476 Identifier idnext;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
477 if (op == TOK.TOKforeach)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
478 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
479 idhead = Id.Fhead;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
480 idnext = Id.Fnext;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
481 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
482 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
483 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
484 idhead = Id.Ftoe;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
485 idnext = Id.Fretreat;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
486 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
487
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
488 Dsymbol shead = search_function(ad, idhead);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
489 if (!shead)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
490 goto Lapply;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
491
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
492 /* Generate a temporary __r and initialize it with the aggregate.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
493 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
494 Identifier id = Identifier.generateId("__r");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
495 Expression rinit = new SliceExp(loc, aggr, null, null);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
496 rinit = rinit.trySemantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
497 if (!rinit) // if application of [] failed
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
498 rinit = aggr;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
499
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
500 VarDeclaration r = new VarDeclaration(loc, null, id, new ExpInitializer(loc, rinit));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
501
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
502 // r.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
503 //printf("r: %s, init: %s\n", r.toChars(), r.init.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
504 Statement init = new DeclarationStatement(loc, r);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
505 //printf("init: %s\n", init.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
506
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
507 // !__r.empty
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
508 Expression e = new VarExp(loc, r);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
509 e = new DotIdExp(loc, e, Id.Fempty);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
510 Expression condition = new NotExp(loc, e);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
511
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
512 // __r.next
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
513 e = new VarExp(loc, r);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
514 Expression increment = new DotIdExp(loc, e, idnext);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
515
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
516 /* Declaration statement for e:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
517 * auto e = __r.idhead;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
518 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
519 e = new VarExp(loc, r);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
520 Expression einit = new DotIdExp(loc, e, idhead);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
521 // einit = einit.semantic(sc);
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
522 auto arg = arguments[0];
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
523 auto ve = new VarDeclaration(loc, arg.type, arg.ident, new ExpInitializer(loc, einit));
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
524 ve.storage_class |= STC.STCforeach;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
525 ve.storage_class |= arg.storageClass & (STC.STCin | STC.STCout | STC.STCref | STC.STC_TYPECTOR);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
526
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
527 auto de = new DeclarationExp(loc, ve);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
528
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
529 Statement body2 = new CompoundStatement(loc, new DeclarationStatement(loc, de), this.body_);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
530 s = new ForStatement(loc, init, condition, increment, body2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
531
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
532 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
533 printf("init: %s\n", init.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
534 printf("condition: %s\n", condition.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
535 printf("increment: %s\n", increment.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
536 printf("body: %s\n", body2.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
537 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
538 s = s.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
539 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
540 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
541 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
542 case TY.Tdelegate:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
543 Lapply:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
544 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
545 Expression ec;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
546 Expression e;
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
547 Parameter a;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
548
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
549 if (!checkForArgTypes())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
550 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
551 body_ = body_.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
552 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
553 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
554
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
555 Type tret = func.type.nextOf();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
556
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
557 // Need a variable to hold value from any return statements in body.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
558 if (!sc.func.vresult && tret && tret != Type.tvoid)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
559 {
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
560 auto v = new VarDeclaration(loc, tret, Id.result, null);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
561 v.noauto = true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
562 v.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
563 if (!sc.insert(v))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
564 assert(0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
565
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
566 v.parent = sc.func;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
567 sc.func.vresult = v;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
568 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
569
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
570 /* Turn body into the function literal:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
571 * int delegate(ref T arg) { body }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
572 */
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
573 auto args = new Parameters();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
574 for (size_t i = 0; i < dim; i++)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
575 {
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
576 auto arg = arguments[i];
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
577 Identifier id;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
578
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
579 arg.type = arg.type.semantic(loc, sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
580 if (arg.storageClass & STC.STCref)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
581 id = arg.ident;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
582 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
583 { // Make a copy of the ref argument so it isn't
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
584 // a reference.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
585 id = Lexer.uniqueId("__applyArg", i);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
586
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
587 Initializer ie = new ExpInitializer(Loc(0), new IdentifierExp(Loc(0), id));
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
588 auto v = new VarDeclaration(Loc(0), arg.type, arg.ident, ie);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
589 s = new DeclarationStatement(Loc(0), v);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
590 body_ = new CompoundStatement(loc, s, body_);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
591 }
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
592 a = new Parameter(STC.STCref, arg.type, id, null);
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
593 args.push(a);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
594 }
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
595 Type t = new TypeFunction(args, Type.tint32, 0, LINK.LINKd);
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
596 FuncLiteralDeclaration fld = new FuncLiteralDeclaration(loc, Loc(0), t, TOK.TOKdelegate, this);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
597 fld.fbody = body_;
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
598 Expression flde = new FuncExp(loc, fld);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
599 flde = flde.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
600 fld.tookAddressOf = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
601
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
602 // Resolve any forward referenced goto's
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
603 for (size_t i = 0; i < gotos.dim; i++)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
604 {
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
605 auto cs = cast(CompoundStatement)gotos.data[i];
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
606 auto gs = cast(GotoStatement)cs.statements[0];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
607
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
608 if (!gs.label.statement)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
609 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
610 // 'Promote' it to this scope, and replace with a return
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
611 cases.push(cast(void*)gs);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
612 s = new ReturnStatement(Loc(0), new IntegerExp(cases.dim + 1));
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
613 cs.statements[0] = s;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
614 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
615 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
616
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
617 if (tab.ty == TY.Taarray)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
618 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
619 // Check types
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
620 auto arg = arguments[0];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
621 if (dim == 2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
622 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
623 if (arg.storageClass & STC.STCref)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
624 error("foreach: index cannot be ref");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
625 if (!arg.type.equals(taa.index))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
626 error("foreach: index must be type %s, not %s", taa.index.toChars(), arg.type.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
627
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
628 arg = arguments[1];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
629 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
630 if (!arg.type.equals(taa.nextOf()))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
631 error("foreach: value must be type %s, not %s", taa.nextOf().toChars(), arg.type.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
632
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
633 /* Call:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
634 * _aaApply(aggr, keysize, flde)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
635 */
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
636 FuncDeclaration fdapply;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
637 if (dim == 2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
638 fdapply = FuncDeclaration.genCfunc(Type.tindex, "_aaApply2");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
639 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
640 fdapply = FuncDeclaration.genCfunc(Type.tindex, "_aaApply");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
641
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
642 ec = new VarExp(Loc(0), fdapply);
84
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
643 auto exps = new Expressions();
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
644 exps.push(aggr);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
645 size_t keysize = cast(uint)taa.index.size();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
646 keysize = (keysize + (PTRSIZE-1)) & ~(PTRSIZE-1);
84
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
647 exps.push(new IntegerExp(Loc(0), keysize, Type.tsize_t));
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
648 exps.push(flde);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
649 e = new CallExp(loc, ec, exps);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
650 e.type = Type.tindex; // don't run semantic() on e
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
651 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
652 else if (tab.ty == TY.Tarray || tab.ty == TY.Tsarray)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
653 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
654 /* Call:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
655 * _aApply(aggr, flde)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
656 */
175
94b6033c07f3 get rid of globals
korDen
parents: 135
diff changeset
657 enum char fntab[9][3] =
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
658 [ "cc","cw","cd",
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
659 "wc","cc","wd",
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
660 "dc","dw","dd"
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
661 ];
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
662 char fdname[7+1+2+ dim.sizeof*3 + 1];
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
663 int flag;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
664
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
665 switch (tn.ty)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
666 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
667 case TY.Tchar: flag = 0; break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
668 case TY.Twchar: flag = 3; break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
669 case TY.Tdchar: flag = 6; break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
670 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
671 switch (tnv.ty)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
672 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
673 case TY.Tchar: flag += 0; break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
674 case TY.Twchar: flag += 1; break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
675 case TY.Tdchar: flag += 2; break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
676 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
677 string r = (op == TOK.TOKforeach_reverse) ? "R" : "";
73
ef02e2e203c2 Updating to dmd2.033
korDen
parents: 72
diff changeset
678 int j = sprintf(fdname.ptr, "_aApply%.*s%.*s%zd".ptr, r, 2, fntab[flag].ptr, dim);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
679 assert(j < fdname.sizeof);
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
680 FuncDeclaration fdapply = FuncDeclaration.genCfunc(Type.tindex, fdname[0..j].idup);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
681
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
682 ec = new VarExp(Loc(0), fdapply);
84
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
683 auto exps = new Expressions();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
684 if (tab.ty == TY.Tsarray)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
685 aggr = aggr.castTo(sc, tn.arrayOf());
84
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
686 exps.push(aggr);
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
687 exps.push(flde);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
688 e = new CallExp(loc, ec, exps);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
689 e.type = Type.tindex; // don't run semantic() on e
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
690 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
691 else if (tab.ty == TY.Tdelegate)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
692 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
693 /* Call:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
694 * aggr(flde)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
695 */
84
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
696 auto exps = new Expressions();
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
697 exps.push(flde);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
698 e = new CallExp(loc, aggr, exps);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
699 e = e.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
700 if (e.type != Type.tint32)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
701 error("opApply() function for %s must return an int", tab.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
702 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
703 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
704 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
705 assert(tab.ty == TY.Tstruct || tab.ty == TY.Tclass);
135
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
706 auto exps = new Expressions();
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
707 if (!sapply)
af1bebfd96a4 dmd 2.038
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 130
diff changeset
708 sapply = search_function(cast(AggregateDeclaration)tab.toDsymbol(sc), idapply);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
709 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
710 TemplateDeclaration td;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
711 if (sapply && (td = sapply.isTemplateDeclaration()) !is null)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
712 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
713 /* Call:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
714 * aggr.apply!(fld)()
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
715 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
716 Objects tiargs = new Objects();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
717 tiargs.push(cast(void*)fld);
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 126
diff changeset
718 ec = new DotTemplateInstanceExp(loc, aggr, idapply, tiargs);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
719 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
720 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
721 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
722 /* Call:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
723 * aggr.apply(flde)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
724 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
725 ec = new DotIdExp(loc, aggr, idapply);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
726 exps.push(cast(void*)flde);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
727 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
728 } else {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
729 ec = new DotIdExp(loc, aggr, idapply);
84
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 73
diff changeset
730 exps.push(flde);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
731 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
732 e = new CallExp(loc, ec, exps);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
733 e = e.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
734 if (e.type != Type.tint32) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
735 error("opApply() function for %s must return an int", tab.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
736 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
737 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
738
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
739 if (!cases.dim)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
740 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
741 // Easy case, a clean exit from the loop
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
742 s = new ExpStatement(loc, e);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
743 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
744 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
745 { // Construct a switch statement around the return value
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
746 // of the apply function.
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
747 auto a2 = new Statements();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
748
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
749 // default: break; takes care of cases 0 and 1
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
750 s = new BreakStatement(Loc(0), null);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
751 s = new DefaultStatement(Loc(0), s);
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
752 a2.push(s);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
753
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
754 // cases 2...
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
755 for (int i = 0; i < cases.dim; i++)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
756 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
757 s = cast(Statement)cases.data[i];
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
758 s = new CaseStatement(Loc(0), new IntegerExp(i + 2), s);
122
c77e9f4f1793 Statements -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
759 a2.push(s);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
760 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
761
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
762 s = new CompoundStatement(loc, a2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
763 s = new SwitchStatement(loc, e, s, false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
764 s = s.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
765 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
766 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
767 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
768
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
769 default:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
770 error("foreach: %s is not an aggregate type", aggr.type.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
771 s = null; // error recovery
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
772 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
773 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
774
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
775 sc.noctor--;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
776 sc.pop();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
777 return s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
778 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
779
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
780 bool checkForArgTypes()
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
781 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
782 bool result = true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
783
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
784 foreach (arg; arguments)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
785 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
786 if (!arg.type)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
787 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
788 error("cannot infer type for %s", arg.ident.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
789 arg.type = Type.terror;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
790 result = false;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
791 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
792 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
793 return result;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
794 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
795
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
796 override bool hasBreak()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
797 {
66
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
798 return true;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
799 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
800
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
801 override bool hasContinue()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
802 {
66
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
803 return true;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
804 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
805
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
806 override bool usesEH()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
807 {
66
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
808 return body_.usesEH();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
809 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
810
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
811 override BE blockExit()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
812 {
66
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
813 BE result = BEfallthru;
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
814
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
815 if (aggr.canThrow())
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
816 result |= BEthrow;
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
817
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
818 if (body_)
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
819 {
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
820 result |= body_.blockExit() & ~(BEbreak | BEcontinue);
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
821 }
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
822 return result;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
823 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
824
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
825 override bool comeFrom()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
826 {
66
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
827 if (body_)
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
828 return body_.comeFrom();
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
829
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
830 return false;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
831 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
832
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
833 override Expression interpret(InterState istate)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
834 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
835 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
836 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
837
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
838 override void toCBuffer(OutBuffer buf, HdrGenState* hgs)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
839 {
66
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
840 buf.writestring(Token.toChars(op));
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
841 buf.writestring(" (");
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
842 for (int i = 0; i < arguments.dim; i++)
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
843 {
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 122
diff changeset
844 auto a = arguments[i];
66
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
845 if (i)
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
846 buf.writestring(", ");
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
847 if (a.storageClass & STCref)
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
848 buf.writestring((global.params.Dversion == 1) ? "inout " : "ref ");
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
849 if (a.type)
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
850 a.type.toCBuffer(buf, a.ident, hgs);
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
851 else
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
852 buf.writestring(a.ident.toChars());
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
853 }
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
854 buf.writestring("; ");
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
855 aggr.toCBuffer(buf, hgs);
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
856 buf.writebyte(')');
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
857 buf.writenl();
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
858 buf.writebyte('{');
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
859 buf.writenl();
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
860 if (body_)
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
861 body_.toCBuffer(buf, hgs);
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
862 buf.writebyte('}');
efb1e5bdf63c more implementations
korDen
parents: 63
diff changeset
863 buf.writenl();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
864 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
865
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
866 override Statement inlineScan(InlineScanState* iss)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
867 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
868 aggr = aggr.inlineScan(iss);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
869 if (body_)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
870 body_ = body_.inlineScan(iss);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
871 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
872 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
873
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
874 override void toIR(IRState* irs)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
875 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
876 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
877 }
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 66
diff changeset
878 }