0
|
1 module dmd.ThrowStatement;
|
|
2
|
114
|
3 import dmd.common;
|
0
|
4 import dmd.Statement;
|
|
5 import dmd.Expression;
|
|
6 import dmd.Loc;
|
|
7 import dmd.IRState;
|
|
8 import dmd.InlineScanState;
|
|
9 import dmd.HdrGenState;
|
|
10 import dmd.OutBuffer;
|
|
11 import dmd.Scope;
|
|
12 import dmd.Expression;
|
|
13 import dmd.FuncDeclaration;
|
|
14 import dmd.BE;
|
|
15
|
|
16 import dmd.backend.Util;
|
|
17 import dmd.backend.Blockx;
|
|
18 import dmd.backend.elem;
|
|
19 import dmd.backend.RTLSYM;
|
|
20 import dmd.backend.OPER;
|
|
21 import dmd.backend.TYM;
|
|
22
|
|
23 class ThrowStatement : Statement
|
|
24 {
|
|
25 Expression exp;
|
|
26
|
|
27 this(Loc loc, Expression exp)
|
|
28 {
|
|
29 super(loc);
|
|
30 this.exp = exp;
|
|
31 }
|
|
32
|
72
|
33 override Statement syntaxCopy()
|
0
|
34 {
|
53
|
35 ThrowStatement s = new ThrowStatement(loc, exp.syntaxCopy());
|
|
36 return s;
|
0
|
37 }
|
|
38
|
72
|
39 override Statement semantic(Scope sc)
|
0
|
40 {
|
|
41 //printf("ThrowStatement::semantic()\n");
|
|
42
|
|
43 FuncDeclaration fd = sc.parent.isFuncDeclaration();
|
|
44 fd.hasReturnExp |= 2;
|
|
45
|
|
46 if (sc.incontract)
|
|
47 error("Throw statements cannot be in contracts");
|
|
48 exp = exp.semantic(sc);
|
|
49 exp = resolveProperties(sc, exp);
|
|
50 if (!exp.type.toBasetype().isClassHandle())
|
|
51 error("can only throw class objects, not type %s", exp.type.toChars());
|
|
52 return this;
|
|
53 }
|
|
54
|
72
|
55 override void toCBuffer(OutBuffer buf, HdrGenState* hgs)
|
0
|
56 {
|
|
57 assert(false);
|
|
58 }
|
|
59
|
72
|
60 override BE blockExit()
|
0
|
61 {
|
|
62 return BE.BEthrow; // obviously
|
|
63 }
|
|
64
|
72
|
65 override Statement inlineScan(InlineScanState* iss)
|
0
|
66 {
|
|
67 if (exp)
|
|
68 exp = exp.inlineScan(iss);
|
|
69 return this;
|
|
70 }
|
|
71
|
72
|
72 override void toIR(IRState* irs)
|
0
|
73 {
|
|
74 // throw(exp)
|
|
75
|
|
76 Blockx *blx = irs.blx;
|
|
77
|
|
78 incUsage(irs, loc);
|
|
79 elem *e = exp.toElem(irs);
|
|
80 static if (false) {
|
|
81 e = el_bin(OPcall, TYvoid, el_var(rtlsym[RTLSYM_LTHROW]),e);
|
|
82 } else {
|
|
83 e = el_bin(OPcall, TYvoid, el_var(rtlsym[RTLSYM_THROW]),e);
|
|
84 }
|
|
85 block_appendexp(blx.curblock, e);
|
|
86 }
|
72
|
87 }
|