annotate dmd/AsmStatement.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 fa9a71a9f5a8
children b0d41ff5e0df
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1 module dmd.AsmStatement;
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;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
4 import dmd.Loc;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
5 import dmd.Statement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
6 import dmd.Token;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
7 import dmd.Scope;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
8 import dmd.OutBuffer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
9 import dmd.HdrGenState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
10 import dmd.IRState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
11 import dmd.BE;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
12 import dmd.LabelDsymbol;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
13 import dmd.Dsymbol;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
14 import dmd.Id;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
15 import dmd.TOK;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
16 import dmd.Global;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
17 import dmd.FuncDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
18 import dmd.Declaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
19 import dmd.LabelStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
20 import dmd.Util;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
21
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
22 import dmd.backend.code;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
23 import dmd.backend.iasm;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
24 import dmd.backend.block;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
25 import dmd.backend.Blockx;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
26 import dmd.backend.Util;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
27 import dmd.codegen.Util;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
28 import dmd.backend.BC;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
29 import dmd.backend.FL;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
30 import dmd.backend.SFL;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
31 import dmd.backend.SC;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
32 import dmd.backend.mTY;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
33 import dmd.backend.Symbol;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
34 import dmd.backend.LIST;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
35
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
36 import core.stdc.string : memset;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
37 import core.stdc.stdlib : exit, EXIT_FAILURE;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
38
174
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
39 import std.stdio;
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
40
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
41 class AsmStatement : Statement
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
42 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
43 Token* tokens;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
44 code* asmcode;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
45 uint asmalign; // alignment of this statement
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
46 bool refparam; // true if function parameter is referenced
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
47 bool naked; // true if function is to be naked
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
48 uint regs; // mask of registers modified
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
49
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
50 this(Loc loc, Token* tokens)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
51 {
178
e3afd1303184 Many small bugs fixed
korDen
parents: 176
diff changeset
52 register();
e3afd1303184 Many small bugs fixed
korDen
parents: 176
diff changeset
53
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
54 super(loc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
55 this.tokens = tokens;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
56 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
57
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 26
diff changeset
58 override Statement syntaxCopy()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
59 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
60 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
61 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
62
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 26
diff changeset
63 override Statement semantic(Scope sc)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
64 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
65 //printf("AsmStatement.semantic()\n");
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
66 //static if (true) {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
67 if (sc.func && sc.func.isSafe())
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
68 error("inline assembler not allowed in @safe function %s", sc.func.toChars());
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
69 //} else {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
70 // if (global.params.safe && !sc.module_.safe)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
71 // {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
72 // error("inline assembler not allowed in safe mode");
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
73 // }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
74 //}
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
75
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
76 OP* o;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
77 OPND* o1 = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
78 OPND* o2 = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
79 OPND* o3 = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
80
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
81 PTRNTAB ptb;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
82 uint usNumops;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
83 ubyte uchPrefix = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
84 ubyte bAsmseen;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
85 char* pszLabel = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
86 code* c;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
87 FuncDeclaration fd = sc.parent.isFuncDeclaration();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
88
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
89 assert(fd);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
90 fd.inlineAsm = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
91
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
92 if (!tokens)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
93 return null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
94
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
95 auto asmstate = &global.asmstate;
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
96
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
97 memset(asmstate, 0, (*asmstate).sizeof);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
98
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
99 asmstate.statement = this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
100 asmstate.sc = sc;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
101
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
102 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
103 // don't use bReturnax anymore, and will fail anyway if we use return type inference
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
104 // Scalar return values will always be in AX. So if it is a scalar
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
105 // then asm block sets return value if it modifies AX, if it is non-scalar
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
106 // then always assume that the ASM block sets up an appropriate return
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
107 // value.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
108
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
109 asmstate.bReturnax = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
110 if (sc.func.type.nextOf().isscalar())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
111 asmstate.bReturnax = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
112 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
113
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
114 // Assume assembler code takes care of setting the return value
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
115 sc.func.hasReturnExp |= 8;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
116
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
117 if (!asmstate.bInit)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
118 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
119 asmstate.bInit = true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
120 init_optab();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
121 asmstate.psDollar = new LabelDsymbol(Id.__dollar);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
122 //asmstate.psLocalsize = new VarDeclaration(0, Type.tint32, Id.__LOCAL_SIZE, null);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
123 asmstate.psLocalsize = new Dsymbol(Id.__LOCAL_SIZE);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
124 cod3_set386();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
125 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
126
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
127 asmstate.loc = loc;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
128
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
129 global.asmtok = tokens;
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
130 asm_token_trans(global.asmtok);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
131 if (setjmp(asmstate.env))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
132 {
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
133 global.asmtok = null; // skip rest of line
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
134 global.tok_value = TOK.TOKeof;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
135 exit(EXIT_FAILURE);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
136 goto AFTER_EMIT;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
137 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
138
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
139 switch (cast(int)global.tok_value)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
140 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
141 case ASMTK.ASMTKnaked:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
142 naked = true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
143 sc.func.naked = true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
144 asm_token();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
145 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
146
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
147 case ASMTK.ASMTKeven:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
148 asm_token();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
149 asmalign = 2;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
150 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
151
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
152 case TOK.TOKalign:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
153 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
154 asm_token();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
155 uint align_ = asm_getnum();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
156 if (ispow2(align_) == -1)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
157 asmerr(ASMERRMSGS.EM_align, align_); // power of 2 expected
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
158 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
159 asmalign = align_;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
160 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
161 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
162
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
163 // The following three convert the keywords 'int', 'in', 'out'
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
164 // to identifiers, since they are x86 instructions.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
165 case TOK.TOKint32:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
166 o = asm_op_lookup(Id.__int.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
167 goto Lopcode;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
168
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
169 case TOK.TOKin:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
170 o = asm_op_lookup(Id.___in.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
171 goto Lopcode;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
172
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
173 case TOK.TOKout:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
174 o = asm_op_lookup(Id.___out.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
175 goto Lopcode;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
176
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
177 case TOK.TOKidentifier:
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
178 o = asm_op_lookup(global.asmtok.ident.toChars());
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
179 if (!o)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
180 goto OPCODE_EXPECTED;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
181
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
182 Lopcode:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
183 asmstate.ucItype = o.usNumops & IT.ITMASK;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
184 asm_token();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
185 if (o.usNumops > 3)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
186 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
187 switch (asmstate.ucItype)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
188 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
189 case IT.ITdata:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
190 asmcode = asm_db_parse(o);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
191 goto AFTER_EMIT;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
192
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
193 case IT.ITaddr:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
194 asmcode = asm_da_parse(o);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
195 goto AFTER_EMIT;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
196
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
197 default:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
198 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
199 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
200 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
201 // get the first part of an expr
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
202 o1 = asm_cond_exp();
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
203 if (global.tok_value == TOK.TOKcomma)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
204 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
205 asm_token();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
206 o2 = asm_cond_exp();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
207 }
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
208 if (global.tok_value == TOK.TOKcomma)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
209 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
210 asm_token();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
211 o3 = asm_cond_exp();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
212 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
213 // match opcode and operands in ptrntab to verify legal inst and
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
214 // generate
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
215
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
216 ptb = asm_classify(o, o1, o2, o3, &usNumops);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
217 assert(ptb.pptb0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
218
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
219 //
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
220 // The Multiply instruction takes 3 operands, but if only 2 are seen
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
221 // then the third should be the second and the second should
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
222 // be a duplicate of the first.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
223 //
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
224
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
225 if (asmstate.ucItype == IT.ITopt &&
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
226 (usNumops == 2) &&
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
227 (ASM_GET_aopty(o2.usFlags) == ASM_OPERAND_TYPE._imm) &&
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
228 ((o.usNumops & IT.ITSIZE) == 3))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
229 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
230 o3 = o2;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
231 o2 = opnd_calloc();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
232 *o2 = *o1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
233
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
234 // Re-classify the opcode because the first classification
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
235 // assumed 2 operands.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
236
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
237 ptb = asm_classify(o, o1, o2, o3, &usNumops);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
238 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
239 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
240 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
241 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
242 if (asmstate.ucItype == IT.ITshift && (ptb.pptb2.usOp2 == 0 ||
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
243 (ptb.pptb2.usOp2 & _cl))) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
244 opnd_free(o2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
245 o2 = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
246 usNumops = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
247 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
248 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
249 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
250
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
251 asmcode = asm_emit(loc, usNumops, ptb, o, o1, o2, o3);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
252 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
253
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
254 default:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
255 OPCODE_EXPECTED:
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
256 asmerr(ASMERRMSGS.EM_opcode_exp, global.asmtok.toChars()); // assembler opcode expected
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
257 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
258 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
259
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
260 AFTER_EMIT:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
261 opnd_free(o1);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
262 opnd_free(o2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
263 opnd_free(o3);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
264 o1 = o2 = o3 = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
265
176
fa9a71a9f5a8 Moved all the mutable globals to Global
korDen
parents: 174
diff changeset
266 if (global.tok_value != TOK.TOKeof)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
267 asmerr(ASMERRMSGS.EM_eol); // end of line expected
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
268
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
269 //return asmstate.bReturnax;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
270 return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
271 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
272
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 26
diff changeset
273 override BE blockExit()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
274 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
275 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
276 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
277
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 26
diff changeset
278 override bool comeFrom()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
279 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
280 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
281 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
282
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 26
diff changeset
283 override void toCBuffer(OutBuffer buf, HdrGenState* hgs)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
284 {
174
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
285 buf.writestring("asm { ");
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
286 Token* t = tokens;
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
287 while (t)
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
288 {
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
289 buf.writestring(t.toChars());
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
290 if (t.next &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
291 t.value != TOKmin &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
292 t.value != TOKcomma &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
293 t.next.value != TOKcomma &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
294 t.value != TOKlbracket &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
295 t.next.value != TOKlbracket &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
296 t.next.value != TOKrbracket &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
297 t.value != TOKlparen &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
298 t.next.value != TOKlparen &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
299 t.next.value != TOKrparen &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
300 t.value != TOKdot &&
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
301 t.next.value != TOKdot)
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
302 {
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
303 buf.writebyte(' ');
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
304 }
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
305 t = t.next;
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
306 }
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
307 buf.writestring("; }");
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
308 buf.writenl();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
309 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
310
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 26
diff changeset
311 override AsmStatement isAsmStatement() { return this; }
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
312
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 26
diff changeset
313 override void toIR(IRState *irs)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
314 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
315 block* bpre;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
316 block* basm;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
317 Declaration d;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
318 Symbol* s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
319 Blockx* blx = irs.blx;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
320
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
321 // dumpCode(asmcode);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
322
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
323 //printf("AsmStatement::toIR(asmcode = %x)\n", asmcode);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
324 bpre = blx.curblock;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
325 block_next(blx,BCgoto,null);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
326 basm = blx.curblock;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
327 list_append(&bpre.Bsucc, basm);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
328 basm.Bcode = asmcode;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
329 basm.Balign = cast(ubyte)asmalign;
174
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
330
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
331 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
332 if (label)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
333 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
334 block* b = labelToBlock(loc, blx, label);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
335 printf("AsmStatement::toIR() %p\n", b);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
336 if (b)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
337 list_append(&basm.Bsucc, b);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
338 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
339 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
340 // Loop through each instruction, fixing Dsymbols into Symbol's
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
341 for (code* c = asmcode; c; c = c.next)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
342 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
343 LabelDsymbol label;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
344 block* b;
174
af724d3510d7 lot os toCBuffer methods implemented
korDen
parents: 130
diff changeset
345
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
346 switch (c.IFL1)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
347 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
348 case FLblockoff:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
349 case FLblock:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
350 // FLblock and FLblockoff have LabelDsymbol's - convert to blocks
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
351 label = c.IEVlsym1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
352 b = labelToBlock(loc, blx, label);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
353 list_append(&basm.Bsucc, b);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
354 c.IEV1.Vblock = b;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
355 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
356
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
357 case FLdsymbol:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
358 case FLfunc:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
359 s = c.IEVdsym1.toSymbol();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
360 if (s.Sclass == SCauto && s.Ssymnum == -1)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
361 symbol_add(s);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
362 c.IEVsym1() = s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
363 c.IFL1 = s.Sfl ? s.Sfl : FLauto;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
364 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
365 default:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
366 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
367 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
368
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
369 // Repeat for second operand
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
370 switch (c.IFL2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
371 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
372 case FLblockoff:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
373 case FLblock:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
374 label = c.IEVlsym2;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
375 b = labelToBlock(loc, blx, label);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
376 list_append(&basm.Bsucc, b);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
377 c.IEV2.Vblock = b;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
378 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
379
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
380 case FLdsymbol:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
381 case FLfunc:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
382 d = c.IEVdsym2;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
383 s = d.toSymbol();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
384 if (s.Sclass == SCauto && s.Ssymnum == -1)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
385 symbol_add(s);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
386 c.IEVsym2() = s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
387 c.IFL2 = s.Sfl ? s.Sfl : FLauto;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
388 if (d.isDataseg())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
389 s.Sflags |= SFLlivexit;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
390 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
391 default:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
392 break;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
393 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
394 //c.print();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
395 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
396
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
397 basm.bIasmrefparam = refparam; // are parameters reference?
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
398 basm.usIasmregs = cast(ushort)regs; // registers modified
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
399
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
400 block_next(blx,BCasm, null);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
401 list_prepend(&basm.Bsucc, blx.curblock);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
402
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
403 if (naked)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
404 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
405 blx.funcsym.Stype.Tty |= mTYnaked;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
406 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
407 }
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 26
diff changeset
408 }