annotate dmd/ContinueStatement.d @ 155:a43c65469219

+ Statement.interpret() + ContinueStatement.interpret() + DoStatement.interpret()
author trass3r
date Wed, 15 Sep 2010 17:31:22 +0200
parents e28b18c23469
children e3afd1303184
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1 module dmd.ContinueStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
2
114
e28b18c23469 added a module dmd.common for commonly used stuff
Trass3r
parents: 72
diff changeset
3 import dmd.common;
155
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
4 import dmd.interpret.Util;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
5 import dmd.Statement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
6 import dmd.FuncDeclaration;
155
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
7 import dmd.GlobalExpressions;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
8 import dmd.IntegerExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
9 import dmd.ReturnStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
10 import dmd.LabelStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
11 import dmd.Identifier;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
12 import dmd.Loc;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
13 import dmd.Scope;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
14 import dmd.Expression;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
15 import dmd.InterState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
16 import dmd.HdrGenState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
17 import dmd.OutBuffer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
18 import dmd.IRState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
19 import dmd.BE;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
20
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
21 import dmd.backend.Util;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
22 import dmd.backend.BC;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
23 import dmd.backend.block;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
24 import dmd.backend.Blockx;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
25
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
26 class ContinueStatement : Statement
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
27 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
28 Identifier ident;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
29
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
30 this(Loc loc, Identifier ident)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
31 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
32 super(loc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
33 this.ident = ident;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
34 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
35
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 63
diff changeset
36 override Statement syntaxCopy()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
37 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
38 ContinueStatement s = new ContinueStatement(loc, ident);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
39 return s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
40 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
41
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 63
diff changeset
42 override Statement semantic(Scope sc)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
43 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
44 //printf("ContinueStatement.semantic() %p\n", this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
45 if (ident)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
46 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
47 Scope scx;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
48 FuncDeclaration thisfunc = sc.func;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
49
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
50 for (scx = sc; scx; scx = scx.enclosing)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
51 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
52 LabelStatement ls;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
53
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
54 if (scx.func != thisfunc) // if in enclosing function
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
55 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
56 if (sc.fes) // if this is the body of a foreach
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
57 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
58 for (; scx; scx = scx.enclosing)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
59 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
60 ls = scx.slabel;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
61 if (ls && ls.ident == ident && ls.statement == sc.fes)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
62 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
63 // Replace continue ident; with return 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
64 return new ReturnStatement(Loc(0), new IntegerExp(0));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
65 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
66 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
67
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
68 /* Post this statement to the fes, and replace
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
69 * it with a return value that caller will put into
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
70 * a switch. Caller will figure out where the break
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
71 * label actually is.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
72 * Case numbers start with 2, not 0, as 0 is continue
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
73 * and 1 is break.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
74 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
75 Statement s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
76 sc.fes.cases.push(cast(void*)this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
77 s = new ReturnStatement(Loc(0), new IntegerExp(sc.fes.cases.dim + 1));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
78 return s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
79 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
80 break; // can't continue to it
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
81 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
82
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
83 ls = scx.slabel;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
84 if (ls && ls.ident == ident)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
85 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
86 Statement s = ls.statement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
87
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
88 if (!s.hasContinue())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
89 error("label '%s' has no continue", ident.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
90 if (ls.tf != sc.tf)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
91 error("cannot continue out of finally block");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
92 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
93 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
94 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
95 error("enclosing label '%s' for continue not found", ident.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
96 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
97 else if (!sc.scontinue)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
98 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
99 if (sc.fes)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
100 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
101 Statement s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
102
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
103 // Replace continue; with return 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
104 s = new ReturnStatement(Loc(0), new IntegerExp(0));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
105 return s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
106 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
107 error("continue is not inside a loop");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
108 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
109 return this;
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: 63
diff changeset
112 override Expression interpret(InterState istate)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
113 {
155
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
114 version(LOG)
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
115 writef("ContinueStatement::interpret()\n");
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
116
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
117 mixin(START);
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
118 if (ident)
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
119 return EXP_CANT_INTERPRET;
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
120 else
a43c65469219 + Statement.interpret()
trass3r
parents: 114
diff changeset
121 return EXP_CONTINUE_INTERPRET;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
122 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
123
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 63
diff changeset
124 override BE blockExit()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
125 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
126 return ident ? BE.BEgoto : BE.BEcontinue;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
127 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
128
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 63
diff changeset
129 override void toCBuffer(OutBuffer buf, HdrGenState* hgs)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
130 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
131 buf.writestring("continue");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
132 if (ident)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
133 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
134 buf.writebyte(' ');
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
135 buf.writestring(ident.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
136 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
137 buf.writebyte(';');
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
138 buf.writenl();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
139 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
140
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 63
diff changeset
141 override void toIR(IRState* irs)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
142 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
143 block* bcont;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
144 block* b;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
145 Blockx* blx = irs.blx;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
146
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
147 //printf("ContinueStatement.toIR() %p\n", this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
148 bcont = irs.getContBlock(ident);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
149 assert(bcont);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
150 b = blx.curblock;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
151 incUsage(irs, loc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
152
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
153 // Adjust exception handler scope index if in different try blocks
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
154 if (b.Btry != bcont.Btry)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
155 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
156 //setScopeIndex(blx, b, bcont.Btry ? bcont.Btry.Bscope_index : -1);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
157 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
158
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
159 /* Nothing more than a 'goto' to the current continue destination
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
160 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
161 list_append(&b.Bsucc, bcont);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
162 block_next(blx, BCgoto, null);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
163 }
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 63
diff changeset
164 }