annotate dmd/ForeachStatement.d @ 178:e3afd1303184

Many small bugs fixed Made all classes derive from TObject to detect memory leaks (functionality is disabled for now) Began work on overriding backend memory allocations (to avoid memory leaks)
author korDen
date Sun, 17 Oct 2010 07:42:00 +0400
parents 94b6033c07f3
children cd48cb899aee
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));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
349 VarDeclaration tmp = new VarDeclaration(loc, aggr.type.nextOf().arrayOf(), id, ie);
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 }