0
|
1 module dmd.OnScopeStatement;
|
|
2
|
114
|
3 import dmd.common;
|
0
|
4 import dmd.Statement;
|
|
5 import dmd.OutBuffer;
|
|
6 import dmd.HdrGenState;
|
|
7 import dmd.Scope;
|
|
8 import dmd.IRState;
|
|
9 import dmd.TOK;
|
|
10 import dmd.Loc;
|
|
11 import dmd.BE;
|
|
12 import dmd.Identifier;
|
|
13 import dmd.ExpInitializer;
|
174
|
14 import dmd.Token;
|
0
|
15 import dmd.IntegerExp;
|
|
16 import dmd.VarDeclaration;
|
|
17 import dmd.Type;
|
|
18 import dmd.AssignExp;
|
|
19 import dmd.VarExp;
|
|
20 import dmd.NotExp;
|
|
21 import dmd.IfStatement;
|
|
22 import dmd.DeclarationStatement;
|
|
23 import dmd.ExpStatement;
|
|
24 import dmd.Expression;
|
|
25 import dmd.Lexer;
|
|
26
|
|
27 class OnScopeStatement : Statement
|
|
28 {
|
|
29 TOK tok;
|
|
30 Statement statement;
|
|
31
|
|
32 this(Loc loc, TOK tok, Statement statement)
|
|
33 {
|
|
34 super(loc);
|
174
|
35
|
0
|
36 this.tok = tok;
|
|
37 this.statement = statement;
|
|
38 }
|
|
39
|
72
|
40 override Statement syntaxCopy()
|
0
|
41 {
|
61
|
42 OnScopeStatement s = new OnScopeStatement(loc,
|
|
43 tok, statement.syntaxCopy());
|
|
44 return s;
|
0
|
45 }
|
|
46
|
72
|
47 override BE blockExit()
|
0
|
48 {
|
|
49 // At this point, this statement is just an empty placeholder
|
|
50 return BE.BEfallthru;
|
|
51 }
|
|
52
|
72
|
53 override void toCBuffer(OutBuffer buf, HdrGenState* hgs)
|
0
|
54 {
|
174
|
55 buf.writestring(Token.toChars(tok));
|
|
56 buf.writebyte(' ');
|
|
57 statement.toCBuffer(buf, hgs);
|
0
|
58 }
|
|
59
|
72
|
60 override Statement semantic(Scope sc)
|
0
|
61 {
|
|
62 /* semantic is called on results of scopeCode() */
|
|
63 return this;
|
|
64 }
|
|
65
|
72
|
66 override bool usesEH()
|
0
|
67 {
|
|
68 assert(false);
|
|
69 }
|
|
70
|
72
|
71 override void scopeCode(Scope sc, Statement* sentry, Statement* sexception, Statement* sfinally)
|
0
|
72 {
|
|
73 //printf("OnScopeStatement::scopeCode()\n");
|
|
74 //print();
|
|
75 *sentry = null;
|
|
76 *sexception = null;
|
|
77 *sfinally = null;
|
|
78 switch (tok)
|
|
79 {
|
|
80 case TOKon_scope_exit:
|
|
81 *sfinally = statement;
|
|
82 break;
|
|
83
|
|
84 case TOKon_scope_failure:
|
|
85 *sexception = statement;
|
|
86 break;
|
|
87
|
|
88 case TOKon_scope_success:
|
|
89 {
|
|
90 /* Create:
|
|
91 * sentry: int x = 0;
|
|
92 * sexception: x = 1;
|
|
93 * sfinally: if (!x) statement;
|
|
94 */
|
|
95 Identifier id = Lexer.uniqueId("__os");
|
|
96
|
|
97 ExpInitializer ie = new ExpInitializer(loc, new IntegerExp(0));
|
|
98 VarDeclaration v = new VarDeclaration(loc, Type.tint32, id, ie);
|
|
99 *sentry = new DeclarationStatement(loc, v);
|
|
100
|
|
101 Expression e = new IntegerExp(1);
|
|
102 e = new AssignExp(Loc(0), new VarExp(Loc(0), v), e);
|
|
103 *sexception = new ExpStatement(Loc(0), e);
|
|
104
|
|
105 e = new VarExp(Loc(0), v);
|
|
106 e = new NotExp(Loc(0), e);
|
|
107 *sfinally = new IfStatement(Loc(0), null, e, statement, null);
|
|
108
|
|
109 break;
|
|
110 }
|
|
111
|
|
112 default:
|
|
113 assert(0);
|
|
114 }
|
|
115 }
|
|
116
|
72
|
117 override void toIR(IRState* irs)
|
0
|
118 {
|
|
119 }
|
72
|
120 }
|