annotate dmd/AsmStatement.d @ 26:05f02c19bd81

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