annotate dmd/LabelStatement.d @ 0:10317f0c89a5

Initial commit
author korDen
date Sat, 24 Oct 2009 08:42:06 +0400
parents
children cab4c37afb89
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1 module dmd.LabelStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
2
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
3 import dmd.Statement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
4 import dmd.Identifier;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
5 import dmd.TryFinallyStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
6 import dmd.Scope;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
7 import dmd.Loc;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
8 import dmd.ExpStatement;
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.InterState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
12 import dmd.InlineScanState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
13 import dmd.LabelDsymbol;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
14 import dmd.FuncDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
15 import dmd.CSX;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
16 import dmd.IRState;
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.BE;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
20
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
21 import dmd.backend.block;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
22 import dmd.backend.Blockx;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
23 import dmd.backend.Util;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
24 import dmd.backend.BC;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
25
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
26 class LabelStatement : 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 Statement statement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
30 TryFinallyStatement tf = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
31 block* lblock = null; // back end
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
32 int isReturnLabel = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
33
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
34 this(Loc loc, Identifier ident, Statement statement)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
35 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
36 super(loc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
37 this.ident = ident;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
38 this.statement = statement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
39 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
40
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
41 Statement syntaxCopy()
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
42 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
43 LabelStatement s = new LabelStatement(loc, ident, statement.syntaxCopy());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
44 return s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
45 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
46
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
47 Statement semantic(Scope sc)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
48 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
49 LabelDsymbol ls;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
50 FuncDeclaration fd = sc.parent.isFuncDeclaration();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
51
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
52 //printf("LabelStatement.semantic()\n");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
53 ls = fd.searchLabel(ident);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
54 if (ls.statement)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
55 error("Label '%s' already defined", ls.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
56 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
57 ls.statement = this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
58 tf = sc.tf;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
59 sc = sc.push();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
60 sc.scopesym = sc.enclosing.scopesym;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
61 sc.callSuper |= CSXlabel;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
62 sc.slabel = this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
63 if (statement)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
64 statement = statement.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
65 sc.pop();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
66 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
67 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
68
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
69 Statements flatten(Scope sc)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
70 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
71 Statements a = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
72
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
73 if (statement)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
74 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
75 a = statement.flatten(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
76 if (a)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
77 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
78 if (!a.dim)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
79 a.push(cast(void*)new ExpStatement(loc, null));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
80
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
81 Statement s = cast(Statement)a.data[0];
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
82
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
83 s = new LabelStatement(loc, ident, s);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
84 a.data[0] = cast(void*)s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
85 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
86 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
87
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
88 return a;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
89 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
90
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
91 bool usesEH()
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
92 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
93 return statement ? statement.usesEH() : false;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
94 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
95
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
96 BE blockExit()
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
97 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
98 //printf("LabelStatement.blockExit(%p)\n", this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
99 return statement ? statement.blockExit() : BE.BEfallthru;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
100 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
101
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
102 bool comeFrom()
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
103 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
104 //printf("LabelStatement.comeFrom()\n");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
105 return true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
106 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
107
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
108 Expression interpret(InterState* istate)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
109 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
110 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
111 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
112
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
113 void toCBuffer(OutBuffer buf, HdrGenState* hgs)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
114 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
115 buf.writestring(ident.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
116 buf.writebyte(':');
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
117 buf.writenl();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
118 if (statement)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
119 statement.toCBuffer(buf, hgs);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
120 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
121
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
122 Statement inlineScan(InlineScanState* iss)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
123 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
124 if (statement)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
125 statement = statement.inlineScan(iss);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
126 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
127 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
128
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
129 void toIR(IRState* irs)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
130 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
131 //printf("LabelStatement.toIR() %p, statement = %p\n", this, statement);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
132 Blockx* blx = irs.blx;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
133 block* bc = blx.curblock;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
134 IRState mystate = IRState(irs,this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
135 mystate.ident = ident;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
136
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
137 if (lblock)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
138 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
139 // We had made a guess about which tryblock the label is in.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
140 // Error if we guessed wrong.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
141 // BUG: should fix this
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
142 if (lblock.Btry != blx.tryblock)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
143 error("cannot goto forward into different try block level");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
144 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
145 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
146 lblock = block_calloc(blx);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
147
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
148 block_next(blx,BCgoto,lblock);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
149 list_append(&bc.Bsucc,blx.curblock);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
150 if (statement)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
151 statement.toIR(&mystate);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
152 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
153 }