1
2 /*---------------------------------------------------------------*/
3 /*--- begin host_amd64_defs.c ---*/
4 /*---------------------------------------------------------------*/
5
6 /*
7 This file is part of Valgrind, a dynamic binary instrumentation
8 framework.
9
10 Copyright (C) 2004-2011 OpenWorks LLP
11 info@open-works.net
12
13 This program is free software; you can redistribute it and/or
14 modify it under the terms of the GNU General Public License as
15 published by the Free Software Foundation; either version 2 of the
16 License, or (at your option) any later version.
17
18 This program is distributed in the hope that it will be useful, but
19 WITHOUT ANY WARRANTY; without even the implied warranty of
20 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
21 General Public License for more details.
22
23 You should have received a copy of the GNU General Public License
24 along with this program; if not, write to the Free Software
25 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
26 02110-1301, USA.
27
28 The GNU General Public License is contained in the file COPYING.
29
30 Neither the names of the U.S. Department of Energy nor the
31 University of California nor the names of its contributors may be
32 used to endorse or promote products derived from this software
33 without prior written permission.
34 */
35
36 #include "libvex_basictypes.h"
37 #include "libvex.h"
38 #include "libvex_trc_values.h"
39
40 #include "main_util.h"
41 #include "host_generic_regs.h"
42 #include "host_amd64_defs.h"
43
44
45 /* --------- Registers. --------- */
46
ppHRegAMD64(HReg reg)47 void ppHRegAMD64 ( HReg reg )
48 {
49 Int r;
50 static HChar* ireg64_names[16]
51 = { "%rax", "%rcx", "%rdx", "%rbx", "%rsp", "%rbp", "%rsi", "%rdi",
52 "%r8", "%r9", "%r10", "%r11", "%r12", "%r13", "%r14", "%r15" };
53 /* Be generic for all virtual regs. */
54 if (hregIsVirtual(reg)) {
55 ppHReg(reg);
56 return;
57 }
58 /* But specific for real regs. */
59 switch (hregClass(reg)) {
60 case HRcInt64:
61 r = hregNumber(reg);
62 vassert(r >= 0 && r < 16);
63 vex_printf("%s", ireg64_names[r]);
64 return;
65 case HRcFlt64:
66 r = hregNumber(reg);
67 vassert(r >= 0 && r < 6);
68 vex_printf("%%fake%d", r);
69 return;
70 case HRcVec128:
71 r = hregNumber(reg);
72 vassert(r >= 0 && r < 16);
73 vex_printf("%%xmm%d", r);
74 return;
75 default:
76 vpanic("ppHRegAMD64");
77 }
78 }
79
ppHRegAMD64_lo32(HReg reg)80 static void ppHRegAMD64_lo32 ( HReg reg )
81 {
82 Int r;
83 static HChar* ireg32_names[16]
84 = { "%eax", "%ecx", "%edx", "%ebx", "%esp", "%ebp", "%esi", "%edi",
85 "%r8d", "%r9d", "%r10d", "%r11d", "%r12d", "%r13d", "%r14d", "%r15d" };
86 /* Be generic for all virtual regs. */
87 if (hregIsVirtual(reg)) {
88 ppHReg(reg);
89 vex_printf("d");
90 return;
91 }
92 /* But specific for real regs. */
93 switch (hregClass(reg)) {
94 case HRcInt64:
95 r = hregNumber(reg);
96 vassert(r >= 0 && r < 16);
97 vex_printf("%s", ireg32_names[r]);
98 return;
99 default:
100 vpanic("ppHRegAMD64_lo32: invalid regclass");
101 }
102 }
103
hregAMD64_RAX(void)104 HReg hregAMD64_RAX ( void ) { return mkHReg( 0, HRcInt64, False); }
hregAMD64_RCX(void)105 HReg hregAMD64_RCX ( void ) { return mkHReg( 1, HRcInt64, False); }
hregAMD64_RDX(void)106 HReg hregAMD64_RDX ( void ) { return mkHReg( 2, HRcInt64, False); }
hregAMD64_RBX(void)107 HReg hregAMD64_RBX ( void ) { return mkHReg( 3, HRcInt64, False); }
hregAMD64_RSP(void)108 HReg hregAMD64_RSP ( void ) { return mkHReg( 4, HRcInt64, False); }
hregAMD64_RBP(void)109 HReg hregAMD64_RBP ( void ) { return mkHReg( 5, HRcInt64, False); }
hregAMD64_RSI(void)110 HReg hregAMD64_RSI ( void ) { return mkHReg( 6, HRcInt64, False); }
hregAMD64_RDI(void)111 HReg hregAMD64_RDI ( void ) { return mkHReg( 7, HRcInt64, False); }
hregAMD64_R8(void)112 HReg hregAMD64_R8 ( void ) { return mkHReg( 8, HRcInt64, False); }
hregAMD64_R9(void)113 HReg hregAMD64_R9 ( void ) { return mkHReg( 9, HRcInt64, False); }
hregAMD64_R10(void)114 HReg hregAMD64_R10 ( void ) { return mkHReg(10, HRcInt64, False); }
hregAMD64_R11(void)115 HReg hregAMD64_R11 ( void ) { return mkHReg(11, HRcInt64, False); }
hregAMD64_R12(void)116 HReg hregAMD64_R12 ( void ) { return mkHReg(12, HRcInt64, False); }
hregAMD64_R13(void)117 HReg hregAMD64_R13 ( void ) { return mkHReg(13, HRcInt64, False); }
hregAMD64_R14(void)118 HReg hregAMD64_R14 ( void ) { return mkHReg(14, HRcInt64, False); }
hregAMD64_R15(void)119 HReg hregAMD64_R15 ( void ) { return mkHReg(15, HRcInt64, False); }
120
121 //.. HReg hregAMD64_FAKE0 ( void ) { return mkHReg(0, HRcFlt64, False); }
122 //.. HReg hregAMD64_FAKE1 ( void ) { return mkHReg(1, HRcFlt64, False); }
123 //.. HReg hregAMD64_FAKE2 ( void ) { return mkHReg(2, HRcFlt64, False); }
124 //.. HReg hregAMD64_FAKE3 ( void ) { return mkHReg(3, HRcFlt64, False); }
125 //.. HReg hregAMD64_FAKE4 ( void ) { return mkHReg(4, HRcFlt64, False); }
126 //.. HReg hregAMD64_FAKE5 ( void ) { return mkHReg(5, HRcFlt64, False); }
127 //..
hregAMD64_XMM0(void)128 HReg hregAMD64_XMM0 ( void ) { return mkHReg( 0, HRcVec128, False); }
hregAMD64_XMM1(void)129 HReg hregAMD64_XMM1 ( void ) { return mkHReg( 1, HRcVec128, False); }
hregAMD64_XMM2(void)130 HReg hregAMD64_XMM2 ( void ) { return mkHReg( 2, HRcVec128, False); }
hregAMD64_XMM3(void)131 HReg hregAMD64_XMM3 ( void ) { return mkHReg( 3, HRcVec128, False); }
hregAMD64_XMM4(void)132 HReg hregAMD64_XMM4 ( void ) { return mkHReg( 4, HRcVec128, False); }
hregAMD64_XMM5(void)133 HReg hregAMD64_XMM5 ( void ) { return mkHReg( 5, HRcVec128, False); }
hregAMD64_XMM6(void)134 HReg hregAMD64_XMM6 ( void ) { return mkHReg( 6, HRcVec128, False); }
hregAMD64_XMM7(void)135 HReg hregAMD64_XMM7 ( void ) { return mkHReg( 7, HRcVec128, False); }
hregAMD64_XMM8(void)136 HReg hregAMD64_XMM8 ( void ) { return mkHReg( 8, HRcVec128, False); }
hregAMD64_XMM9(void)137 HReg hregAMD64_XMM9 ( void ) { return mkHReg( 9, HRcVec128, False); }
hregAMD64_XMM10(void)138 HReg hregAMD64_XMM10 ( void ) { return mkHReg(10, HRcVec128, False); }
hregAMD64_XMM11(void)139 HReg hregAMD64_XMM11 ( void ) { return mkHReg(11, HRcVec128, False); }
hregAMD64_XMM12(void)140 HReg hregAMD64_XMM12 ( void ) { return mkHReg(12, HRcVec128, False); }
hregAMD64_XMM13(void)141 HReg hregAMD64_XMM13 ( void ) { return mkHReg(13, HRcVec128, False); }
hregAMD64_XMM14(void)142 HReg hregAMD64_XMM14 ( void ) { return mkHReg(14, HRcVec128, False); }
hregAMD64_XMM15(void)143 HReg hregAMD64_XMM15 ( void ) { return mkHReg(15, HRcVec128, False); }
144
145
getAllocableRegs_AMD64(Int * nregs,HReg ** arr)146 void getAllocableRegs_AMD64 ( Int* nregs, HReg** arr )
147 {
148 #if 0
149 *nregs = 6;
150 *arr = LibVEX_Alloc(*nregs * sizeof(HReg));
151 (*arr)[ 0] = hregAMD64_RSI();
152 (*arr)[ 1] = hregAMD64_RDI();
153 (*arr)[ 2] = hregAMD64_RBX();
154
155 (*arr)[ 3] = hregAMD64_XMM7();
156 (*arr)[ 4] = hregAMD64_XMM8();
157 (*arr)[ 5] = hregAMD64_XMM9();
158 #endif
159 #if 1
160 *nregs = 20;
161 *arr = LibVEX_Alloc(*nregs * sizeof(HReg));
162 (*arr)[ 0] = hregAMD64_RSI();
163 (*arr)[ 1] = hregAMD64_RDI();
164 (*arr)[ 2] = hregAMD64_R8();
165 (*arr)[ 3] = hregAMD64_R9();
166 (*arr)[ 4] = hregAMD64_R12();
167 (*arr)[ 5] = hregAMD64_R13();
168 (*arr)[ 6] = hregAMD64_R14();
169 (*arr)[ 7] = hregAMD64_R15();
170 (*arr)[ 8] = hregAMD64_RBX();
171
172 (*arr)[ 9] = hregAMD64_XMM3();
173 (*arr)[10] = hregAMD64_XMM4();
174 (*arr)[11] = hregAMD64_XMM5();
175 (*arr)[12] = hregAMD64_XMM6();
176 (*arr)[13] = hregAMD64_XMM7();
177 (*arr)[14] = hregAMD64_XMM8();
178 (*arr)[15] = hregAMD64_XMM9();
179 (*arr)[16] = hregAMD64_XMM10();
180 (*arr)[17] = hregAMD64_XMM11();
181 (*arr)[18] = hregAMD64_XMM12();
182 (*arr)[19] = hregAMD64_R10();
183 #endif
184 }
185
186
187 /* --------- Condition codes, Intel encoding. --------- */
188
showAMD64CondCode(AMD64CondCode cond)189 HChar* showAMD64CondCode ( AMD64CondCode cond )
190 {
191 switch (cond) {
192 case Acc_O: return "o";
193 case Acc_NO: return "no";
194 case Acc_B: return "b";
195 case Acc_NB: return "nb";
196 case Acc_Z: return "z";
197 case Acc_NZ: return "nz";
198 case Acc_BE: return "be";
199 case Acc_NBE: return "nbe";
200 case Acc_S: return "s";
201 case Acc_NS: return "ns";
202 case Acc_P: return "p";
203 case Acc_NP: return "np";
204 case Acc_L: return "l";
205 case Acc_NL: return "nl";
206 case Acc_LE: return "le";
207 case Acc_NLE: return "nle";
208 case Acc_ALWAYS: return "ALWAYS";
209 default: vpanic("ppAMD64CondCode");
210 }
211 }
212
213
214 /* --------- AMD64AMode: memory address expressions. --------- */
215
AMD64AMode_IR(UInt imm32,HReg reg)216 AMD64AMode* AMD64AMode_IR ( UInt imm32, HReg reg ) {
217 AMD64AMode* am = LibVEX_Alloc(sizeof(AMD64AMode));
218 am->tag = Aam_IR;
219 am->Aam.IR.imm = imm32;
220 am->Aam.IR.reg = reg;
221 return am;
222 }
AMD64AMode_IRRS(UInt imm32,HReg base,HReg indEx,Int shift)223 AMD64AMode* AMD64AMode_IRRS ( UInt imm32, HReg base, HReg indEx, Int shift ) {
224 AMD64AMode* am = LibVEX_Alloc(sizeof(AMD64AMode));
225 am->tag = Aam_IRRS;
226 am->Aam.IRRS.imm = imm32;
227 am->Aam.IRRS.base = base;
228 am->Aam.IRRS.index = indEx;
229 am->Aam.IRRS.shift = shift;
230 vassert(shift >= 0 && shift <= 3);
231 return am;
232 }
233
234 //.. AMD64AMode* dopyAMD64AMode ( AMD64AMode* am ) {
235 //.. switch (am->tag) {
236 //.. case Xam_IR:
237 //.. return AMD64AMode_IR( am->Xam.IR.imm, am->Xam.IR.reg );
238 //.. case Xam_IRRS:
239 //.. return AMD64AMode_IRRS( am->Xam.IRRS.imm, am->Xam.IRRS.base,
240 //.. am->Xam.IRRS.index, am->Xam.IRRS.shift );
241 //.. default:
242 //.. vpanic("dopyAMD64AMode");
243 //.. }
244 //.. }
245
ppAMD64AMode(AMD64AMode * am)246 void ppAMD64AMode ( AMD64AMode* am ) {
247 switch (am->tag) {
248 case Aam_IR:
249 if (am->Aam.IR.imm == 0)
250 vex_printf("(");
251 else
252 vex_printf("0x%x(", am->Aam.IR.imm);
253 ppHRegAMD64(am->Aam.IR.reg);
254 vex_printf(")");
255 return;
256 case Aam_IRRS:
257 vex_printf("0x%x(", am->Aam.IRRS.imm);
258 ppHRegAMD64(am->Aam.IRRS.base);
259 vex_printf(",");
260 ppHRegAMD64(am->Aam.IRRS.index);
261 vex_printf(",%d)", 1 << am->Aam.IRRS.shift);
262 return;
263 default:
264 vpanic("ppAMD64AMode");
265 }
266 }
267
addRegUsage_AMD64AMode(HRegUsage * u,AMD64AMode * am)268 static void addRegUsage_AMD64AMode ( HRegUsage* u, AMD64AMode* am ) {
269 switch (am->tag) {
270 case Aam_IR:
271 addHRegUse(u, HRmRead, am->Aam.IR.reg);
272 return;
273 case Aam_IRRS:
274 addHRegUse(u, HRmRead, am->Aam.IRRS.base);
275 addHRegUse(u, HRmRead, am->Aam.IRRS.index);
276 return;
277 default:
278 vpanic("addRegUsage_AMD64AMode");
279 }
280 }
281
mapRegs_AMD64AMode(HRegRemap * m,AMD64AMode * am)282 static void mapRegs_AMD64AMode ( HRegRemap* m, AMD64AMode* am ) {
283 switch (am->tag) {
284 case Aam_IR:
285 am->Aam.IR.reg = lookupHRegRemap(m, am->Aam.IR.reg);
286 return;
287 case Aam_IRRS:
288 am->Aam.IRRS.base = lookupHRegRemap(m, am->Aam.IRRS.base);
289 am->Aam.IRRS.index = lookupHRegRemap(m, am->Aam.IRRS.index);
290 return;
291 default:
292 vpanic("mapRegs_AMD64AMode");
293 }
294 }
295
296 /* --------- Operand, which can be reg, immediate or memory. --------- */
297
AMD64RMI_Imm(UInt imm32)298 AMD64RMI* AMD64RMI_Imm ( UInt imm32 ) {
299 AMD64RMI* op = LibVEX_Alloc(sizeof(AMD64RMI));
300 op->tag = Armi_Imm;
301 op->Armi.Imm.imm32 = imm32;
302 return op;
303 }
AMD64RMI_Reg(HReg reg)304 AMD64RMI* AMD64RMI_Reg ( HReg reg ) {
305 AMD64RMI* op = LibVEX_Alloc(sizeof(AMD64RMI));
306 op->tag = Armi_Reg;
307 op->Armi.Reg.reg = reg;
308 return op;
309 }
AMD64RMI_Mem(AMD64AMode * am)310 AMD64RMI* AMD64RMI_Mem ( AMD64AMode* am ) {
311 AMD64RMI* op = LibVEX_Alloc(sizeof(AMD64RMI));
312 op->tag = Armi_Mem;
313 op->Armi.Mem.am = am;
314 return op;
315 }
316
ppAMD64RMI_wrk(AMD64RMI * op,Bool lo32)317 static void ppAMD64RMI_wrk ( AMD64RMI* op, Bool lo32 ) {
318 switch (op->tag) {
319 case Armi_Imm:
320 vex_printf("$0x%x", op->Armi.Imm.imm32);
321 return;
322 case Armi_Reg:
323 if (lo32)
324 ppHRegAMD64_lo32(op->Armi.Reg.reg);
325 else
326 ppHRegAMD64(op->Armi.Reg.reg);
327 return;
328 case Armi_Mem:
329 ppAMD64AMode(op->Armi.Mem.am);
330 return;
331 default:
332 vpanic("ppAMD64RMI");
333 }
334 }
ppAMD64RMI(AMD64RMI * op)335 void ppAMD64RMI ( AMD64RMI* op ) {
336 ppAMD64RMI_wrk(op, False/*!lo32*/);
337 }
ppAMD64RMI_lo32(AMD64RMI * op)338 void ppAMD64RMI_lo32 ( AMD64RMI* op ) {
339 ppAMD64RMI_wrk(op, True/*lo32*/);
340 }
341
342 /* An AMD64RMI can only be used in a "read" context (what would it mean
343 to write or modify a literal?) and so we enumerate its registers
344 accordingly. */
addRegUsage_AMD64RMI(HRegUsage * u,AMD64RMI * op)345 static void addRegUsage_AMD64RMI ( HRegUsage* u, AMD64RMI* op ) {
346 switch (op->tag) {
347 case Armi_Imm:
348 return;
349 case Armi_Reg:
350 addHRegUse(u, HRmRead, op->Armi.Reg.reg);
351 return;
352 case Armi_Mem:
353 addRegUsage_AMD64AMode(u, op->Armi.Mem.am);
354 return;
355 default:
356 vpanic("addRegUsage_AMD64RMI");
357 }
358 }
359
mapRegs_AMD64RMI(HRegRemap * m,AMD64RMI * op)360 static void mapRegs_AMD64RMI ( HRegRemap* m, AMD64RMI* op ) {
361 switch (op->tag) {
362 case Armi_Imm:
363 return;
364 case Armi_Reg:
365 op->Armi.Reg.reg = lookupHRegRemap(m, op->Armi.Reg.reg);
366 return;
367 case Armi_Mem:
368 mapRegs_AMD64AMode(m, op->Armi.Mem.am);
369 return;
370 default:
371 vpanic("mapRegs_AMD64RMI");
372 }
373 }
374
375
376 /* --------- Operand, which can be reg or immediate only. --------- */
377
AMD64RI_Imm(UInt imm32)378 AMD64RI* AMD64RI_Imm ( UInt imm32 ) {
379 AMD64RI* op = LibVEX_Alloc(sizeof(AMD64RI));
380 op->tag = Ari_Imm;
381 op->Ari.Imm.imm32 = imm32;
382 return op;
383 }
AMD64RI_Reg(HReg reg)384 AMD64RI* AMD64RI_Reg ( HReg reg ) {
385 AMD64RI* op = LibVEX_Alloc(sizeof(AMD64RI));
386 op->tag = Ari_Reg;
387 op->Ari.Reg.reg = reg;
388 return op;
389 }
390
ppAMD64RI(AMD64RI * op)391 void ppAMD64RI ( AMD64RI* op ) {
392 switch (op->tag) {
393 case Ari_Imm:
394 vex_printf("$0x%x", op->Ari.Imm.imm32);
395 return;
396 case Ari_Reg:
397 ppHRegAMD64(op->Ari.Reg.reg);
398 return;
399 default:
400 vpanic("ppAMD64RI");
401 }
402 }
403
404 /* An AMD64RI can only be used in a "read" context (what would it mean
405 to write or modify a literal?) and so we enumerate its registers
406 accordingly. */
addRegUsage_AMD64RI(HRegUsage * u,AMD64RI * op)407 static void addRegUsage_AMD64RI ( HRegUsage* u, AMD64RI* op ) {
408 switch (op->tag) {
409 case Ari_Imm:
410 return;
411 case Ari_Reg:
412 addHRegUse(u, HRmRead, op->Ari.Reg.reg);
413 return;
414 default:
415 vpanic("addRegUsage_AMD64RI");
416 }
417 }
418
mapRegs_AMD64RI(HRegRemap * m,AMD64RI * op)419 static void mapRegs_AMD64RI ( HRegRemap* m, AMD64RI* op ) {
420 switch (op->tag) {
421 case Ari_Imm:
422 return;
423 case Ari_Reg:
424 op->Ari.Reg.reg = lookupHRegRemap(m, op->Ari.Reg.reg);
425 return;
426 default:
427 vpanic("mapRegs_AMD64RI");
428 }
429 }
430
431
432 /* --------- Operand, which can be reg or memory only. --------- */
433
AMD64RM_Reg(HReg reg)434 AMD64RM* AMD64RM_Reg ( HReg reg ) {
435 AMD64RM* op = LibVEX_Alloc(sizeof(AMD64RM));
436 op->tag = Arm_Reg;
437 op->Arm.Reg.reg = reg;
438 return op;
439 }
AMD64RM_Mem(AMD64AMode * am)440 AMD64RM* AMD64RM_Mem ( AMD64AMode* am ) {
441 AMD64RM* op = LibVEX_Alloc(sizeof(AMD64RM));
442 op->tag = Arm_Mem;
443 op->Arm.Mem.am = am;
444 return op;
445 }
446
ppAMD64RM(AMD64RM * op)447 void ppAMD64RM ( AMD64RM* op ) {
448 switch (op->tag) {
449 case Arm_Mem:
450 ppAMD64AMode(op->Arm.Mem.am);
451 return;
452 case Arm_Reg:
453 ppHRegAMD64(op->Arm.Reg.reg);
454 return;
455 default:
456 vpanic("ppAMD64RM");
457 }
458 }
459
460 /* Because an AMD64RM can be both a source or destination operand, we
461 have to supply a mode -- pertaining to the operand as a whole --
462 indicating how it's being used. */
addRegUsage_AMD64RM(HRegUsage * u,AMD64RM * op,HRegMode mode)463 static void addRegUsage_AMD64RM ( HRegUsage* u, AMD64RM* op, HRegMode mode ) {
464 switch (op->tag) {
465 case Arm_Mem:
466 /* Memory is read, written or modified. So we just want to
467 know the regs read by the amode. */
468 addRegUsage_AMD64AMode(u, op->Arm.Mem.am);
469 return;
470 case Arm_Reg:
471 /* reg is read, written or modified. Add it in the
472 appropriate way. */
473 addHRegUse(u, mode, op->Arm.Reg.reg);
474 return;
475 default:
476 vpanic("addRegUsage_AMD64RM");
477 }
478 }
479
mapRegs_AMD64RM(HRegRemap * m,AMD64RM * op)480 static void mapRegs_AMD64RM ( HRegRemap* m, AMD64RM* op )
481 {
482 switch (op->tag) {
483 case Arm_Mem:
484 mapRegs_AMD64AMode(m, op->Arm.Mem.am);
485 return;
486 case Arm_Reg:
487 op->Arm.Reg.reg = lookupHRegRemap(m, op->Arm.Reg.reg);
488 return;
489 default:
490 vpanic("mapRegs_AMD64RM");
491 }
492 }
493
494
495 /* --------- Instructions. --------- */
496
showAMD64ScalarSz(Int sz)497 static HChar* showAMD64ScalarSz ( Int sz ) {
498 switch (sz) {
499 case 2: return "w";
500 case 4: return "l";
501 case 8: return "q";
502 default: vpanic("showAMD64ScalarSz");
503 }
504 }
505
showAMD64UnaryOp(AMD64UnaryOp op)506 HChar* showAMD64UnaryOp ( AMD64UnaryOp op ) {
507 switch (op) {
508 case Aun_NOT: return "not";
509 case Aun_NEG: return "neg";
510 default: vpanic("showAMD64UnaryOp");
511 }
512 }
513
showAMD64AluOp(AMD64AluOp op)514 HChar* showAMD64AluOp ( AMD64AluOp op ) {
515 switch (op) {
516 case Aalu_MOV: return "mov";
517 case Aalu_CMP: return "cmp";
518 case Aalu_ADD: return "add";
519 case Aalu_SUB: return "sub";
520 case Aalu_ADC: return "adc";
521 case Aalu_SBB: return "sbb";
522 case Aalu_AND: return "and";
523 case Aalu_OR: return "or";
524 case Aalu_XOR: return "xor";
525 case Aalu_MUL: return "imul";
526 default: vpanic("showAMD64AluOp");
527 }
528 }
529
showAMD64ShiftOp(AMD64ShiftOp op)530 HChar* showAMD64ShiftOp ( AMD64ShiftOp op ) {
531 switch (op) {
532 case Ash_SHL: return "shl";
533 case Ash_SHR: return "shr";
534 case Ash_SAR: return "sar";
535 default: vpanic("showAMD64ShiftOp");
536 }
537 }
538
showA87FpOp(A87FpOp op)539 HChar* showA87FpOp ( A87FpOp op ) {
540 switch (op) {
541 //.. case Xfp_ADD: return "add";
542 //.. case Xfp_SUB: return "sub";
543 //.. case Xfp_MUL: return "mul";
544 //.. case Xfp_DIV: return "div";
545 case Afp_SCALE: return "scale";
546 case Afp_ATAN: return "atan";
547 case Afp_YL2X: return "yl2x";
548 case Afp_YL2XP1: return "yl2xp1";
549 case Afp_PREM: return "prem";
550 case Afp_PREM1: return "prem1";
551 case Afp_SQRT: return "sqrt";
552 //.. case Xfp_ABS: return "abs";
553 //.. case Xfp_NEG: return "chs";
554 //.. case Xfp_MOV: return "mov";
555 case Afp_SIN: return "sin";
556 case Afp_COS: return "cos";
557 case Afp_TAN: return "tan";
558 case Afp_ROUND: return "round";
559 case Afp_2XM1: return "2xm1";
560 default: vpanic("showA87FpOp");
561 }
562 }
563
showAMD64SseOp(AMD64SseOp op)564 HChar* showAMD64SseOp ( AMD64SseOp op ) {
565 switch (op) {
566 case Asse_MOV: return "movups";
567 case Asse_ADDF: return "add";
568 case Asse_SUBF: return "sub";
569 case Asse_MULF: return "mul";
570 case Asse_DIVF: return "div";
571 case Asse_MAXF: return "max";
572 case Asse_MINF: return "min";
573 case Asse_CMPEQF: return "cmpFeq";
574 case Asse_CMPLTF: return "cmpFlt";
575 case Asse_CMPLEF: return "cmpFle";
576 case Asse_CMPUNF: return "cmpFun";
577 case Asse_RCPF: return "rcp";
578 case Asse_RSQRTF: return "rsqrt";
579 case Asse_SQRTF: return "sqrt";
580 case Asse_AND: return "and";
581 case Asse_OR: return "or";
582 case Asse_XOR: return "xor";
583 case Asse_ANDN: return "andn";
584 case Asse_ADD8: return "paddb";
585 case Asse_ADD16: return "paddw";
586 case Asse_ADD32: return "paddd";
587 case Asse_ADD64: return "paddq";
588 case Asse_QADD8U: return "paddusb";
589 case Asse_QADD16U: return "paddusw";
590 case Asse_QADD8S: return "paddsb";
591 case Asse_QADD16S: return "paddsw";
592 case Asse_SUB8: return "psubb";
593 case Asse_SUB16: return "psubw";
594 case Asse_SUB32: return "psubd";
595 case Asse_SUB64: return "psubq";
596 case Asse_QSUB8U: return "psubusb";
597 case Asse_QSUB16U: return "psubusw";
598 case Asse_QSUB8S: return "psubsb";
599 case Asse_QSUB16S: return "psubsw";
600 case Asse_MUL16: return "pmullw";
601 case Asse_MULHI16U: return "pmulhuw";
602 case Asse_MULHI16S: return "pmulhw";
603 case Asse_AVG8U: return "pavgb";
604 case Asse_AVG16U: return "pavgw";
605 case Asse_MAX16S: return "pmaxw";
606 case Asse_MAX8U: return "pmaxub";
607 case Asse_MIN16S: return "pminw";
608 case Asse_MIN8U: return "pminub";
609 case Asse_CMPEQ8: return "pcmpeqb";
610 case Asse_CMPEQ16: return "pcmpeqw";
611 case Asse_CMPEQ32: return "pcmpeqd";
612 case Asse_CMPGT8S: return "pcmpgtb";
613 case Asse_CMPGT16S: return "pcmpgtw";
614 case Asse_CMPGT32S: return "pcmpgtd";
615 case Asse_SHL16: return "psllw";
616 case Asse_SHL32: return "pslld";
617 case Asse_SHL64: return "psllq";
618 case Asse_SHR16: return "psrlw";
619 case Asse_SHR32: return "psrld";
620 case Asse_SHR64: return "psrlq";
621 case Asse_SAR16: return "psraw";
622 case Asse_SAR32: return "psrad";
623 case Asse_PACKSSD: return "packssdw";
624 case Asse_PACKSSW: return "packsswb";
625 case Asse_PACKUSW: return "packuswb";
626 case Asse_UNPCKHB: return "punpckhb";
627 case Asse_UNPCKHW: return "punpckhw";
628 case Asse_UNPCKHD: return "punpckhd";
629 case Asse_UNPCKHQ: return "punpckhq";
630 case Asse_UNPCKLB: return "punpcklb";
631 case Asse_UNPCKLW: return "punpcklw";
632 case Asse_UNPCKLD: return "punpckld";
633 case Asse_UNPCKLQ: return "punpcklq";
634 default: vpanic("showAMD64SseOp");
635 }
636 }
637
AMD64Instr_Imm64(ULong imm64,HReg dst)638 AMD64Instr* AMD64Instr_Imm64 ( ULong imm64, HReg dst ) {
639 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
640 i->tag = Ain_Imm64;
641 i->Ain.Imm64.imm64 = imm64;
642 i->Ain.Imm64.dst = dst;
643 return i;
644 }
AMD64Instr_Alu64R(AMD64AluOp op,AMD64RMI * src,HReg dst)645 AMD64Instr* AMD64Instr_Alu64R ( AMD64AluOp op, AMD64RMI* src, HReg dst ) {
646 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
647 i->tag = Ain_Alu64R;
648 i->Ain.Alu64R.op = op;
649 i->Ain.Alu64R.src = src;
650 i->Ain.Alu64R.dst = dst;
651 return i;
652 }
AMD64Instr_Alu64M(AMD64AluOp op,AMD64RI * src,AMD64AMode * dst)653 AMD64Instr* AMD64Instr_Alu64M ( AMD64AluOp op, AMD64RI* src, AMD64AMode* dst ) {
654 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
655 i->tag = Ain_Alu64M;
656 i->Ain.Alu64M.op = op;
657 i->Ain.Alu64M.src = src;
658 i->Ain.Alu64M.dst = dst;
659 vassert(op != Aalu_MUL);
660 return i;
661 }
AMD64Instr_Sh64(AMD64ShiftOp op,UInt src,HReg dst)662 AMD64Instr* AMD64Instr_Sh64 ( AMD64ShiftOp op, UInt src, HReg dst ) {
663 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
664 i->tag = Ain_Sh64;
665 i->Ain.Sh64.op = op;
666 i->Ain.Sh64.src = src;
667 i->Ain.Sh64.dst = dst;
668 return i;
669 }
AMD64Instr_Test64(UInt imm32,HReg dst)670 AMD64Instr* AMD64Instr_Test64 ( UInt imm32, HReg dst ) {
671 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
672 i->tag = Ain_Test64;
673 i->Ain.Test64.imm32 = imm32;
674 i->Ain.Test64.dst = dst;
675 return i;
676 }
AMD64Instr_Unary64(AMD64UnaryOp op,HReg dst)677 AMD64Instr* AMD64Instr_Unary64 ( AMD64UnaryOp op, HReg dst ) {
678 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
679 i->tag = Ain_Unary64;
680 i->Ain.Unary64.op = op;
681 i->Ain.Unary64.dst = dst;
682 return i;
683 }
AMD64Instr_Lea64(AMD64AMode * am,HReg dst)684 AMD64Instr* AMD64Instr_Lea64 ( AMD64AMode* am, HReg dst ) {
685 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
686 i->tag = Ain_Lea64;
687 i->Ain.Lea64.am = am;
688 i->Ain.Lea64.dst = dst;
689 return i;
690 }
AMD64Instr_Alu32R(AMD64AluOp op,AMD64RMI * src,HReg dst)691 AMD64Instr* AMD64Instr_Alu32R ( AMD64AluOp op, AMD64RMI* src, HReg dst ) {
692 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
693 i->tag = Ain_Alu32R;
694 i->Ain.Alu32R.op = op;
695 i->Ain.Alu32R.src = src;
696 i->Ain.Alu32R.dst = dst;
697 switch (op) {
698 case Aalu_ADD: case Aalu_SUB: case Aalu_CMP:
699 case Aalu_AND: case Aalu_OR: case Aalu_XOR: break;
700 default: vassert(0);
701 }
702 return i;
703 }
AMD64Instr_MulL(Bool syned,AMD64RM * src)704 AMD64Instr* AMD64Instr_MulL ( Bool syned, AMD64RM* src ) {
705 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
706 i->tag = Ain_MulL;
707 i->Ain.MulL.syned = syned;
708 i->Ain.MulL.src = src;
709 return i;
710 }
AMD64Instr_Div(Bool syned,Int sz,AMD64RM * src)711 AMD64Instr* AMD64Instr_Div ( Bool syned, Int sz, AMD64RM* src ) {
712 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
713 i->tag = Ain_Div;
714 i->Ain.Div.syned = syned;
715 i->Ain.Div.sz = sz;
716 i->Ain.Div.src = src;
717 vassert(sz == 4 || sz == 8);
718 return i;
719 }
720 //.. AMD64Instr* AMD64Instr_Sh3232 ( AMD64ShiftOp op, UInt amt, HReg src, HReg dst ) {
721 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
722 //.. i->tag = Xin_Sh3232;
723 //.. i->Xin.Sh3232.op = op;
724 //.. i->Xin.Sh3232.amt = amt;
725 //.. i->Xin.Sh3232.src = src;
726 //.. i->Xin.Sh3232.dst = dst;
727 //.. vassert(op == Xsh_SHL || op == Xsh_SHR);
728 //.. return i;
729 //.. }
AMD64Instr_Push(AMD64RMI * src)730 AMD64Instr* AMD64Instr_Push( AMD64RMI* src ) {
731 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
732 i->tag = Ain_Push;
733 i->Ain.Push.src = src;
734 return i;
735 }
AMD64Instr_Call(AMD64CondCode cond,Addr64 target,Int regparms)736 AMD64Instr* AMD64Instr_Call ( AMD64CondCode cond, Addr64 target, Int regparms ) {
737 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
738 i->tag = Ain_Call;
739 i->Ain.Call.cond = cond;
740 i->Ain.Call.target = target;
741 i->Ain.Call.regparms = regparms;
742 vassert(regparms >= 0 && regparms <= 6);
743 return i;
744 }
AMD64Instr_Goto(IRJumpKind jk,AMD64CondCode cond,AMD64RI * dst)745 AMD64Instr* AMD64Instr_Goto ( IRJumpKind jk, AMD64CondCode cond, AMD64RI* dst ) {
746 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
747 i->tag = Ain_Goto;
748 i->Ain.Goto.cond = cond;
749 i->Ain.Goto.dst = dst;
750 i->Ain.Goto.jk = jk;
751 return i;
752 }
AMD64Instr_CMov64(AMD64CondCode cond,AMD64RM * src,HReg dst)753 AMD64Instr* AMD64Instr_CMov64 ( AMD64CondCode cond, AMD64RM* src, HReg dst ) {
754 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
755 i->tag = Ain_CMov64;
756 i->Ain.CMov64.cond = cond;
757 i->Ain.CMov64.src = src;
758 i->Ain.CMov64.dst = dst;
759 vassert(cond != Acc_ALWAYS);
760 return i;
761 }
AMD64Instr_MovxLQ(Bool syned,HReg src,HReg dst)762 AMD64Instr* AMD64Instr_MovxLQ ( Bool syned, HReg src, HReg dst ) {
763 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
764 i->tag = Ain_MovxLQ;
765 i->Ain.MovxLQ.syned = syned;
766 i->Ain.MovxLQ.src = src;
767 i->Ain.MovxLQ.dst = dst;
768 return i;
769 }
AMD64Instr_LoadEX(UChar szSmall,Bool syned,AMD64AMode * src,HReg dst)770 AMD64Instr* AMD64Instr_LoadEX ( UChar szSmall, Bool syned,
771 AMD64AMode* src, HReg dst ) {
772 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
773 i->tag = Ain_LoadEX;
774 i->Ain.LoadEX.szSmall = szSmall;
775 i->Ain.LoadEX.syned = syned;
776 i->Ain.LoadEX.src = src;
777 i->Ain.LoadEX.dst = dst;
778 vassert(szSmall == 1 || szSmall == 2 || szSmall == 4);
779 return i;
780 }
AMD64Instr_Store(UChar sz,HReg src,AMD64AMode * dst)781 AMD64Instr* AMD64Instr_Store ( UChar sz, HReg src, AMD64AMode* dst ) {
782 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
783 i->tag = Ain_Store;
784 i->Ain.Store.sz = sz;
785 i->Ain.Store.src = src;
786 i->Ain.Store.dst = dst;
787 vassert(sz == 1 || sz == 2 || sz == 4);
788 return i;
789 }
AMD64Instr_Set64(AMD64CondCode cond,HReg dst)790 AMD64Instr* AMD64Instr_Set64 ( AMD64CondCode cond, HReg dst ) {
791 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
792 i->tag = Ain_Set64;
793 i->Ain.Set64.cond = cond;
794 i->Ain.Set64.dst = dst;
795 return i;
796 }
AMD64Instr_Bsfr64(Bool isFwds,HReg src,HReg dst)797 AMD64Instr* AMD64Instr_Bsfr64 ( Bool isFwds, HReg src, HReg dst ) {
798 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
799 i->tag = Ain_Bsfr64;
800 i->Ain.Bsfr64.isFwds = isFwds;
801 i->Ain.Bsfr64.src = src;
802 i->Ain.Bsfr64.dst = dst;
803 return i;
804 }
AMD64Instr_MFence(void)805 AMD64Instr* AMD64Instr_MFence ( void ) {
806 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
807 i->tag = Ain_MFence;
808 return i;
809 }
AMD64Instr_ACAS(AMD64AMode * addr,UChar sz)810 AMD64Instr* AMD64Instr_ACAS ( AMD64AMode* addr, UChar sz ) {
811 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
812 i->tag = Ain_ACAS;
813 i->Ain.ACAS.addr = addr;
814 i->Ain.ACAS.sz = sz;
815 vassert(sz == 8 || sz == 4 || sz == 2 || sz == 1);
816 return i;
817 }
AMD64Instr_DACAS(AMD64AMode * addr,UChar sz)818 AMD64Instr* AMD64Instr_DACAS ( AMD64AMode* addr, UChar sz ) {
819 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
820 i->tag = Ain_DACAS;
821 i->Ain.DACAS.addr = addr;
822 i->Ain.DACAS.sz = sz;
823 vassert(sz == 8 || sz == 4);
824 return i;
825 }
826
AMD64Instr_A87Free(Int nregs)827 AMD64Instr* AMD64Instr_A87Free ( Int nregs )
828 {
829 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
830 i->tag = Ain_A87Free;
831 i->Ain.A87Free.nregs = nregs;
832 vassert(nregs >= 1 && nregs <= 7);
833 return i;
834 }
AMD64Instr_A87PushPop(AMD64AMode * addr,Bool isPush,UChar szB)835 AMD64Instr* AMD64Instr_A87PushPop ( AMD64AMode* addr, Bool isPush, UChar szB )
836 {
837 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
838 i->tag = Ain_A87PushPop;
839 i->Ain.A87PushPop.addr = addr;
840 i->Ain.A87PushPop.isPush = isPush;
841 i->Ain.A87PushPop.szB = szB;
842 vassert(szB == 8 || szB == 4);
843 return i;
844 }
AMD64Instr_A87FpOp(A87FpOp op)845 AMD64Instr* AMD64Instr_A87FpOp ( A87FpOp op )
846 {
847 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
848 i->tag = Ain_A87FpOp;
849 i->Ain.A87FpOp.op = op;
850 return i;
851 }
AMD64Instr_A87LdCW(AMD64AMode * addr)852 AMD64Instr* AMD64Instr_A87LdCW ( AMD64AMode* addr )
853 {
854 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
855 i->tag = Ain_A87LdCW;
856 i->Ain.A87LdCW.addr = addr;
857 return i;
858 }
AMD64Instr_A87StSW(AMD64AMode * addr)859 AMD64Instr* AMD64Instr_A87StSW ( AMD64AMode* addr )
860 {
861 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
862 i->tag = Ain_A87StSW;
863 i->Ain.A87StSW.addr = addr;
864 return i;
865 }
866
867 //.. AMD64Instr* AMD64Instr_FpUnary ( AMD64FpOp op, HReg src, HReg dst ) {
868 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
869 //.. i->tag = Xin_FpUnary;
870 //.. i->Xin.FpUnary.op = op;
871 //.. i->Xin.FpUnary.src = src;
872 //.. i->Xin.FpUnary.dst = dst;
873 //.. return i;
874 //.. }
875 //.. AMD64Instr* AMD64Instr_FpBinary ( AMD64FpOp op, HReg srcL, HReg srcR, HReg dst ) {
876 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
877 //.. i->tag = Xin_FpBinary;
878 //.. i->Xin.FpBinary.op = op;
879 //.. i->Xin.FpBinary.srcL = srcL;
880 //.. i->Xin.FpBinary.srcR = srcR;
881 //.. i->Xin.FpBinary.dst = dst;
882 //.. return i;
883 //.. }
884 //.. AMD64Instr* AMD64Instr_FpLdSt ( Bool isLoad, UChar sz, HReg reg, AMD64AMode* addr ) {
885 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
886 //.. i->tag = Xin_FpLdSt;
887 //.. i->Xin.FpLdSt.isLoad = isLoad;
888 //.. i->Xin.FpLdSt.sz = sz;
889 //.. i->Xin.FpLdSt.reg = reg;
890 //.. i->Xin.FpLdSt.addr = addr;
891 //.. vassert(sz == 4 || sz == 8);
892 //.. return i;
893 //.. }
894 //.. AMD64Instr* AMD64Instr_FpLdStI ( Bool isLoad, UChar sz,
895 //.. HReg reg, AMD64AMode* addr ) {
896 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
897 //.. i->tag = Xin_FpLdStI;
898 //.. i->Xin.FpLdStI.isLoad = isLoad;
899 //.. i->Xin.FpLdStI.sz = sz;
900 //.. i->Xin.FpLdStI.reg = reg;
901 //.. i->Xin.FpLdStI.addr = addr;
902 //.. vassert(sz == 2 || sz == 4 || sz == 8);
903 //.. return i;
904 //.. }
905 //.. AMD64Instr* AMD64Instr_Fp64to32 ( HReg src, HReg dst ) {
906 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
907 //.. i->tag = Xin_Fp64to32;
908 //.. i->Xin.Fp64to32.src = src;
909 //.. i->Xin.Fp64to32.dst = dst;
910 //.. return i;
911 //.. }
912 //.. AMD64Instr* AMD64Instr_FpCMov ( AMD64CondCode cond, HReg src, HReg dst ) {
913 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
914 //.. i->tag = Xin_FpCMov;
915 //.. i->Xin.FpCMov.cond = cond;
916 //.. i->Xin.FpCMov.src = src;
917 //.. i->Xin.FpCMov.dst = dst;
918 //.. vassert(cond != Xcc_ALWAYS);
919 //.. return i;
920 //.. }
AMD64Instr_LdMXCSR(AMD64AMode * addr)921 AMD64Instr* AMD64Instr_LdMXCSR ( AMD64AMode* addr ) {
922 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
923 i->tag = Ain_LdMXCSR;
924 i->Ain.LdMXCSR.addr = addr;
925 return i;
926 }
927 //.. AMD64Instr* AMD64Instr_FpStSW_AX ( void ) {
928 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
929 //.. i->tag = Xin_FpStSW_AX;
930 //.. return i;
931 //.. }
AMD64Instr_SseUComIS(Int sz,HReg srcL,HReg srcR,HReg dst)932 AMD64Instr* AMD64Instr_SseUComIS ( Int sz, HReg srcL, HReg srcR, HReg dst ) {
933 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
934 i->tag = Ain_SseUComIS;
935 i->Ain.SseUComIS.sz = toUChar(sz);
936 i->Ain.SseUComIS.srcL = srcL;
937 i->Ain.SseUComIS.srcR = srcR;
938 i->Ain.SseUComIS.dst = dst;
939 vassert(sz == 4 || sz == 8);
940 return i;
941 }
AMD64Instr_SseSI2SF(Int szS,Int szD,HReg src,HReg dst)942 AMD64Instr* AMD64Instr_SseSI2SF ( Int szS, Int szD, HReg src, HReg dst ) {
943 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
944 i->tag = Ain_SseSI2SF;
945 i->Ain.SseSI2SF.szS = toUChar(szS);
946 i->Ain.SseSI2SF.szD = toUChar(szD);
947 i->Ain.SseSI2SF.src = src;
948 i->Ain.SseSI2SF.dst = dst;
949 vassert(szS == 4 || szS == 8);
950 vassert(szD == 4 || szD == 8);
951 return i;
952 }
AMD64Instr_SseSF2SI(Int szS,Int szD,HReg src,HReg dst)953 AMD64Instr* AMD64Instr_SseSF2SI ( Int szS, Int szD, HReg src, HReg dst ) {
954 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
955 i->tag = Ain_SseSF2SI;
956 i->Ain.SseSF2SI.szS = toUChar(szS);
957 i->Ain.SseSF2SI.szD = toUChar(szD);
958 i->Ain.SseSF2SI.src = src;
959 i->Ain.SseSF2SI.dst = dst;
960 vassert(szS == 4 || szS == 8);
961 vassert(szD == 4 || szD == 8);
962 return i;
963 }
AMD64Instr_SseSDSS(Bool from64,HReg src,HReg dst)964 AMD64Instr* AMD64Instr_SseSDSS ( Bool from64, HReg src, HReg dst )
965 {
966 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
967 i->tag = Ain_SseSDSS;
968 i->Ain.SseSDSS.from64 = from64;
969 i->Ain.SseSDSS.src = src;
970 i->Ain.SseSDSS.dst = dst;
971 return i;
972 }
973
974 //.. AMD64Instr* AMD64Instr_SseConst ( UShort con, HReg dst ) {
975 //.. AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
976 //.. i->tag = Xin_SseConst;
977 //.. i->Xin.SseConst.con = con;
978 //.. i->Xin.SseConst.dst = dst;
979 //.. vassert(hregClass(dst) == HRcVec128);
980 //.. return i;
981 //.. }
AMD64Instr_SseLdSt(Bool isLoad,Int sz,HReg reg,AMD64AMode * addr)982 AMD64Instr* AMD64Instr_SseLdSt ( Bool isLoad, Int sz,
983 HReg reg, AMD64AMode* addr ) {
984 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
985 i->tag = Ain_SseLdSt;
986 i->Ain.SseLdSt.isLoad = isLoad;
987 i->Ain.SseLdSt.sz = toUChar(sz);
988 i->Ain.SseLdSt.reg = reg;
989 i->Ain.SseLdSt.addr = addr;
990 vassert(sz == 4 || sz == 8 || sz == 16);
991 return i;
992 }
AMD64Instr_SseLdzLO(Int sz,HReg reg,AMD64AMode * addr)993 AMD64Instr* AMD64Instr_SseLdzLO ( Int sz, HReg reg, AMD64AMode* addr )
994 {
995 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
996 i->tag = Ain_SseLdzLO;
997 i->Ain.SseLdzLO.sz = sz;
998 i->Ain.SseLdzLO.reg = reg;
999 i->Ain.SseLdzLO.addr = addr;
1000 vassert(sz == 4 || sz == 8);
1001 return i;
1002 }
AMD64Instr_Sse32Fx4(AMD64SseOp op,HReg src,HReg dst)1003 AMD64Instr* AMD64Instr_Sse32Fx4 ( AMD64SseOp op, HReg src, HReg dst ) {
1004 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
1005 i->tag = Ain_Sse32Fx4;
1006 i->Ain.Sse32Fx4.op = op;
1007 i->Ain.Sse32Fx4.src = src;
1008 i->Ain.Sse32Fx4.dst = dst;
1009 vassert(op != Asse_MOV);
1010 return i;
1011 }
AMD64Instr_Sse32FLo(AMD64SseOp op,HReg src,HReg dst)1012 AMD64Instr* AMD64Instr_Sse32FLo ( AMD64SseOp op, HReg src, HReg dst ) {
1013 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
1014 i->tag = Ain_Sse32FLo;
1015 i->Ain.Sse32FLo.op = op;
1016 i->Ain.Sse32FLo.src = src;
1017 i->Ain.Sse32FLo.dst = dst;
1018 vassert(op != Asse_MOV);
1019 return i;
1020 }
AMD64Instr_Sse64Fx2(AMD64SseOp op,HReg src,HReg dst)1021 AMD64Instr* AMD64Instr_Sse64Fx2 ( AMD64SseOp op, HReg src, HReg dst ) {
1022 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
1023 i->tag = Ain_Sse64Fx2;
1024 i->Ain.Sse64Fx2.op = op;
1025 i->Ain.Sse64Fx2.src = src;
1026 i->Ain.Sse64Fx2.dst = dst;
1027 vassert(op != Asse_MOV);
1028 return i;
1029 }
AMD64Instr_Sse64FLo(AMD64SseOp op,HReg src,HReg dst)1030 AMD64Instr* AMD64Instr_Sse64FLo ( AMD64SseOp op, HReg src, HReg dst ) {
1031 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
1032 i->tag = Ain_Sse64FLo;
1033 i->Ain.Sse64FLo.op = op;
1034 i->Ain.Sse64FLo.src = src;
1035 i->Ain.Sse64FLo.dst = dst;
1036 vassert(op != Asse_MOV);
1037 return i;
1038 }
AMD64Instr_SseReRg(AMD64SseOp op,HReg re,HReg rg)1039 AMD64Instr* AMD64Instr_SseReRg ( AMD64SseOp op, HReg re, HReg rg ) {
1040 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
1041 i->tag = Ain_SseReRg;
1042 i->Ain.SseReRg.op = op;
1043 i->Ain.SseReRg.src = re;
1044 i->Ain.SseReRg.dst = rg;
1045 return i;
1046 }
AMD64Instr_SseCMov(AMD64CondCode cond,HReg src,HReg dst)1047 AMD64Instr* AMD64Instr_SseCMov ( AMD64CondCode cond, HReg src, HReg dst ) {
1048 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
1049 i->tag = Ain_SseCMov;
1050 i->Ain.SseCMov.cond = cond;
1051 i->Ain.SseCMov.src = src;
1052 i->Ain.SseCMov.dst = dst;
1053 vassert(cond != Acc_ALWAYS);
1054 return i;
1055 }
AMD64Instr_SseShuf(Int order,HReg src,HReg dst)1056 AMD64Instr* AMD64Instr_SseShuf ( Int order, HReg src, HReg dst ) {
1057 AMD64Instr* i = LibVEX_Alloc(sizeof(AMD64Instr));
1058 i->tag = Ain_SseShuf;
1059 i->Ain.SseShuf.order = order;
1060 i->Ain.SseShuf.src = src;
1061 i->Ain.SseShuf.dst = dst;
1062 vassert(order >= 0 && order <= 0xFF);
1063 return i;
1064 }
1065
ppAMD64Instr(AMD64Instr * i,Bool mode64)1066 void ppAMD64Instr ( AMD64Instr* i, Bool mode64 )
1067 {
1068 vassert(mode64 == True);
1069 switch (i->tag) {
1070 case Ain_Imm64:
1071 vex_printf("movabsq $0x%llx,", i->Ain.Imm64.imm64);
1072 ppHRegAMD64(i->Ain.Imm64.dst);
1073 return;
1074 case Ain_Alu64R:
1075 vex_printf("%sq ", showAMD64AluOp(i->Ain.Alu64R.op));
1076 ppAMD64RMI(i->Ain.Alu64R.src);
1077 vex_printf(",");
1078 ppHRegAMD64(i->Ain.Alu64R.dst);
1079 return;
1080 case Ain_Alu64M:
1081 vex_printf("%sq ", showAMD64AluOp(i->Ain.Alu64M.op));
1082 ppAMD64RI(i->Ain.Alu64M.src);
1083 vex_printf(",");
1084 ppAMD64AMode(i->Ain.Alu64M.dst);
1085 return;
1086 case Ain_Sh64:
1087 vex_printf("%sq ", showAMD64ShiftOp(i->Ain.Sh64.op));
1088 if (i->Ain.Sh64.src == 0)
1089 vex_printf("%%cl,");
1090 else
1091 vex_printf("$%d,", (Int)i->Ain.Sh64.src);
1092 ppHRegAMD64(i->Ain.Sh64.dst);
1093 return;
1094 case Ain_Test64:
1095 vex_printf("testq $%d,", (Int)i->Ain.Test64.imm32);
1096 ppHRegAMD64(i->Ain.Test64.dst);
1097 return;
1098 case Ain_Unary64:
1099 vex_printf("%sq ", showAMD64UnaryOp(i->Ain.Unary64.op));
1100 ppHRegAMD64(i->Ain.Unary64.dst);
1101 return;
1102 case Ain_Lea64:
1103 vex_printf("leaq ");
1104 ppAMD64AMode(i->Ain.Lea64.am);
1105 vex_printf(",");
1106 ppHRegAMD64(i->Ain.Lea64.dst);
1107 return;
1108 case Ain_Alu32R:
1109 vex_printf("%sl ", showAMD64AluOp(i->Ain.Alu32R.op));
1110 ppAMD64RMI_lo32(i->Ain.Alu32R.src);
1111 vex_printf(",");
1112 ppHRegAMD64_lo32(i->Ain.Alu32R.dst);
1113 return;
1114 case Ain_MulL:
1115 vex_printf("%cmulq ", i->Ain.MulL.syned ? 's' : 'u');
1116 ppAMD64RM(i->Ain.MulL.src);
1117 return;
1118 case Ain_Div:
1119 vex_printf("%cdiv%s ",
1120 i->Ain.Div.syned ? 's' : 'u',
1121 showAMD64ScalarSz(i->Ain.Div.sz));
1122 ppAMD64RM(i->Ain.Div.src);
1123 return;
1124 //.. case Xin_Sh3232:
1125 //.. vex_printf("%sdl ", showAMD64ShiftOp(i->Xin.Sh3232.op));
1126 //.. if (i->Xin.Sh3232.amt == 0)
1127 //.. vex_printf(" %%cl,");
1128 //.. else
1129 //.. vex_printf(" $%d,", i->Xin.Sh3232.amt);
1130 //.. ppHRegAMD64(i->Xin.Sh3232.src);
1131 //.. vex_printf(",");
1132 //.. ppHRegAMD64(i->Xin.Sh3232.dst);
1133 //.. return;
1134 case Ain_Push:
1135 vex_printf("pushq ");
1136 ppAMD64RMI(i->Ain.Push.src);
1137 return;
1138 case Ain_Call:
1139 vex_printf("call%s[%d] ",
1140 i->Ain.Call.cond==Acc_ALWAYS
1141 ? "" : showAMD64CondCode(i->Ain.Call.cond),
1142 i->Ain.Call.regparms );
1143 vex_printf("0x%llx", i->Ain.Call.target);
1144 break;
1145 case Ain_Goto:
1146 if (i->Ain.Goto.cond != Acc_ALWAYS) {
1147 vex_printf("if (%%rflags.%s) { ",
1148 showAMD64CondCode(i->Ain.Goto.cond));
1149 }
1150 if (i->Ain.Goto.jk != Ijk_Boring
1151 && i->Ain.Goto.jk != Ijk_Call
1152 && i->Ain.Goto.jk != Ijk_Ret) {
1153 vex_printf("movl $");
1154 ppIRJumpKind(i->Ain.Goto.jk);
1155 vex_printf(",%%ebp ; ");
1156 }
1157 vex_printf("movq ");
1158 ppAMD64RI(i->Ain.Goto.dst);
1159 vex_printf(",%%rax ; movabsq $dispatcher_addr,%%rdx ; jmp *%%rdx");
1160 if (i->Ain.Goto.cond != Acc_ALWAYS) {
1161 vex_printf(" }");
1162 }
1163 return;
1164 case Ain_CMov64:
1165 vex_printf("cmov%s ", showAMD64CondCode(i->Ain.CMov64.cond));
1166 ppAMD64RM(i->Ain.CMov64.src);
1167 vex_printf(",");
1168 ppHRegAMD64(i->Ain.CMov64.dst);
1169 return;
1170 case Ain_MovxLQ:
1171 vex_printf("mov%clq ", i->Ain.MovxLQ.syned ? 's' : 'z');
1172 ppHRegAMD64_lo32(i->Ain.MovxLQ.src);
1173 vex_printf(",");
1174 ppHRegAMD64(i->Ain.MovxLQ.dst);
1175 return;
1176 case Ain_LoadEX:
1177 if (i->Ain.LoadEX.szSmall==4 && !i->Ain.LoadEX.syned) {
1178 vex_printf("movl ");
1179 ppAMD64AMode(i->Ain.LoadEX.src);
1180 vex_printf(",");
1181 ppHRegAMD64_lo32(i->Ain.LoadEX.dst);
1182 } else {
1183 vex_printf("mov%c%cq ",
1184 i->Ain.LoadEX.syned ? 's' : 'z',
1185 i->Ain.LoadEX.szSmall==1
1186 ? 'b'
1187 : (i->Ain.LoadEX.szSmall==2 ? 'w' : 'l'));
1188 ppAMD64AMode(i->Ain.LoadEX.src);
1189 vex_printf(",");
1190 ppHRegAMD64(i->Ain.LoadEX.dst);
1191 }
1192 return;
1193 case Ain_Store:
1194 vex_printf("mov%c ", i->Ain.Store.sz==1 ? 'b'
1195 : (i->Ain.Store.sz==2 ? 'w' : 'l'));
1196 ppHRegAMD64(i->Ain.Store.src);
1197 vex_printf(",");
1198 ppAMD64AMode(i->Ain.Store.dst);
1199 return;
1200 case Ain_Set64:
1201 vex_printf("setq%s ", showAMD64CondCode(i->Ain.Set64.cond));
1202 ppHRegAMD64(i->Ain.Set64.dst);
1203 return;
1204 case Ain_Bsfr64:
1205 vex_printf("bs%cq ", i->Ain.Bsfr64.isFwds ? 'f' : 'r');
1206 ppHRegAMD64(i->Ain.Bsfr64.src);
1207 vex_printf(",");
1208 ppHRegAMD64(i->Ain.Bsfr64.dst);
1209 return;
1210 case Ain_MFence:
1211 vex_printf("mfence" );
1212 return;
1213 case Ain_ACAS:
1214 vex_printf("lock cmpxchg%c ",
1215 i->Ain.ACAS.sz==1 ? 'b' : i->Ain.ACAS.sz==2 ? 'w'
1216 : i->Ain.ACAS.sz==4 ? 'l' : 'q' );
1217 vex_printf("{%%rax->%%rbx},");
1218 ppAMD64AMode(i->Ain.ACAS.addr);
1219 return;
1220 case Ain_DACAS:
1221 vex_printf("lock cmpxchg%db {%%rdx:%%rax->%%rcx:%%rbx},",
1222 (Int)(2 * i->Ain.DACAS.sz));
1223 ppAMD64AMode(i->Ain.DACAS.addr);
1224 return;
1225 case Ain_A87Free:
1226 vex_printf("ffree %%st(7..%d)", 8 - i->Ain.A87Free.nregs );
1227 break;
1228 case Ain_A87PushPop:
1229 vex_printf(i->Ain.A87PushPop.isPush ? "fld%c " : "fstp%c ",
1230 i->Ain.A87PushPop.szB == 4 ? 's' : 'l');
1231 ppAMD64AMode(i->Ain.A87PushPop.addr);
1232 break;
1233 case Ain_A87FpOp:
1234 vex_printf("f%s", showA87FpOp(i->Ain.A87FpOp.op));
1235 break;
1236 case Ain_A87LdCW:
1237 vex_printf("fldcw ");
1238 ppAMD64AMode(i->Ain.A87LdCW.addr);
1239 break;
1240 case Ain_A87StSW:
1241 vex_printf("fstsw ");
1242 ppAMD64AMode(i->Ain.A87StSW.addr);
1243 break;
1244 //.. case Xin_FpUnary:
1245 //.. vex_printf("g%sD ", showAMD64FpOp(i->Xin.FpUnary.op));
1246 //.. ppHRegAMD64(i->Xin.FpUnary.src);
1247 //.. vex_printf(",");
1248 //.. ppHRegAMD64(i->Xin.FpUnary.dst);
1249 //.. break;
1250 //.. case Xin_FpBinary:
1251 //.. vex_printf("g%sD ", showAMD64FpOp(i->Xin.FpBinary.op));
1252 //.. ppHRegAMD64(i->Xin.FpBinary.srcL);
1253 //.. vex_printf(",");
1254 //.. ppHRegAMD64(i->Xin.FpBinary.srcR);
1255 //.. vex_printf(",");
1256 //.. ppHRegAMD64(i->Xin.FpBinary.dst);
1257 //.. break;
1258 //.. case Xin_FpLdSt:
1259 //.. if (i->Xin.FpLdSt.isLoad) {
1260 //.. vex_printf("gld%c " , i->Xin.FpLdSt.sz==8 ? 'D' : 'F');
1261 //.. ppAMD64AMode(i->Xin.FpLdSt.addr);
1262 //.. vex_printf(", ");
1263 //.. ppHRegAMD64(i->Xin.FpLdSt.reg);
1264 //.. } else {
1265 //.. vex_printf("gst%c " , i->Xin.FpLdSt.sz==8 ? 'D' : 'F');
1266 //.. ppHRegAMD64(i->Xin.FpLdSt.reg);
1267 //.. vex_printf(", ");
1268 //.. ppAMD64AMode(i->Xin.FpLdSt.addr);
1269 //.. }
1270 //.. return;
1271 //.. case Xin_FpLdStI:
1272 //.. if (i->Xin.FpLdStI.isLoad) {
1273 //.. vex_printf("gild%s ", i->Xin.FpLdStI.sz==8 ? "ll" :
1274 //.. i->Xin.FpLdStI.sz==4 ? "l" : "w");
1275 //.. ppAMD64AMode(i->Xin.FpLdStI.addr);
1276 //.. vex_printf(", ");
1277 //.. ppHRegAMD64(i->Xin.FpLdStI.reg);
1278 //.. } else {
1279 //.. vex_printf("gist%s ", i->Xin.FpLdStI.sz==8 ? "ll" :
1280 //.. i->Xin.FpLdStI.sz==4 ? "l" : "w");
1281 //.. ppHRegAMD64(i->Xin.FpLdStI.reg);
1282 //.. vex_printf(", ");
1283 //.. ppAMD64AMode(i->Xin.FpLdStI.addr);
1284 //.. }
1285 //.. return;
1286 //.. case Xin_Fp64to32:
1287 //.. vex_printf("gdtof ");
1288 //.. ppHRegAMD64(i->Xin.Fp64to32.src);
1289 //.. vex_printf(",");
1290 //.. ppHRegAMD64(i->Xin.Fp64to32.dst);
1291 //.. return;
1292 //.. case Xin_FpCMov:
1293 //.. vex_printf("gcmov%s ", showAMD64CondCode(i->Xin.FpCMov.cond));
1294 //.. ppHRegAMD64(i->Xin.FpCMov.src);
1295 //.. vex_printf(",");
1296 //.. ppHRegAMD64(i->Xin.FpCMov.dst);
1297 //.. return;
1298 //.. case Xin_FpLdStCW:
1299 //.. vex_printf(i->Xin.FpLdStCW.isLoad ? "fldcw " : "fstcw ");
1300 //.. ppAMD64AMode(i->Xin.FpLdStCW.addr);
1301 //.. return;
1302 //.. case Xin_FpStSW_AX:
1303 //.. vex_printf("fstsw %%ax");
1304 //.. return;
1305 case Ain_LdMXCSR:
1306 vex_printf("ldmxcsr ");
1307 ppAMD64AMode(i->Ain.LdMXCSR.addr);
1308 break;
1309 case Ain_SseUComIS:
1310 vex_printf("ucomis%s ", i->Ain.SseUComIS.sz==4 ? "s" : "d");
1311 ppHRegAMD64(i->Ain.SseUComIS.srcL);
1312 vex_printf(",");
1313 ppHRegAMD64(i->Ain.SseUComIS.srcR);
1314 vex_printf(" ; pushfq ; popq ");
1315 ppHRegAMD64(i->Ain.SseUComIS.dst);
1316 break;
1317 case Ain_SseSI2SF:
1318 vex_printf("cvtsi2s%s ", i->Ain.SseSI2SF.szD==4 ? "s" : "d");
1319 (i->Ain.SseSI2SF.szS==4 ? ppHRegAMD64_lo32 : ppHRegAMD64)
1320 (i->Ain.SseSI2SF.src);
1321 vex_printf(",");
1322 ppHRegAMD64(i->Ain.SseSI2SF.dst);
1323 break;
1324 case Ain_SseSF2SI:
1325 vex_printf("cvts%s2si ", i->Ain.SseSF2SI.szS==4 ? "s" : "d");
1326 ppHRegAMD64(i->Ain.SseSF2SI.src);
1327 vex_printf(",");
1328 (i->Ain.SseSF2SI.szD==4 ? ppHRegAMD64_lo32 : ppHRegAMD64)
1329 (i->Ain.SseSF2SI.dst);
1330 break;
1331 case Ain_SseSDSS:
1332 vex_printf(i->Ain.SseSDSS.from64 ? "cvtsd2ss " : "cvtss2sd ");
1333 ppHRegAMD64(i->Ain.SseSDSS.src);
1334 vex_printf(",");
1335 ppHRegAMD64(i->Ain.SseSDSS.dst);
1336 break;
1337 //.. case Xin_SseConst:
1338 //.. vex_printf("const $0x%04x,", (Int)i->Xin.SseConst.con);
1339 //.. ppHRegAMD64(i->Xin.SseConst.dst);
1340 //.. break;
1341 case Ain_SseLdSt:
1342 switch (i->Ain.SseLdSt.sz) {
1343 case 4: vex_printf("movss "); break;
1344 case 8: vex_printf("movsd "); break;
1345 case 16: vex_printf("movups "); break;
1346 default: vassert(0);
1347 }
1348 if (i->Ain.SseLdSt.isLoad) {
1349 ppAMD64AMode(i->Ain.SseLdSt.addr);
1350 vex_printf(",");
1351 ppHRegAMD64(i->Ain.SseLdSt.reg);
1352 } else {
1353 ppHRegAMD64(i->Ain.SseLdSt.reg);
1354 vex_printf(",");
1355 ppAMD64AMode(i->Ain.SseLdSt.addr);
1356 }
1357 return;
1358 case Ain_SseLdzLO:
1359 vex_printf("movs%s ", i->Ain.SseLdzLO.sz==4 ? "s" : "d");
1360 ppAMD64AMode(i->Ain.SseLdzLO.addr);
1361 vex_printf(",");
1362 ppHRegAMD64(i->Ain.SseLdzLO.reg);
1363 return;
1364 case Ain_Sse32Fx4:
1365 vex_printf("%sps ", showAMD64SseOp(i->Ain.Sse32Fx4.op));
1366 ppHRegAMD64(i->Ain.Sse32Fx4.src);
1367 vex_printf(",");
1368 ppHRegAMD64(i->Ain.Sse32Fx4.dst);
1369 return;
1370 case Ain_Sse32FLo:
1371 vex_printf("%sss ", showAMD64SseOp(i->Ain.Sse32FLo.op));
1372 ppHRegAMD64(i->Ain.Sse32FLo.src);
1373 vex_printf(",");
1374 ppHRegAMD64(i->Ain.Sse32FLo.dst);
1375 return;
1376 case Ain_Sse64Fx2:
1377 vex_printf("%spd ", showAMD64SseOp(i->Ain.Sse64Fx2.op));
1378 ppHRegAMD64(i->Ain.Sse64Fx2.src);
1379 vex_printf(",");
1380 ppHRegAMD64(i->Ain.Sse64Fx2.dst);
1381 return;
1382 case Ain_Sse64FLo:
1383 vex_printf("%ssd ", showAMD64SseOp(i->Ain.Sse64FLo.op));
1384 ppHRegAMD64(i->Ain.Sse64FLo.src);
1385 vex_printf(",");
1386 ppHRegAMD64(i->Ain.Sse64FLo.dst);
1387 return;
1388 case Ain_SseReRg:
1389 vex_printf("%s ", showAMD64SseOp(i->Ain.SseReRg.op));
1390 ppHRegAMD64(i->Ain.SseReRg.src);
1391 vex_printf(",");
1392 ppHRegAMD64(i->Ain.SseReRg.dst);
1393 return;
1394 case Ain_SseCMov:
1395 vex_printf("cmov%s ", showAMD64CondCode(i->Ain.SseCMov.cond));
1396 ppHRegAMD64(i->Ain.SseCMov.src);
1397 vex_printf(",");
1398 ppHRegAMD64(i->Ain.SseCMov.dst);
1399 return;
1400 case Ain_SseShuf:
1401 vex_printf("pshufd $0x%x,", i->Ain.SseShuf.order);
1402 ppHRegAMD64(i->Ain.SseShuf.src);
1403 vex_printf(",");
1404 ppHRegAMD64(i->Ain.SseShuf.dst);
1405 return;
1406
1407 default:
1408 vpanic("ppAMD64Instr");
1409 }
1410 }
1411
1412 /* --------- Helpers for register allocation. --------- */
1413
getRegUsage_AMD64Instr(HRegUsage * u,AMD64Instr * i,Bool mode64)1414 void getRegUsage_AMD64Instr ( HRegUsage* u, AMD64Instr* i, Bool mode64 )
1415 {
1416 Bool unary;
1417 vassert(mode64 == True);
1418 initHRegUsage(u);
1419 switch (i->tag) {
1420 case Ain_Imm64:
1421 addHRegUse(u, HRmWrite, i->Ain.Imm64.dst);
1422 return;
1423 case Ain_Alu64R:
1424 addRegUsage_AMD64RMI(u, i->Ain.Alu64R.src);
1425 if (i->Ain.Alu64R.op == Aalu_MOV) {
1426 addHRegUse(u, HRmWrite, i->Ain.Alu64R.dst);
1427 return;
1428 }
1429 if (i->Ain.Alu64R.op == Aalu_CMP) {
1430 addHRegUse(u, HRmRead, i->Ain.Alu64R.dst);
1431 return;
1432 }
1433 addHRegUse(u, HRmModify, i->Ain.Alu64R.dst);
1434 return;
1435 case Ain_Alu64M:
1436 addRegUsage_AMD64RI(u, i->Ain.Alu64M.src);
1437 addRegUsage_AMD64AMode(u, i->Ain.Alu64M.dst);
1438 return;
1439 case Ain_Sh64:
1440 addHRegUse(u, HRmModify, i->Ain.Sh64.dst);
1441 if (i->Ain.Sh64.src == 0)
1442 addHRegUse(u, HRmRead, hregAMD64_RCX());
1443 return;
1444 case Ain_Test64:
1445 addHRegUse(u, HRmRead, i->Ain.Test64.dst);
1446 return;
1447 case Ain_Unary64:
1448 addHRegUse(u, HRmModify, i->Ain.Unary64.dst);
1449 return;
1450 case Ain_Lea64:
1451 addRegUsage_AMD64AMode(u, i->Ain.Lea64.am);
1452 addHRegUse(u, HRmWrite, i->Ain.Lea64.dst);
1453 return;
1454 case Ain_Alu32R:
1455 vassert(i->Ain.Alu32R.op != Aalu_MOV);
1456 addRegUsage_AMD64RMI(u, i->Ain.Alu32R.src);
1457 if (i->Ain.Alu32R.op == Aalu_CMP) {
1458 addHRegUse(u, HRmRead, i->Ain.Alu32R.dst);
1459 return;
1460 }
1461 addHRegUse(u, HRmModify, i->Ain.Alu32R.dst);
1462 return;
1463 case Ain_MulL:
1464 addRegUsage_AMD64RM(u, i->Ain.MulL.src, HRmRead);
1465 addHRegUse(u, HRmModify, hregAMD64_RAX());
1466 addHRegUse(u, HRmWrite, hregAMD64_RDX());
1467 return;
1468 case Ain_Div:
1469 addRegUsage_AMD64RM(u, i->Ain.Div.src, HRmRead);
1470 addHRegUse(u, HRmModify, hregAMD64_RAX());
1471 addHRegUse(u, HRmModify, hregAMD64_RDX());
1472 return;
1473 //.. case Xin_Sh3232:
1474 //.. addHRegUse(u, HRmRead, i->Xin.Sh3232.src);
1475 //.. addHRegUse(u, HRmModify, i->Xin.Sh3232.dst);
1476 //.. if (i->Xin.Sh3232.amt == 0)
1477 //.. addHRegUse(u, HRmRead, hregAMD64_ECX());
1478 //.. return;
1479 case Ain_Push:
1480 addRegUsage_AMD64RMI(u, i->Ain.Push.src);
1481 addHRegUse(u, HRmModify, hregAMD64_RSP());
1482 return;
1483 case Ain_Call:
1484 /* This is a bit subtle. */
1485 /* First off, claim it trashes all the caller-saved regs
1486 which fall within the register allocator's jurisdiction.
1487 These I believe to be: rax rcx rdx rsi rdi r8 r9 r10 r11
1488 and all the xmm registers.
1489 */
1490 addHRegUse(u, HRmWrite, hregAMD64_RAX());
1491 addHRegUse(u, HRmWrite, hregAMD64_RCX());
1492 addHRegUse(u, HRmWrite, hregAMD64_RDX());
1493 addHRegUse(u, HRmWrite, hregAMD64_RSI());
1494 addHRegUse(u, HRmWrite, hregAMD64_RDI());
1495 addHRegUse(u, HRmWrite, hregAMD64_R8());
1496 addHRegUse(u, HRmWrite, hregAMD64_R9());
1497 addHRegUse(u, HRmWrite, hregAMD64_R10());
1498 addHRegUse(u, HRmWrite, hregAMD64_R11());
1499 addHRegUse(u, HRmWrite, hregAMD64_XMM0());
1500 addHRegUse(u, HRmWrite, hregAMD64_XMM1());
1501 addHRegUse(u, HRmWrite, hregAMD64_XMM2());
1502 addHRegUse(u, HRmWrite, hregAMD64_XMM3());
1503 addHRegUse(u, HRmWrite, hregAMD64_XMM4());
1504 addHRegUse(u, HRmWrite, hregAMD64_XMM5());
1505 addHRegUse(u, HRmWrite, hregAMD64_XMM6());
1506 addHRegUse(u, HRmWrite, hregAMD64_XMM7());
1507 addHRegUse(u, HRmWrite, hregAMD64_XMM8());
1508 addHRegUse(u, HRmWrite, hregAMD64_XMM9());
1509 addHRegUse(u, HRmWrite, hregAMD64_XMM10());
1510 addHRegUse(u, HRmWrite, hregAMD64_XMM11());
1511 addHRegUse(u, HRmWrite, hregAMD64_XMM12());
1512 addHRegUse(u, HRmWrite, hregAMD64_XMM13());
1513 addHRegUse(u, HRmWrite, hregAMD64_XMM14());
1514 addHRegUse(u, HRmWrite, hregAMD64_XMM15());
1515
1516 /* Now we have to state any parameter-carrying registers
1517 which might be read. This depends on the regparmness. */
1518 switch (i->Ain.Call.regparms) {
1519 case 6: addHRegUse(u, HRmRead, hregAMD64_R9()); /*fallthru*/
1520 case 5: addHRegUse(u, HRmRead, hregAMD64_R8()); /*fallthru*/
1521 case 4: addHRegUse(u, HRmRead, hregAMD64_RCX()); /*fallthru*/
1522 case 3: addHRegUse(u, HRmRead, hregAMD64_RDX()); /*fallthru*/
1523 case 2: addHRegUse(u, HRmRead, hregAMD64_RSI()); /*fallthru*/
1524 case 1: addHRegUse(u, HRmRead, hregAMD64_RDI()); break;
1525 case 0: break;
1526 default: vpanic("getRegUsage_AMD64Instr:Call:regparms");
1527 }
1528 /* Finally, there is the issue that the insn trashes a
1529 register because the literal target address has to be
1530 loaded into a register. Fortunately, r11 is stated in the
1531 ABI as a scratch register, and so seems a suitable victim. */
1532 addHRegUse(u, HRmWrite, hregAMD64_R11());
1533 /* Upshot of this is that the assembler really must use r11,
1534 and no other, as a destination temporary. */
1535 return;
1536 case Ain_Goto:
1537 addRegUsage_AMD64RI(u, i->Ain.Goto.dst);
1538 addHRegUse(u, HRmWrite, hregAMD64_RAX()); /* used for next guest addr */
1539 addHRegUse(u, HRmWrite, hregAMD64_RDX()); /* used for dispatcher addr */
1540 if (i->Ain.Goto.jk != Ijk_Boring
1541 && i->Ain.Goto.jk != Ijk_Call
1542 && i->Ain.Goto.jk != Ijk_Ret)
1543 /* note, this is irrelevant since rbp is not actually
1544 available to the allocator. But still .. */
1545 addHRegUse(u, HRmWrite, hregAMD64_RBP());
1546 return;
1547 case Ain_CMov64:
1548 addRegUsage_AMD64RM(u, i->Ain.CMov64.src, HRmRead);
1549 addHRegUse(u, HRmModify, i->Ain.CMov64.dst);
1550 return;
1551 case Ain_MovxLQ:
1552 addHRegUse(u, HRmRead, i->Ain.MovxLQ.src);
1553 addHRegUse(u, HRmWrite, i->Ain.MovxLQ.dst);
1554 return;
1555 case Ain_LoadEX:
1556 addRegUsage_AMD64AMode(u, i->Ain.LoadEX.src);
1557 addHRegUse(u, HRmWrite, i->Ain.LoadEX.dst);
1558 return;
1559 case Ain_Store:
1560 addHRegUse(u, HRmRead, i->Ain.Store.src);
1561 addRegUsage_AMD64AMode(u, i->Ain.Store.dst);
1562 return;
1563 case Ain_Set64:
1564 addHRegUse(u, HRmWrite, i->Ain.Set64.dst);
1565 return;
1566 case Ain_Bsfr64:
1567 addHRegUse(u, HRmRead, i->Ain.Bsfr64.src);
1568 addHRegUse(u, HRmWrite, i->Ain.Bsfr64.dst);
1569 return;
1570 case Ain_MFence:
1571 return;
1572 case Ain_ACAS:
1573 addRegUsage_AMD64AMode(u, i->Ain.ACAS.addr);
1574 addHRegUse(u, HRmRead, hregAMD64_RBX());
1575 addHRegUse(u, HRmModify, hregAMD64_RAX());
1576 return;
1577 case Ain_DACAS:
1578 addRegUsage_AMD64AMode(u, i->Ain.DACAS.addr);
1579 addHRegUse(u, HRmRead, hregAMD64_RCX());
1580 addHRegUse(u, HRmRead, hregAMD64_RBX());
1581 addHRegUse(u, HRmModify, hregAMD64_RDX());
1582 addHRegUse(u, HRmModify, hregAMD64_RAX());
1583 return;
1584 case Ain_A87Free:
1585 return;
1586 case Ain_A87PushPop:
1587 addRegUsage_AMD64AMode(u, i->Ain.A87PushPop.addr);
1588 return;
1589 case Ain_A87FpOp:
1590 return;
1591 case Ain_A87LdCW:
1592 addRegUsage_AMD64AMode(u, i->Ain.A87LdCW.addr);
1593 return;
1594 case Ain_A87StSW:
1595 addRegUsage_AMD64AMode(u, i->Ain.A87StSW.addr);
1596 return;
1597 //.. case Xin_FpUnary:
1598 //.. addHRegUse(u, HRmRead, i->Xin.FpUnary.src);
1599 //.. addHRegUse(u, HRmWrite, i->Xin.FpUnary.dst);
1600 //.. return;
1601 //.. case Xin_FpBinary:
1602 //.. addHRegUse(u, HRmRead, i->Xin.FpBinary.srcL);
1603 //.. addHRegUse(u, HRmRead, i->Xin.FpBinary.srcR);
1604 //.. addHRegUse(u, HRmWrite, i->Xin.FpBinary.dst);
1605 //.. return;
1606 //.. case Xin_FpLdSt:
1607 //.. addRegUsage_AMD64AMode(u, i->Xin.FpLdSt.addr);
1608 //.. addHRegUse(u, i->Xin.FpLdSt.isLoad ? HRmWrite : HRmRead,
1609 //.. i->Xin.FpLdSt.reg);
1610 //.. return;
1611 //.. case Xin_FpLdStI:
1612 //.. addRegUsage_AMD64AMode(u, i->Xin.FpLdStI.addr);
1613 //.. addHRegUse(u, i->Xin.FpLdStI.isLoad ? HRmWrite : HRmRead,
1614 //.. i->Xin.FpLdStI.reg);
1615 //.. return;
1616 //.. case Xin_Fp64to32:
1617 //.. addHRegUse(u, HRmRead, i->Xin.Fp64to32.src);
1618 //.. addHRegUse(u, HRmWrite, i->Xin.Fp64to32.dst);
1619 //.. return;
1620 //.. case Xin_FpCMov:
1621 //.. addHRegUse(u, HRmRead, i->Xin.FpCMov.src);
1622 //.. addHRegUse(u, HRmModify, i->Xin.FpCMov.dst);
1623 //.. return;
1624 case Ain_LdMXCSR:
1625 addRegUsage_AMD64AMode(u, i->Ain.LdMXCSR.addr);
1626 return;
1627 //.. case Xin_FpStSW_AX:
1628 //.. addHRegUse(u, HRmWrite, hregAMD64_EAX());
1629 //.. return;
1630 case Ain_SseUComIS:
1631 addHRegUse(u, HRmRead, i->Ain.SseUComIS.srcL);
1632 addHRegUse(u, HRmRead, i->Ain.SseUComIS.srcR);
1633 addHRegUse(u, HRmWrite, i->Ain.SseUComIS.dst);
1634 return;
1635 case Ain_SseSI2SF:
1636 addHRegUse(u, HRmRead, i->Ain.SseSI2SF.src);
1637 addHRegUse(u, HRmWrite, i->Ain.SseSI2SF.dst);
1638 return;
1639 case Ain_SseSF2SI:
1640 addHRegUse(u, HRmRead, i->Ain.SseSF2SI.src);
1641 addHRegUse(u, HRmWrite, i->Ain.SseSF2SI.dst);
1642 return;
1643 case Ain_SseSDSS:
1644 addHRegUse(u, HRmRead, i->Ain.SseSDSS.src);
1645 addHRegUse(u, HRmWrite, i->Ain.SseSDSS.dst);
1646 return;
1647 case Ain_SseLdSt:
1648 addRegUsage_AMD64AMode(u, i->Ain.SseLdSt.addr);
1649 addHRegUse(u, i->Ain.SseLdSt.isLoad ? HRmWrite : HRmRead,
1650 i->Ain.SseLdSt.reg);
1651 return;
1652 case Ain_SseLdzLO:
1653 addRegUsage_AMD64AMode(u, i->Ain.SseLdzLO.addr);
1654 addHRegUse(u, HRmWrite, i->Ain.SseLdzLO.reg);
1655 return;
1656 //.. case Xin_SseConst:
1657 //.. addHRegUse(u, HRmWrite, i->Xin.SseConst.dst);
1658 //.. return;
1659 case Ain_Sse32Fx4:
1660 vassert(i->Ain.Sse32Fx4.op != Asse_MOV);
1661 unary = toBool( i->Ain.Sse32Fx4.op == Asse_RCPF
1662 || i->Ain.Sse32Fx4.op == Asse_RSQRTF
1663 || i->Ain.Sse32Fx4.op == Asse_SQRTF );
1664 addHRegUse(u, HRmRead, i->Ain.Sse32Fx4.src);
1665 addHRegUse(u, unary ? HRmWrite : HRmModify,
1666 i->Ain.Sse32Fx4.dst);
1667 return;
1668 case Ain_Sse32FLo:
1669 vassert(i->Ain.Sse32FLo.op != Asse_MOV);
1670 unary = toBool( i->Ain.Sse32FLo.op == Asse_RCPF
1671 || i->Ain.Sse32FLo.op == Asse_RSQRTF
1672 || i->Ain.Sse32FLo.op == Asse_SQRTF );
1673 addHRegUse(u, HRmRead, i->Ain.Sse32FLo.src);
1674 addHRegUse(u, unary ? HRmWrite : HRmModify,
1675 i->Ain.Sse32FLo.dst);
1676 return;
1677 case Ain_Sse64Fx2:
1678 vassert(i->Ain.Sse64Fx2.op != Asse_MOV);
1679 unary = toBool( i->Ain.Sse64Fx2.op == Asse_RCPF
1680 || i->Ain.Sse64Fx2.op == Asse_RSQRTF
1681 || i->Ain.Sse64Fx2.op == Asse_SQRTF );
1682 addHRegUse(u, HRmRead, i->Ain.Sse64Fx2.src);
1683 addHRegUse(u, unary ? HRmWrite : HRmModify,
1684 i->Ain.Sse64Fx2.dst);
1685 return;
1686 case Ain_Sse64FLo:
1687 vassert(i->Ain.Sse64FLo.op != Asse_MOV);
1688 unary = toBool( i->Ain.Sse64FLo.op == Asse_RCPF
1689 || i->Ain.Sse64FLo.op == Asse_RSQRTF
1690 || i->Ain.Sse64FLo.op == Asse_SQRTF );
1691 addHRegUse(u, HRmRead, i->Ain.Sse64FLo.src);
1692 addHRegUse(u, unary ? HRmWrite : HRmModify,
1693 i->Ain.Sse64FLo.dst);
1694 return;
1695 case Ain_SseReRg:
1696 if ( (i->Ain.SseReRg.op == Asse_XOR
1697 || i->Ain.SseReRg.op == Asse_CMPEQ32)
1698 && i->Ain.SseReRg.src == i->Ain.SseReRg.dst) {
1699 /* reg-alloc needs to understand 'xor r,r' and 'cmpeqd
1700 r,r' as a write of a value to r, and independent of any
1701 previous value in r */
1702 /* (as opposed to a rite of passage :-) */
1703 addHRegUse(u, HRmWrite, i->Ain.SseReRg.dst);
1704 } else {
1705 addHRegUse(u, HRmRead, i->Ain.SseReRg.src);
1706 addHRegUse(u, i->Ain.SseReRg.op == Asse_MOV
1707 ? HRmWrite : HRmModify,
1708 i->Ain.SseReRg.dst);
1709 }
1710 return;
1711 case Ain_SseCMov:
1712 addHRegUse(u, HRmRead, i->Ain.SseCMov.src);
1713 addHRegUse(u, HRmModify, i->Ain.SseCMov.dst);
1714 return;
1715 case Ain_SseShuf:
1716 addHRegUse(u, HRmRead, i->Ain.SseShuf.src);
1717 addHRegUse(u, HRmWrite, i->Ain.SseShuf.dst);
1718 return;
1719 default:
1720 ppAMD64Instr(i, mode64);
1721 vpanic("getRegUsage_AMD64Instr");
1722 }
1723 }
1724
1725 /* local helper */
mapReg(HRegRemap * m,HReg * r)1726 static inline void mapReg(HRegRemap* m, HReg* r)
1727 {
1728 *r = lookupHRegRemap(m, *r);
1729 }
1730
mapRegs_AMD64Instr(HRegRemap * m,AMD64Instr * i,Bool mode64)1731 void mapRegs_AMD64Instr ( HRegRemap* m, AMD64Instr* i, Bool mode64 )
1732 {
1733 vassert(mode64 == True);
1734 switch (i->tag) {
1735 case Ain_Imm64:
1736 mapReg(m, &i->Ain.Imm64.dst);
1737 return;
1738 case Ain_Alu64R:
1739 mapRegs_AMD64RMI(m, i->Ain.Alu64R.src);
1740 mapReg(m, &i->Ain.Alu64R.dst);
1741 return;
1742 case Ain_Alu64M:
1743 mapRegs_AMD64RI(m, i->Ain.Alu64M.src);
1744 mapRegs_AMD64AMode(m, i->Ain.Alu64M.dst);
1745 return;
1746 case Ain_Sh64:
1747 mapReg(m, &i->Ain.Sh64.dst);
1748 return;
1749 case Ain_Test64:
1750 mapReg(m, &i->Ain.Test64.dst);
1751 return;
1752 case Ain_Unary64:
1753 mapReg(m, &i->Ain.Unary64.dst);
1754 return;
1755 case Ain_Lea64:
1756 mapRegs_AMD64AMode(m, i->Ain.Lea64.am);
1757 mapReg(m, &i->Ain.Lea64.dst);
1758 return;
1759 case Ain_Alu32R:
1760 mapRegs_AMD64RMI(m, i->Ain.Alu32R.src);
1761 mapReg(m, &i->Ain.Alu32R.dst);
1762 return;
1763 case Ain_MulL:
1764 mapRegs_AMD64RM(m, i->Ain.MulL.src);
1765 return;
1766 case Ain_Div:
1767 mapRegs_AMD64RM(m, i->Ain.Div.src);
1768 return;
1769 //.. case Xin_Sh3232:
1770 //.. mapReg(m, &i->Xin.Sh3232.src);
1771 //.. mapReg(m, &i->Xin.Sh3232.dst);
1772 //.. return;
1773 case Ain_Push:
1774 mapRegs_AMD64RMI(m, i->Ain.Push.src);
1775 return;
1776 case Ain_Call:
1777 return;
1778 case Ain_Goto:
1779 mapRegs_AMD64RI(m, i->Ain.Goto.dst);
1780 return;
1781 case Ain_CMov64:
1782 mapRegs_AMD64RM(m, i->Ain.CMov64.src);
1783 mapReg(m, &i->Ain.CMov64.dst);
1784 return;
1785 case Ain_MovxLQ:
1786 mapReg(m, &i->Ain.MovxLQ.src);
1787 mapReg(m, &i->Ain.MovxLQ.dst);
1788 return;
1789 case Ain_LoadEX:
1790 mapRegs_AMD64AMode(m, i->Ain.LoadEX.src);
1791 mapReg(m, &i->Ain.LoadEX.dst);
1792 return;
1793 case Ain_Store:
1794 mapReg(m, &i->Ain.Store.src);
1795 mapRegs_AMD64AMode(m, i->Ain.Store.dst);
1796 return;
1797 case Ain_Set64:
1798 mapReg(m, &i->Ain.Set64.dst);
1799 return;
1800 case Ain_Bsfr64:
1801 mapReg(m, &i->Ain.Bsfr64.src);
1802 mapReg(m, &i->Ain.Bsfr64.dst);
1803 return;
1804 case Ain_MFence:
1805 return;
1806 case Ain_ACAS:
1807 mapRegs_AMD64AMode(m, i->Ain.ACAS.addr);
1808 return;
1809 case Ain_DACAS:
1810 mapRegs_AMD64AMode(m, i->Ain.DACAS.addr);
1811 return;
1812 case Ain_A87Free:
1813 return;
1814 case Ain_A87PushPop:
1815 mapRegs_AMD64AMode(m, i->Ain.A87PushPop.addr);
1816 return;
1817 case Ain_A87FpOp:
1818 return;
1819 case Ain_A87LdCW:
1820 mapRegs_AMD64AMode(m, i->Ain.A87LdCW.addr);
1821 return;
1822 case Ain_A87StSW:
1823 mapRegs_AMD64AMode(m, i->Ain.A87StSW.addr);
1824 return;
1825 //.. case Xin_FpUnary:
1826 //.. mapReg(m, &i->Xin.FpUnary.src);
1827 //.. mapReg(m, &i->Xin.FpUnary.dst);
1828 //.. return;
1829 //.. case Xin_FpBinary:
1830 //.. mapReg(m, &i->Xin.FpBinary.srcL);
1831 //.. mapReg(m, &i->Xin.FpBinary.srcR);
1832 //.. mapReg(m, &i->Xin.FpBinary.dst);
1833 //.. return;
1834 //.. case Xin_FpLdSt:
1835 //.. mapRegs_AMD64AMode(m, i->Xin.FpLdSt.addr);
1836 //.. mapReg(m, &i->Xin.FpLdSt.reg);
1837 //.. return;
1838 //.. case Xin_FpLdStI:
1839 //.. mapRegs_AMD64AMode(m, i->Xin.FpLdStI.addr);
1840 //.. mapReg(m, &i->Xin.FpLdStI.reg);
1841 //.. return;
1842 //.. case Xin_Fp64to32:
1843 //.. mapReg(m, &i->Xin.Fp64to32.src);
1844 //.. mapReg(m, &i->Xin.Fp64to32.dst);
1845 //.. return;
1846 //.. case Xin_FpCMov:
1847 //.. mapReg(m, &i->Xin.FpCMov.src);
1848 //.. mapReg(m, &i->Xin.FpCMov.dst);
1849 //.. return;
1850 case Ain_LdMXCSR:
1851 mapRegs_AMD64AMode(m, i->Ain.LdMXCSR.addr);
1852 return;
1853 //.. case Xin_FpStSW_AX:
1854 //.. return;
1855 case Ain_SseUComIS:
1856 mapReg(m, &i->Ain.SseUComIS.srcL);
1857 mapReg(m, &i->Ain.SseUComIS.srcR);
1858 mapReg(m, &i->Ain.SseUComIS.dst);
1859 return;
1860 case Ain_SseSI2SF:
1861 mapReg(m, &i->Ain.SseSI2SF.src);
1862 mapReg(m, &i->Ain.SseSI2SF.dst);
1863 return;
1864 case Ain_SseSF2SI:
1865 mapReg(m, &i->Ain.SseSF2SI.src);
1866 mapReg(m, &i->Ain.SseSF2SI.dst);
1867 return;
1868 case Ain_SseSDSS:
1869 mapReg(m, &i->Ain.SseSDSS.src);
1870 mapReg(m, &i->Ain.SseSDSS.dst);
1871 return;
1872 //.. case Xin_SseConst:
1873 //.. mapReg(m, &i->Xin.SseConst.dst);
1874 //.. return;
1875 case Ain_SseLdSt:
1876 mapReg(m, &i->Ain.SseLdSt.reg);
1877 mapRegs_AMD64AMode(m, i->Ain.SseLdSt.addr);
1878 break;
1879 case Ain_SseLdzLO:
1880 mapReg(m, &i->Ain.SseLdzLO.reg);
1881 mapRegs_AMD64AMode(m, i->Ain.SseLdzLO.addr);
1882 break;
1883 case Ain_Sse32Fx4:
1884 mapReg(m, &i->Ain.Sse32Fx4.src);
1885 mapReg(m, &i->Ain.Sse32Fx4.dst);
1886 return;
1887 case Ain_Sse32FLo:
1888 mapReg(m, &i->Ain.Sse32FLo.src);
1889 mapReg(m, &i->Ain.Sse32FLo.dst);
1890 return;
1891 case Ain_Sse64Fx2:
1892 mapReg(m, &i->Ain.Sse64Fx2.src);
1893 mapReg(m, &i->Ain.Sse64Fx2.dst);
1894 return;
1895 case Ain_Sse64FLo:
1896 mapReg(m, &i->Ain.Sse64FLo.src);
1897 mapReg(m, &i->Ain.Sse64FLo.dst);
1898 return;
1899 case Ain_SseReRg:
1900 mapReg(m, &i->Ain.SseReRg.src);
1901 mapReg(m, &i->Ain.SseReRg.dst);
1902 return;
1903 case Ain_SseCMov:
1904 mapReg(m, &i->Ain.SseCMov.src);
1905 mapReg(m, &i->Ain.SseCMov.dst);
1906 return;
1907 case Ain_SseShuf:
1908 mapReg(m, &i->Ain.SseShuf.src);
1909 mapReg(m, &i->Ain.SseShuf.dst);
1910 return;
1911 default:
1912 ppAMD64Instr(i, mode64);
1913 vpanic("mapRegs_AMD64Instr");
1914 }
1915 }
1916
1917 /* Figure out if i represents a reg-reg move, and if so assign the
1918 source and destination to *src and *dst. If in doubt say No. Used
1919 by the register allocator to do move coalescing.
1920 */
isMove_AMD64Instr(AMD64Instr * i,HReg * src,HReg * dst)1921 Bool isMove_AMD64Instr ( AMD64Instr* i, HReg* src, HReg* dst )
1922 {
1923 /* Moves between integer regs */
1924 if (i->tag == Ain_Alu64R) {
1925 if (i->Ain.Alu64R.op != Aalu_MOV)
1926 return False;
1927 if (i->Ain.Alu64R.src->tag != Armi_Reg)
1928 return False;
1929 *src = i->Ain.Alu64R.src->Armi.Reg.reg;
1930 *dst = i->Ain.Alu64R.dst;
1931 return True;
1932 }
1933 /* Moves between vector regs */
1934 if (i->tag == Ain_SseReRg) {
1935 if (i->Ain.SseReRg.op != Asse_MOV)
1936 return False;
1937 *src = i->Ain.SseReRg.src;
1938 *dst = i->Ain.SseReRg.dst;
1939 return True;
1940 }
1941 return False;
1942 }
1943
1944
1945 /* Generate amd64 spill/reload instructions under the direction of the
1946 register allocator. Note it's critical these don't write the
1947 condition codes. */
1948
genSpill_AMD64(HInstr ** i1,HInstr ** i2,HReg rreg,Int offsetB,Bool mode64)1949 void genSpill_AMD64 ( /*OUT*/HInstr** i1, /*OUT*/HInstr** i2,
1950 HReg rreg, Int offsetB, Bool mode64 )
1951 {
1952 AMD64AMode* am;
1953 vassert(offsetB >= 0);
1954 vassert(!hregIsVirtual(rreg));
1955 vassert(mode64 == True);
1956 *i1 = *i2 = NULL;
1957 am = AMD64AMode_IR(offsetB, hregAMD64_RBP());
1958 switch (hregClass(rreg)) {
1959 case HRcInt64:
1960 *i1 = AMD64Instr_Alu64M ( Aalu_MOV, AMD64RI_Reg(rreg), am );
1961 return;
1962 case HRcVec128:
1963 *i1 = AMD64Instr_SseLdSt ( False/*store*/, 16, rreg, am );
1964 return;
1965 default:
1966 ppHRegClass(hregClass(rreg));
1967 vpanic("genSpill_AMD64: unimplemented regclass");
1968 }
1969 }
1970
genReload_AMD64(HInstr ** i1,HInstr ** i2,HReg rreg,Int offsetB,Bool mode64)1971 void genReload_AMD64 ( /*OUT*/HInstr** i1, /*OUT*/HInstr** i2,
1972 HReg rreg, Int offsetB, Bool mode64 )
1973 {
1974 AMD64AMode* am;
1975 vassert(offsetB >= 0);
1976 vassert(!hregIsVirtual(rreg));
1977 vassert(mode64 == True);
1978 *i1 = *i2 = NULL;
1979 am = AMD64AMode_IR(offsetB, hregAMD64_RBP());
1980 switch (hregClass(rreg)) {
1981 case HRcInt64:
1982 *i1 = AMD64Instr_Alu64R ( Aalu_MOV, AMD64RMI_Mem(am), rreg );
1983 return;
1984 case HRcVec128:
1985 *i1 = AMD64Instr_SseLdSt ( True/*load*/, 16, rreg, am );
1986 return;
1987 default:
1988 ppHRegClass(hregClass(rreg));
1989 vpanic("genReload_AMD64: unimplemented regclass");
1990 }
1991 }
1992
1993
1994 /* --------- The amd64 assembler (bleh.) --------- */
1995
1996 /* Produce the low three bits of an integer register number. */
iregBits210(HReg r)1997 static UChar iregBits210 ( HReg r )
1998 {
1999 UInt n;
2000 vassert(hregClass(r) == HRcInt64);
2001 vassert(!hregIsVirtual(r));
2002 n = hregNumber(r);
2003 vassert(n <= 15);
2004 return toUChar(n & 7);
2005 }
2006
2007 /* Produce bit 3 of an integer register number. */
iregBit3(HReg r)2008 static UChar iregBit3 ( HReg r )
2009 {
2010 UInt n;
2011 vassert(hregClass(r) == HRcInt64);
2012 vassert(!hregIsVirtual(r));
2013 n = hregNumber(r);
2014 vassert(n <= 15);
2015 return toUChar((n >> 3) & 1);
2016 }
2017
2018 /* Produce a complete 4-bit integer register number. */
iregBits3210(HReg r)2019 static UChar iregBits3210 ( HReg r )
2020 {
2021 UInt n;
2022 vassert(hregClass(r) == HRcInt64);
2023 vassert(!hregIsVirtual(r));
2024 n = hregNumber(r);
2025 vassert(n <= 15);
2026 return toUChar(n);
2027 }
2028
2029 /* Given an xmm (128bit V-class) register number, produce the
2030 equivalent numbered register in 64-bit I-class. This is a bit of
2031 fakery which facilitates using functions that work on integer
2032 register numbers to be used when assembling SSE instructions
2033 too. */
vreg2ireg(HReg r)2034 static UInt vreg2ireg ( HReg r )
2035 {
2036 UInt n;
2037 vassert(hregClass(r) == HRcVec128);
2038 vassert(!hregIsVirtual(r));
2039 n = hregNumber(r);
2040 vassert(n <= 15);
2041 return mkHReg(n, HRcInt64, False);
2042 }
2043
mkModRegRM(UChar mod,UChar reg,UChar regmem)2044 static UChar mkModRegRM ( UChar mod, UChar reg, UChar regmem )
2045 {
2046 return toUChar( ((mod & 3) << 6)
2047 | ((reg & 7) << 3)
2048 | (regmem & 7) );
2049 }
2050
mkSIB(Int shift,Int regindex,Int regbase)2051 static UChar mkSIB ( Int shift, Int regindex, Int regbase )
2052 {
2053 return toUChar( ((shift & 3) << 6)
2054 | ((regindex & 7) << 3)
2055 | (regbase & 7) );
2056 }
2057
emit32(UChar * p,UInt w32)2058 static UChar* emit32 ( UChar* p, UInt w32 )
2059 {
2060 *p++ = toUChar((w32) & 0x000000FF);
2061 *p++ = toUChar((w32 >> 8) & 0x000000FF);
2062 *p++ = toUChar((w32 >> 16) & 0x000000FF);
2063 *p++ = toUChar((w32 >> 24) & 0x000000FF);
2064 return p;
2065 }
2066
emit64(UChar * p,ULong w64)2067 static UChar* emit64 ( UChar* p, ULong w64 )
2068 {
2069 p = emit32(p, toUInt(w64 & 0xFFFFFFFF));
2070 p = emit32(p, toUInt((w64 >> 32) & 0xFFFFFFFF));
2071 return p;
2072 }
2073
2074 /* Does a sign-extend of the lowest 8 bits give
2075 the original number? */
fits8bits(UInt w32)2076 static Bool fits8bits ( UInt w32 )
2077 {
2078 Int i32 = (Int)w32;
2079 return toBool(i32 == ((i32 << 24) >> 24));
2080 }
2081 /* Can the lower 32 bits be signedly widened to produce the whole
2082 64-bit value? In other words, are the top 33 bits either all 0 or
2083 all 1 ? */
fitsIn32Bits(ULong x)2084 static Bool fitsIn32Bits ( ULong x )
2085 {
2086 Long y0 = (Long)x;
2087 Long y1 = y0;
2088 y1 <<= 32;
2089 y1 >>=/*s*/ 32;
2090 return toBool(x == y1);
2091 }
2092
2093
2094 /* Forming mod-reg-rm bytes and scale-index-base bytes.
2095
2096 greg, 0(ereg) | ereg is not any of: RSP RBP R12 R13
2097 = 00 greg ereg
2098
2099 greg, d8(ereg) | ereg is neither of: RSP R12
2100 = 01 greg ereg, d8
2101
2102 greg, d32(ereg) | ereg is neither of: RSP R12
2103 = 10 greg ereg, d32
2104
2105 greg, d8(ereg) | ereg is either: RSP R12
2106 = 01 greg 100, 0x24, d8
2107 (lowest bit of rex distinguishes R12/RSP)
2108
2109 greg, d32(ereg) | ereg is either: RSP R12
2110 = 10 greg 100, 0x24, d32
2111 (lowest bit of rex distinguishes R12/RSP)
2112
2113 -----------------------------------------------
2114
2115 greg, d8(base,index,scale)
2116 | index != RSP
2117 = 01 greg 100, scale index base, d8
2118
2119 greg, d32(base,index,scale)
2120 | index != RSP
2121 = 10 greg 100, scale index base, d32
2122 */
doAMode_M(UChar * p,HReg greg,AMD64AMode * am)2123 static UChar* doAMode_M ( UChar* p, HReg greg, AMD64AMode* am )
2124 {
2125 if (am->tag == Aam_IR) {
2126 if (am->Aam.IR.imm == 0
2127 && am->Aam.IR.reg != hregAMD64_RSP()
2128 && am->Aam.IR.reg != hregAMD64_RBP()
2129 && am->Aam.IR.reg != hregAMD64_R12()
2130 && am->Aam.IR.reg != hregAMD64_R13()
2131 ) {
2132 *p++ = mkModRegRM(0, iregBits210(greg),
2133 iregBits210(am->Aam.IR.reg));
2134 return p;
2135 }
2136 if (fits8bits(am->Aam.IR.imm)
2137 && am->Aam.IR.reg != hregAMD64_RSP()
2138 && am->Aam.IR.reg != hregAMD64_R12()
2139 ) {
2140 *p++ = mkModRegRM(1, iregBits210(greg),
2141 iregBits210(am->Aam.IR.reg));
2142 *p++ = toUChar(am->Aam.IR.imm & 0xFF);
2143 return p;
2144 }
2145 if (am->Aam.IR.reg != hregAMD64_RSP()
2146 && am->Aam.IR.reg != hregAMD64_R12()
2147 ) {
2148 *p++ = mkModRegRM(2, iregBits210(greg),
2149 iregBits210(am->Aam.IR.reg));
2150 p = emit32(p, am->Aam.IR.imm);
2151 return p;
2152 }
2153 if ((am->Aam.IR.reg == hregAMD64_RSP()
2154 || am->Aam.IR.reg == hregAMD64_R12())
2155 && fits8bits(am->Aam.IR.imm)) {
2156 *p++ = mkModRegRM(1, iregBits210(greg), 4);
2157 *p++ = 0x24;
2158 *p++ = toUChar(am->Aam.IR.imm & 0xFF);
2159 return p;
2160 }
2161 if (/* (am->Aam.IR.reg == hregAMD64_RSP()
2162 || wait for test case for RSP case */
2163 am->Aam.IR.reg == hregAMD64_R12()) {
2164 *p++ = mkModRegRM(2, iregBits210(greg), 4);
2165 *p++ = 0x24;
2166 p = emit32(p, am->Aam.IR.imm);
2167 return p;
2168 }
2169 ppAMD64AMode(am);
2170 vpanic("doAMode_M: can't emit amode IR");
2171 /*NOTREACHED*/
2172 }
2173 if (am->tag == Aam_IRRS) {
2174 if (fits8bits(am->Aam.IRRS.imm)
2175 && am->Aam.IRRS.index != hregAMD64_RSP()) {
2176 *p++ = mkModRegRM(1, iregBits210(greg), 4);
2177 *p++ = mkSIB(am->Aam.IRRS.shift, am->Aam.IRRS.index,
2178 am->Aam.IRRS.base);
2179 *p++ = toUChar(am->Aam.IRRS.imm & 0xFF);
2180 return p;
2181 }
2182 if (am->Aam.IRRS.index != hregAMD64_RSP()) {
2183 *p++ = mkModRegRM(2, iregBits210(greg), 4);
2184 *p++ = mkSIB(am->Aam.IRRS.shift, am->Aam.IRRS.index,
2185 am->Aam.IRRS.base);
2186 p = emit32(p, am->Aam.IRRS.imm);
2187 return p;
2188 }
2189 ppAMD64AMode(am);
2190 vpanic("doAMode_M: can't emit amode IRRS");
2191 /*NOTREACHED*/
2192 }
2193 vpanic("doAMode_M: unknown amode");
2194 /*NOTREACHED*/
2195 }
2196
2197
2198 /* Emit a mod-reg-rm byte when the rm bit denotes a reg. */
doAMode_R(UChar * p,HReg greg,HReg ereg)2199 static UChar* doAMode_R ( UChar* p, HReg greg, HReg ereg )
2200 {
2201 *p++ = mkModRegRM(3, iregBits210(greg), iregBits210(ereg));
2202 return p;
2203 }
2204
2205
2206 /* Clear the W bit on a REX byte, thereby changing the operand size
2207 back to whatever that instruction's default operand size is. */
clearWBit(UChar rex)2208 static inline UChar clearWBit ( UChar rex )
2209 {
2210 return toUChar(rex & ~(1<<3));
2211 }
2212
2213
2214 /* Make up a REX byte, with W=1 (size=64), for a (greg,amode) pair. */
rexAMode_M(HReg greg,AMD64AMode * am)2215 static UChar rexAMode_M ( HReg greg, AMD64AMode* am )
2216 {
2217 if (am->tag == Aam_IR) {
2218 UChar W = 1; /* we want 64-bit mode */
2219 UChar R = iregBit3(greg);
2220 UChar X = 0; /* not relevant */
2221 UChar B = iregBit3(am->Aam.IR.reg);
2222 return toUChar(0x40 + ((W << 3) | (R << 2) | (X << 1) | (B << 0)));
2223 }
2224 if (am->tag == Aam_IRRS) {
2225 UChar W = 1; /* we want 64-bit mode */
2226 UChar R = iregBit3(greg);
2227 UChar X = iregBit3(am->Aam.IRRS.index);
2228 UChar B = iregBit3(am->Aam.IRRS.base);
2229 return toUChar(0x40 + ((W << 3) | (R << 2) | (X << 1) | (B << 0)));
2230 }
2231 vassert(0);
2232 return 0; /*NOTREACHED*/
2233 }
2234
2235 /* Make up a REX byte, with W=1 (size=64), for a (greg,ereg) pair. */
rexAMode_R(HReg greg,HReg ereg)2236 static UChar rexAMode_R ( HReg greg, HReg ereg )
2237 {
2238 UChar W = 1; /* we want 64-bit mode */
2239 UChar R = iregBit3(greg);
2240 UChar X = 0; /* not relevant */
2241 UChar B = iregBit3(ereg);
2242 return toUChar(0x40 + ((W << 3) | (R << 2) | (X << 1) | (B << 0)));
2243 }
2244
2245
2246 /* Emit ffree %st(N) */
do_ffree_st(UChar * p,Int n)2247 static UChar* do_ffree_st ( UChar* p, Int n )
2248 {
2249 vassert(n >= 0 && n <= 7);
2250 *p++ = 0xDD;
2251 *p++ = toUChar(0xC0 + n);
2252 return p;
2253 }
2254
2255 //.. /* Emit fstp %st(i), 1 <= i <= 7 */
2256 //.. static UChar* do_fstp_st ( UChar* p, Int i )
2257 //.. {
2258 //.. vassert(1 <= i && i <= 7);
2259 //.. *p++ = 0xDD;
2260 //.. *p++ = 0xD8+i;
2261 //.. return p;
2262 //.. }
2263 //..
2264 //.. /* Emit fld %st(i), 0 <= i <= 6 */
2265 //.. static UChar* do_fld_st ( UChar* p, Int i )
2266 //.. {
2267 //.. vassert(0 <= i && i <= 6);
2268 //.. *p++ = 0xD9;
2269 //.. *p++ = 0xC0+i;
2270 //.. return p;
2271 //.. }
2272 //..
2273 //.. /* Emit f<op> %st(0) */
2274 //.. static UChar* do_fop1_st ( UChar* p, AMD64FpOp op )
2275 //.. {
2276 //.. switch (op) {
2277 //.. case Xfp_NEG: *p++ = 0xD9; *p++ = 0xE0; break;
2278 //.. case Xfp_ABS: *p++ = 0xD9; *p++ = 0xE1; break;
2279 //.. case Xfp_SQRT: *p++ = 0xD9; *p++ = 0xFA; break;
2280 //.. case Xfp_ROUND: *p++ = 0xD9; *p++ = 0xFC; break;
2281 //.. case Xfp_SIN: *p++ = 0xD9; *p++ = 0xFE; break;
2282 //.. case Xfp_COS: *p++ = 0xD9; *p++ = 0xFF; break;
2283 //.. case Xfp_2XM1: *p++ = 0xD9; *p++ = 0xF0; break;
2284 //.. case Xfp_MOV: break;
2285 //.. case Xfp_TAN: p = do_ffree_st7(p); /* since fptan pushes 1.0 */
2286 //.. *p++ = 0xD9; *p++ = 0xF2; /* fptan */
2287 //.. *p++ = 0xD9; *p++ = 0xF7; /* fincstp */
2288 //.. break;
2289 //.. default: vpanic("do_fop1_st: unknown op");
2290 //.. }
2291 //.. return p;
2292 //.. }
2293 //..
2294 //.. /* Emit f<op> %st(i), 1 <= i <= 5 */
2295 //.. static UChar* do_fop2_st ( UChar* p, AMD64FpOp op, Int i )
2296 //.. {
2297 //.. # define fake(_n) mkHReg((_n), HRcInt32, False)
2298 //.. Int subopc;
2299 //.. switch (op) {
2300 //.. case Xfp_ADD: subopc = 0; break;
2301 //.. case Xfp_SUB: subopc = 4; break;
2302 //.. case Xfp_MUL: subopc = 1; break;
2303 //.. case Xfp_DIV: subopc = 6; break;
2304 //.. default: vpanic("do_fop2_st: unknown op");
2305 //.. }
2306 //.. *p++ = 0xD8;
2307 //.. p = doAMode_R(p, fake(subopc), fake(i));
2308 //.. return p;
2309 //.. # undef fake
2310 //.. }
2311 //..
2312 //.. /* Push a 32-bit word on the stack. The word depends on tags[3:0];
2313 //.. each byte is either 0x00 or 0xFF depending on the corresponding bit in tags[].
2314 //.. */
2315 //.. static UChar* push_word_from_tags ( UChar* p, UShort tags )
2316 //.. {
2317 //.. UInt w;
2318 //.. vassert(0 == (tags & ~0xF));
2319 //.. if (tags == 0) {
2320 //.. /* pushl $0x00000000 */
2321 //.. *p++ = 0x6A;
2322 //.. *p++ = 0x00;
2323 //.. }
2324 //.. else
2325 //.. /* pushl $0xFFFFFFFF */
2326 //.. if (tags == 0xF) {
2327 //.. *p++ = 0x6A;
2328 //.. *p++ = 0xFF;
2329 //.. } else {
2330 //.. vassert(0); /* awaiting test case */
2331 //.. w = 0;
2332 //.. if (tags & 1) w |= 0x000000FF;
2333 //.. if (tags & 2) w |= 0x0000FF00;
2334 //.. if (tags & 4) w |= 0x00FF0000;
2335 //.. if (tags & 8) w |= 0xFF000000;
2336 //.. *p++ = 0x68;
2337 //.. p = emit32(p, w);
2338 //.. }
2339 //.. return p;
2340 //.. }
2341
2342 /* Emit an instruction into buf and return the number of bytes used.
2343 Note that buf is not the insn's final place, and therefore it is
2344 imperative to emit position-independent code. */
2345
emit_AMD64Instr(UChar * buf,Int nbuf,AMD64Instr * i,Bool mode64,void * dispatch_unassisted,void * dispatch_assisted)2346 Int emit_AMD64Instr ( UChar* buf, Int nbuf, AMD64Instr* i,
2347 Bool mode64,
2348 void* dispatch_unassisted,
2349 void* dispatch_assisted )
2350 {
2351 UInt /*irno,*/ opc, opc_rr, subopc_imm, opc_imma, opc_cl, opc_imm, subopc;
2352 UInt xtra;
2353 UInt reg;
2354 UChar rex;
2355 UChar* p = &buf[0];
2356 UChar* ptmp;
2357 Int j;
2358 vassert(nbuf >= 32);
2359 vassert(mode64 == True);
2360
2361 /* Wrap an integer as a int register, for use assembling
2362 GrpN insns, in which the greg field is used as a sub-opcode
2363 and does not really contain a register. */
2364 # define fake(_n) mkHReg((_n), HRcInt64, False)
2365
2366 /* vex_printf("asm "); ppAMD64Instr(i, mode64); vex_printf("\n"); */
2367
2368 switch (i->tag) {
2369
2370 case Ain_Imm64:
2371 if (i->Ain.Imm64.imm64 <= 0xFFFFFULL) {
2372 /* Use the short form (load into 32 bit reg, + default
2373 widening rule) for constants under 1 million. We could
2374 use this form for the range 0 to 0x7FFFFFFF inclusive, but
2375 limit it to a smaller range for verifiability purposes. */
2376 if (1 & iregBit3(i->Ain.Imm64.dst))
2377 *p++ = 0x41;
2378 *p++ = 0xB8 + iregBits210(i->Ain.Imm64.dst);
2379 p = emit32(p, (UInt)i->Ain.Imm64.imm64);
2380 } else {
2381 *p++ = toUChar(0x48 + (1 & iregBit3(i->Ain.Imm64.dst)));
2382 *p++ = toUChar(0xB8 + iregBits210(i->Ain.Imm64.dst));
2383 p = emit64(p, i->Ain.Imm64.imm64);
2384 }
2385 goto done;
2386
2387 case Ain_Alu64R:
2388 /* Deal specially with MOV */
2389 if (i->Ain.Alu64R.op == Aalu_MOV) {
2390 switch (i->Ain.Alu64R.src->tag) {
2391 case Armi_Imm:
2392 if (0 == (i->Ain.Alu64R.src->Armi.Imm.imm32 & ~0xFFFFF)) {
2393 /* Actually we could use this form for constants in
2394 the range 0 through 0x7FFFFFFF inclusive, but
2395 limit it to a small range for verifiability
2396 purposes. */
2397 /* Generate "movl $imm32, 32-bit-register" and let
2398 the default zero-extend rule cause the upper half
2399 of the dst to be zeroed out too. This saves 1
2400 and sometimes 2 bytes compared to the more
2401 obvious encoding in the 'else' branch. */
2402 if (1 & iregBit3(i->Ain.Alu64R.dst))
2403 *p++ = 0x41;
2404 *p++ = 0xB8 + iregBits210(i->Ain.Alu64R.dst);
2405 p = emit32(p, i->Ain.Alu64R.src->Armi.Imm.imm32);
2406 } else {
2407 *p++ = toUChar(0x48 + (1 & iregBit3(i->Ain.Alu64R.dst)));
2408 *p++ = 0xC7;
2409 *p++ = toUChar(0xC0 + iregBits210(i->Ain.Alu64R.dst));
2410 p = emit32(p, i->Ain.Alu64R.src->Armi.Imm.imm32);
2411 }
2412 goto done;
2413 case Armi_Reg:
2414 *p++ = rexAMode_R( i->Ain.Alu64R.src->Armi.Reg.reg,
2415 i->Ain.Alu64R.dst );
2416 *p++ = 0x89;
2417 p = doAMode_R(p, i->Ain.Alu64R.src->Armi.Reg.reg,
2418 i->Ain.Alu64R.dst);
2419 goto done;
2420 case Armi_Mem:
2421 *p++ = rexAMode_M(i->Ain.Alu64R.dst,
2422 i->Ain.Alu64R.src->Armi.Mem.am);
2423 *p++ = 0x8B;
2424 p = doAMode_M(p, i->Ain.Alu64R.dst,
2425 i->Ain.Alu64R.src->Armi.Mem.am);
2426 goto done;
2427 default:
2428 goto bad;
2429 }
2430 }
2431 /* MUL */
2432 if (i->Ain.Alu64R.op == Aalu_MUL) {
2433 switch (i->Ain.Alu64R.src->tag) {
2434 case Armi_Reg:
2435 *p++ = rexAMode_R( i->Ain.Alu64R.dst,
2436 i->Ain.Alu64R.src->Armi.Reg.reg);
2437 *p++ = 0x0F;
2438 *p++ = 0xAF;
2439 p = doAMode_R(p, i->Ain.Alu64R.dst,
2440 i->Ain.Alu64R.src->Armi.Reg.reg);
2441 goto done;
2442 case Armi_Mem:
2443 *p++ = rexAMode_M(i->Ain.Alu64R.dst,
2444 i->Ain.Alu64R.src->Armi.Mem.am);
2445 *p++ = 0x0F;
2446 *p++ = 0xAF;
2447 p = doAMode_M(p, i->Ain.Alu64R.dst,
2448 i->Ain.Alu64R.src->Armi.Mem.am);
2449 goto done;
2450 case Armi_Imm:
2451 if (fits8bits(i->Ain.Alu64R.src->Armi.Imm.imm32)) {
2452 *p++ = rexAMode_R(i->Ain.Alu64R.dst, i->Ain.Alu64R.dst);
2453 *p++ = 0x6B;
2454 p = doAMode_R(p, i->Ain.Alu64R.dst, i->Ain.Alu64R.dst);
2455 *p++ = toUChar(0xFF & i->Ain.Alu64R.src->Armi.Imm.imm32);
2456 } else {
2457 *p++ = rexAMode_R(i->Ain.Alu64R.dst, i->Ain.Alu64R.dst);
2458 *p++ = 0x69;
2459 p = doAMode_R(p, i->Ain.Alu64R.dst, i->Ain.Alu64R.dst);
2460 p = emit32(p, i->Ain.Alu64R.src->Armi.Imm.imm32);
2461 }
2462 goto done;
2463 default:
2464 goto bad;
2465 }
2466 }
2467 /* ADD/SUB/ADC/SBB/AND/OR/XOR/CMP */
2468 opc = opc_rr = subopc_imm = opc_imma = 0;
2469 switch (i->Ain.Alu64R.op) {
2470 case Aalu_ADC: opc = 0x13; opc_rr = 0x11;
2471 subopc_imm = 2; opc_imma = 0x15; break;
2472 case Aalu_ADD: opc = 0x03; opc_rr = 0x01;
2473 subopc_imm = 0; opc_imma = 0x05; break;
2474 case Aalu_SUB: opc = 0x2B; opc_rr = 0x29;
2475 subopc_imm = 5; opc_imma = 0x2D; break;
2476 case Aalu_SBB: opc = 0x1B; opc_rr = 0x19;
2477 subopc_imm = 3; opc_imma = 0x1D; break;
2478 case Aalu_AND: opc = 0x23; opc_rr = 0x21;
2479 subopc_imm = 4; opc_imma = 0x25; break;
2480 case Aalu_XOR: opc = 0x33; opc_rr = 0x31;
2481 subopc_imm = 6; opc_imma = 0x35; break;
2482 case Aalu_OR: opc = 0x0B; opc_rr = 0x09;
2483 subopc_imm = 1; opc_imma = 0x0D; break;
2484 case Aalu_CMP: opc = 0x3B; opc_rr = 0x39;
2485 subopc_imm = 7; opc_imma = 0x3D; break;
2486 default: goto bad;
2487 }
2488 switch (i->Ain.Alu64R.src->tag) {
2489 case Armi_Imm:
2490 if (i->Ain.Alu64R.dst == hregAMD64_RAX()
2491 && !fits8bits(i->Ain.Alu64R.src->Armi.Imm.imm32)) {
2492 goto bad; /* FIXME: awaiting test case */
2493 *p++ = toUChar(opc_imma);
2494 p = emit32(p, i->Ain.Alu64R.src->Armi.Imm.imm32);
2495 } else
2496 if (fits8bits(i->Ain.Alu64R.src->Armi.Imm.imm32)) {
2497 *p++ = rexAMode_R( fake(0), i->Ain.Alu64R.dst );
2498 *p++ = 0x83;
2499 p = doAMode_R(p, fake(subopc_imm), i->Ain.Alu64R.dst);
2500 *p++ = toUChar(0xFF & i->Ain.Alu64R.src->Armi.Imm.imm32);
2501 } else {
2502 *p++ = rexAMode_R( fake(0), i->Ain.Alu64R.dst);
2503 *p++ = 0x81;
2504 p = doAMode_R(p, fake(subopc_imm), i->Ain.Alu64R.dst);
2505 p = emit32(p, i->Ain.Alu64R.src->Armi.Imm.imm32);
2506 }
2507 goto done;
2508 case Armi_Reg:
2509 *p++ = rexAMode_R( i->Ain.Alu64R.src->Armi.Reg.reg,
2510 i->Ain.Alu64R.dst);
2511 *p++ = toUChar(opc_rr);
2512 p = doAMode_R(p, i->Ain.Alu64R.src->Armi.Reg.reg,
2513 i->Ain.Alu64R.dst);
2514 goto done;
2515 case Armi_Mem:
2516 *p++ = rexAMode_M( i->Ain.Alu64R.dst,
2517 i->Ain.Alu64R.src->Armi.Mem.am);
2518 *p++ = toUChar(opc);
2519 p = doAMode_M(p, i->Ain.Alu64R.dst,
2520 i->Ain.Alu64R.src->Armi.Mem.am);
2521 goto done;
2522 default:
2523 goto bad;
2524 }
2525 break;
2526
2527 case Ain_Alu64M:
2528 /* Deal specially with MOV */
2529 if (i->Ain.Alu64M.op == Aalu_MOV) {
2530 switch (i->Ain.Alu64M.src->tag) {
2531 case Ari_Reg:
2532 *p++ = rexAMode_M(i->Ain.Alu64M.src->Ari.Reg.reg,
2533 i->Ain.Alu64M.dst);
2534 *p++ = 0x89;
2535 p = doAMode_M(p, i->Ain.Alu64M.src->Ari.Reg.reg,
2536 i->Ain.Alu64M.dst);
2537 goto done;
2538 case Ari_Imm:
2539 *p++ = rexAMode_M(fake(0), i->Ain.Alu64M.dst);
2540 *p++ = 0xC7;
2541 p = doAMode_M(p, fake(0), i->Ain.Alu64M.dst);
2542 p = emit32(p, i->Ain.Alu64M.src->Ari.Imm.imm32);
2543 goto done;
2544 default:
2545 goto bad;
2546 }
2547 }
2548 //.. /* ADD/SUB/ADC/SBB/AND/OR/XOR/CMP. MUL is not
2549 //.. allowed here. */
2550 //.. opc = subopc_imm = opc_imma = 0;
2551 //.. switch (i->Xin.Alu32M.op) {
2552 //.. case Xalu_ADD: opc = 0x01; subopc_imm = 0; break;
2553 //.. case Xalu_SUB: opc = 0x29; subopc_imm = 5; break;
2554 //.. default: goto bad;
2555 //.. }
2556 //.. switch (i->Xin.Alu32M.src->tag) {
2557 //.. case Xri_Reg:
2558 //.. *p++ = opc;
2559 //.. p = doAMode_M(p, i->Xin.Alu32M.src->Xri.Reg.reg,
2560 //.. i->Xin.Alu32M.dst);
2561 //.. goto done;
2562 //.. case Xri_Imm:
2563 //.. if (fits8bits(i->Xin.Alu32M.src->Xri.Imm.imm32)) {
2564 //.. *p++ = 0x83;
2565 //.. p = doAMode_M(p, fake(subopc_imm), i->Xin.Alu32M.dst);
2566 //.. *p++ = 0xFF & i->Xin.Alu32M.src->Xri.Imm.imm32;
2567 //.. goto done;
2568 //.. } else {
2569 //.. *p++ = 0x81;
2570 //.. p = doAMode_M(p, fake(subopc_imm), i->Xin.Alu32M.dst);
2571 //.. p = emit32(p, i->Xin.Alu32M.src->Xri.Imm.imm32);
2572 //.. goto done;
2573 //.. }
2574 //.. default:
2575 //.. goto bad;
2576 //.. }
2577 break;
2578
2579 case Ain_Sh64:
2580 opc_cl = opc_imm = subopc = 0;
2581 switch (i->Ain.Sh64.op) {
2582 case Ash_SHR: opc_cl = 0xD3; opc_imm = 0xC1; subopc = 5; break;
2583 case Ash_SAR: opc_cl = 0xD3; opc_imm = 0xC1; subopc = 7; break;
2584 case Ash_SHL: opc_cl = 0xD3; opc_imm = 0xC1; subopc = 4; break;
2585 default: goto bad;
2586 }
2587 if (i->Ain.Sh64.src == 0) {
2588 *p++ = rexAMode_R(fake(0), i->Ain.Sh64.dst);
2589 *p++ = toUChar(opc_cl);
2590 p = doAMode_R(p, fake(subopc), i->Ain.Sh64.dst);
2591 goto done;
2592 } else {
2593 *p++ = rexAMode_R(fake(0), i->Ain.Sh64.dst);
2594 *p++ = toUChar(opc_imm);
2595 p = doAMode_R(p, fake(subopc), i->Ain.Sh64.dst);
2596 *p++ = (UChar)(i->Ain.Sh64.src);
2597 goto done;
2598 }
2599 break;
2600
2601 case Ain_Test64:
2602 /* testq sign-extend($imm32), %reg */
2603 *p++ = rexAMode_R(fake(0), i->Ain.Test64.dst);
2604 *p++ = 0xF7;
2605 p = doAMode_R(p, fake(0), i->Ain.Test64.dst);
2606 p = emit32(p, i->Ain.Test64.imm32);
2607 goto done;
2608
2609 case Ain_Unary64:
2610 if (i->Ain.Unary64.op == Aun_NOT) {
2611 *p++ = rexAMode_R(fake(0), i->Ain.Unary64.dst);
2612 *p++ = 0xF7;
2613 p = doAMode_R(p, fake(2), i->Ain.Unary64.dst);
2614 goto done;
2615 }
2616 if (i->Ain.Unary64.op == Aun_NEG) {
2617 *p++ = rexAMode_R(fake(0), i->Ain.Unary64.dst);
2618 *p++ = 0xF7;
2619 p = doAMode_R(p, fake(3), i->Ain.Unary64.dst);
2620 goto done;
2621 }
2622 break;
2623
2624 case Ain_Lea64:
2625 *p++ = rexAMode_M(i->Ain.Lea64.dst, i->Ain.Lea64.am);
2626 *p++ = 0x8D;
2627 p = doAMode_M(p, i->Ain.Lea64.dst, i->Ain.Lea64.am);
2628 goto done;
2629
2630 case Ain_Alu32R:
2631 /* ADD/SUB/AND/OR/XOR/CMP */
2632 opc = opc_rr = subopc_imm = opc_imma = 0;
2633 switch (i->Ain.Alu32R.op) {
2634 case Aalu_ADD: opc = 0x03; opc_rr = 0x01;
2635 subopc_imm = 0; opc_imma = 0x05; break;
2636 case Aalu_SUB: opc = 0x2B; opc_rr = 0x29;
2637 subopc_imm = 5; opc_imma = 0x2D; break;
2638 case Aalu_AND: opc = 0x23; opc_rr = 0x21;
2639 subopc_imm = 4; opc_imma = 0x25; break;
2640 case Aalu_XOR: opc = 0x33; opc_rr = 0x31;
2641 subopc_imm = 6; opc_imma = 0x35; break;
2642 case Aalu_OR: opc = 0x0B; opc_rr = 0x09;
2643 subopc_imm = 1; opc_imma = 0x0D; break;
2644 case Aalu_CMP: opc = 0x3B; opc_rr = 0x39;
2645 subopc_imm = 7; opc_imma = 0x3D; break;
2646 default: goto bad;
2647 }
2648 switch (i->Ain.Alu32R.src->tag) {
2649 case Armi_Imm:
2650 if (i->Ain.Alu32R.dst == hregAMD64_RAX()
2651 && !fits8bits(i->Ain.Alu32R.src->Armi.Imm.imm32)) {
2652 goto bad; /* FIXME: awaiting test case */
2653 *p++ = toUChar(opc_imma);
2654 p = emit32(p, i->Ain.Alu32R.src->Armi.Imm.imm32);
2655 } else
2656 if (fits8bits(i->Ain.Alu32R.src->Armi.Imm.imm32)) {
2657 rex = clearWBit( rexAMode_R( fake(0), i->Ain.Alu32R.dst ) );
2658 if (rex != 0x40) *p++ = rex;
2659 *p++ = 0x83;
2660 p = doAMode_R(p, fake(subopc_imm), i->Ain.Alu32R.dst);
2661 *p++ = toUChar(0xFF & i->Ain.Alu32R.src->Armi.Imm.imm32);
2662 } else {
2663 rex = clearWBit( rexAMode_R( fake(0), i->Ain.Alu32R.dst) );
2664 if (rex != 0x40) *p++ = rex;
2665 *p++ = 0x81;
2666 p = doAMode_R(p, fake(subopc_imm), i->Ain.Alu32R.dst);
2667 p = emit32(p, i->Ain.Alu32R.src->Armi.Imm.imm32);
2668 }
2669 goto done;
2670 case Armi_Reg:
2671 rex = clearWBit(
2672 rexAMode_R( i->Ain.Alu32R.src->Armi.Reg.reg,
2673 i->Ain.Alu32R.dst) );
2674 if (rex != 0x40) *p++ = rex;
2675 *p++ = toUChar(opc_rr);
2676 p = doAMode_R(p, i->Ain.Alu32R.src->Armi.Reg.reg,
2677 i->Ain.Alu32R.dst);
2678 goto done;
2679 case Armi_Mem:
2680 rex = clearWBit(
2681 rexAMode_M( i->Ain.Alu32R.dst,
2682 i->Ain.Alu32R.src->Armi.Mem.am) );
2683 if (rex != 0x40) *p++ = rex;
2684 *p++ = toUChar(opc);
2685 p = doAMode_M(p, i->Ain.Alu32R.dst,
2686 i->Ain.Alu32R.src->Armi.Mem.am);
2687 goto done;
2688 default:
2689 goto bad;
2690 }
2691 break;
2692
2693 case Ain_MulL:
2694 subopc = i->Ain.MulL.syned ? 5 : 4;
2695 switch (i->Ain.MulL.src->tag) {
2696 case Arm_Mem:
2697 *p++ = rexAMode_M( fake(0),
2698 i->Ain.MulL.src->Arm.Mem.am);
2699 *p++ = 0xF7;
2700 p = doAMode_M(p, fake(subopc),
2701 i->Ain.MulL.src->Arm.Mem.am);
2702 goto done;
2703 case Arm_Reg:
2704 *p++ = rexAMode_R(fake(0),
2705 i->Ain.MulL.src->Arm.Reg.reg);
2706 *p++ = 0xF7;
2707 p = doAMode_R(p, fake(subopc),
2708 i->Ain.MulL.src->Arm.Reg.reg);
2709 goto done;
2710 default:
2711 goto bad;
2712 }
2713 break;
2714
2715 case Ain_Div:
2716 subopc = i->Ain.Div.syned ? 7 : 6;
2717 if (i->Ain.Div.sz == 4) {
2718 switch (i->Ain.Div.src->tag) {
2719 case Arm_Mem:
2720 goto bad;
2721 /*FIXME*/
2722 *p++ = 0xF7;
2723 p = doAMode_M(p, fake(subopc),
2724 i->Ain.Div.src->Arm.Mem.am);
2725 goto done;
2726 case Arm_Reg:
2727 *p++ = clearWBit(
2728 rexAMode_R( fake(0), i->Ain.Div.src->Arm.Reg.reg));
2729 *p++ = 0xF7;
2730 p = doAMode_R(p, fake(subopc),
2731 i->Ain.Div.src->Arm.Reg.reg);
2732 goto done;
2733 default:
2734 goto bad;
2735 }
2736 }
2737 if (i->Ain.Div.sz == 8) {
2738 switch (i->Ain.Div.src->tag) {
2739 case Arm_Mem:
2740 *p++ = rexAMode_M( fake(0),
2741 i->Ain.Div.src->Arm.Mem.am);
2742 *p++ = 0xF7;
2743 p = doAMode_M(p, fake(subopc),
2744 i->Ain.Div.src->Arm.Mem.am);
2745 goto done;
2746 case Arm_Reg:
2747 *p++ = rexAMode_R( fake(0),
2748 i->Ain.Div.src->Arm.Reg.reg);
2749 *p++ = 0xF7;
2750 p = doAMode_R(p, fake(subopc),
2751 i->Ain.Div.src->Arm.Reg.reg);
2752 goto done;
2753 default:
2754 goto bad;
2755 }
2756 }
2757 break;
2758
2759 //.. case Xin_Sh3232:
2760 //.. vassert(i->Xin.Sh3232.op == Xsh_SHL || i->Xin.Sh3232.op == Xsh_SHR);
2761 //.. if (i->Xin.Sh3232.amt == 0) {
2762 //.. /* shldl/shrdl by %cl */
2763 //.. *p++ = 0x0F;
2764 //.. if (i->Xin.Sh3232.op == Xsh_SHL) {
2765 //.. *p++ = 0xA5;
2766 //.. } else {
2767 //.. *p++ = 0xAD;
2768 //.. }
2769 //.. p = doAMode_R(p, i->Xin.Sh3232.src, i->Xin.Sh3232.dst);
2770 //.. goto done;
2771 //.. }
2772 //.. break;
2773
2774 case Ain_Push:
2775 switch (i->Ain.Push.src->tag) {
2776 case Armi_Mem:
2777 *p++ = clearWBit(
2778 rexAMode_M(fake(0), i->Ain.Push.src->Armi.Mem.am));
2779 *p++ = 0xFF;
2780 p = doAMode_M(p, fake(6), i->Ain.Push.src->Armi.Mem.am);
2781 goto done;
2782 case Armi_Imm:
2783 *p++ = 0x68;
2784 p = emit32(p, i->Ain.Push.src->Armi.Imm.imm32);
2785 goto done;
2786 case Armi_Reg:
2787 *p++ = toUChar(0x40 + (1 & iregBit3(i->Ain.Push.src->Armi.Reg.reg)));
2788 *p++ = toUChar(0x50 + iregBits210(i->Ain.Push.src->Armi.Reg.reg));
2789 goto done;
2790 default:
2791 goto bad;
2792 }
2793
2794 case Ain_Call: {
2795 /* As per detailed comment for Ain_Call in
2796 getRegUsage_AMD64Instr above, %r11 is used as an address
2797 temporary. */
2798 /* jump over the following two insns if the condition does not
2799 hold */
2800 Bool shortImm = fitsIn32Bits(i->Ain.Call.target);
2801 if (i->Ain.Call.cond != Acc_ALWAYS) {
2802 *p++ = toUChar(0x70 + (0xF & (i->Ain.Call.cond ^ 1)));
2803 *p++ = shortImm ? 10 : 13;
2804 /* 10 or 13 bytes in the next two insns */
2805 }
2806 if (shortImm) {
2807 /* 7 bytes: movl sign-extend(imm32), %r11 */
2808 *p++ = 0x49;
2809 *p++ = 0xC7;
2810 *p++ = 0xC3;
2811 p = emit32(p, (UInt)i->Ain.Call.target);
2812 } else {
2813 /* 10 bytes: movabsq $target, %r11 */
2814 *p++ = 0x49;
2815 *p++ = 0xBB;
2816 p = emit64(p, i->Ain.Call.target);
2817 }
2818 /* 3 bytes: call *%r11 */
2819 *p++ = 0x41;
2820 *p++ = 0xFF;
2821 *p++ = 0xD3;
2822 goto done;
2823 }
2824
2825 case Ain_Goto: {
2826 void* dispatch_to_use = NULL;
2827 vassert(dispatch_unassisted != NULL);
2828 vassert(dispatch_assisted != NULL);
2829
2830 /* Use ptmp for backpatching conditional jumps. */
2831 ptmp = NULL;
2832
2833 /* First off, if this is conditional, create a conditional
2834 jump over the rest of it. */
2835 if (i->Ain.Goto.cond != Acc_ALWAYS) {
2836 /* jmp fwds if !condition */
2837 *p++ = toUChar(0x70 + (i->Ain.Goto.cond ^ 1));
2838 ptmp = p; /* fill in this bit later */
2839 *p++ = 0; /* # of bytes to jump over; don't know how many yet. */
2840 }
2841
2842 /* If a non-boring, set %rbp (the guest state pointer)
2843 appropriately. Since these numbers are all small positive
2844 integers, we can get away with "movl $N, %ebp" rather than
2845 the longer "movq $N, %rbp". Also, decide which dispatcher we
2846 need to use. */
2847 dispatch_to_use = dispatch_assisted;
2848
2849 /* movl $magic_number, %ebp */
2850 switch (i->Ain.Goto.jk) {
2851 case Ijk_ClientReq:
2852 *p++ = 0xBD;
2853 p = emit32(p, VEX_TRC_JMP_CLIENTREQ); break;
2854 case Ijk_Sys_syscall:
2855 *p++ = 0xBD;
2856 p = emit32(p, VEX_TRC_JMP_SYS_SYSCALL); break;
2857 case Ijk_Sys_int32:
2858 *p++ = 0xBD;
2859 p = emit32(p, VEX_TRC_JMP_SYS_INT32); break;
2860 case Ijk_Yield:
2861 *p++ = 0xBD;
2862 p = emit32(p, VEX_TRC_JMP_YIELD); break;
2863 case Ijk_YieldNoRedir:
2864 *p++ = 0xBD;
2865 p = emit32(p, VEX_TRC_JMP_YIELD_NOREDIR); break;
2866 case Ijk_EmWarn:
2867 *p++ = 0xBD;
2868 p = emit32(p, VEX_TRC_JMP_EMWARN); break;
2869 case Ijk_MapFail:
2870 *p++ = 0xBD;
2871 p = emit32(p, VEX_TRC_JMP_MAPFAIL); break;
2872 case Ijk_NoDecode:
2873 *p++ = 0xBD;
2874 p = emit32(p, VEX_TRC_JMP_NODECODE); break;
2875 case Ijk_TInval:
2876 *p++ = 0xBD;
2877 p = emit32(p, VEX_TRC_JMP_TINVAL); break;
2878 case Ijk_NoRedir:
2879 *p++ = 0xBD;
2880 p = emit32(p, VEX_TRC_JMP_NOREDIR); break;
2881 case Ijk_SigTRAP:
2882 *p++ = 0xBD;
2883 p = emit32(p, VEX_TRC_JMP_SIGTRAP); break;
2884 case Ijk_SigSEGV:
2885 *p++ = 0xBD;
2886 p = emit32(p, VEX_TRC_JMP_SIGSEGV); break;
2887 case Ijk_Ret:
2888 case Ijk_Call:
2889 case Ijk_Boring:
2890 dispatch_to_use = dispatch_unassisted;
2891 break;
2892 default:
2893 ppIRJumpKind(i->Ain.Goto.jk);
2894 vpanic("emit_AMD64Instr.Ain_Goto: unknown jump kind");
2895 }
2896
2897 /* Get the destination address into %rax */
2898 if (i->Ain.Goto.dst->tag == Ari_Imm) {
2899 /* movl sign-ext($immediate), %rax ; ret */
2900 *p++ = 0x48;
2901 *p++ = 0xC7;
2902 *p++ = 0xC0;
2903 p = emit32(p, i->Ain.Goto.dst->Ari.Imm.imm32);
2904 } else {
2905 vassert(i->Ain.Goto.dst->tag == Ari_Reg);
2906 /* movq %reg, %rax ; ret */
2907 if (i->Ain.Goto.dst->Ari.Reg.reg != hregAMD64_RAX()) {
2908 *p++ = rexAMode_R(i->Ain.Goto.dst->Ari.Reg.reg, hregAMD64_RAX());
2909 *p++ = 0x89;
2910 p = doAMode_R(p, i->Ain.Goto.dst->Ari.Reg.reg, hregAMD64_RAX());
2911 }
2912 }
2913
2914 /* Get the dispatcher address into %rdx. This has to happen
2915 after the load of %rax since %rdx might be carrying the value
2916 destined for %rax immediately prior to this Ain_Goto. */
2917 vassert(sizeof(ULong) == sizeof(void*));
2918
2919 if (fitsIn32Bits(Ptr_to_ULong(dispatch_to_use))) {
2920 /* movl sign-extend(imm32), %rdx */
2921 *p++ = 0x48;
2922 *p++ = 0xC7;
2923 *p++ = 0xC2;
2924 p = emit32(p, (UInt)Ptr_to_ULong(dispatch_to_use));
2925 } else {
2926 /* movabsq $imm64, %rdx */
2927 *p++ = 0x48;
2928 *p++ = 0xBA;
2929 p = emit64(p, Ptr_to_ULong(dispatch_to_use));
2930 }
2931 /* jmp *%rdx */
2932 *p++ = 0xFF;
2933 *p++ = 0xE2;
2934
2935 /* Fix up the conditional jump, if there was one. */
2936 if (i->Ain.Goto.cond != Acc_ALWAYS) {
2937 Int delta = p - ptmp;
2938 vassert(delta > 0 && delta < 30);
2939 *ptmp = toUChar(delta-1);
2940 }
2941 goto done;
2942 }
2943
2944 case Ain_CMov64:
2945 vassert(i->Ain.CMov64.cond != Acc_ALWAYS);
2946 if (i->Ain.CMov64.src->tag == Arm_Reg) {
2947 *p++ = rexAMode_R(i->Ain.CMov64.dst, i->Ain.CMov64.src->Arm.Reg.reg);
2948 *p++ = 0x0F;
2949 *p++ = toUChar(0x40 + (0xF & i->Ain.CMov64.cond));
2950 p = doAMode_R(p, i->Ain.CMov64.dst, i->Ain.CMov64.src->Arm.Reg.reg);
2951 goto done;
2952 }
2953 if (i->Ain.CMov64.src->tag == Arm_Mem) {
2954 *p++ = rexAMode_M(i->Ain.CMov64.dst, i->Ain.CMov64.src->Arm.Mem.am);
2955 *p++ = 0x0F;
2956 *p++ = toUChar(0x40 + (0xF & i->Ain.CMov64.cond));
2957 p = doAMode_M(p, i->Ain.CMov64.dst, i->Ain.CMov64.src->Arm.Mem.am);
2958 goto done;
2959 }
2960 break;
2961
2962 case Ain_MovxLQ:
2963 /* No, _don't_ ask me why the sense of the args has to be
2964 different in the S vs Z case. I don't know. */
2965 if (i->Ain.MovxLQ.syned) {
2966 /* Need REX.W = 1 here, but rexAMode_R does that for us. */
2967 *p++ = rexAMode_R(i->Ain.MovxLQ.dst, i->Ain.MovxLQ.src);
2968 *p++ = 0x63;
2969 p = doAMode_R(p, i->Ain.MovxLQ.dst, i->Ain.MovxLQ.src);
2970 } else {
2971 /* Produce a 32-bit reg-reg move, since the implicit
2972 zero-extend does what we want. */
2973 *p++ = clearWBit (
2974 rexAMode_R(i->Ain.MovxLQ.src, i->Ain.MovxLQ.dst));
2975 *p++ = 0x89;
2976 p = doAMode_R(p, i->Ain.MovxLQ.src, i->Ain.MovxLQ.dst);
2977 }
2978 goto done;
2979
2980 case Ain_LoadEX:
2981 if (i->Ain.LoadEX.szSmall == 1 && !i->Ain.LoadEX.syned) {
2982 /* movzbq */
2983 *p++ = rexAMode_M(i->Ain.LoadEX.dst, i->Ain.LoadEX.src);
2984 *p++ = 0x0F;
2985 *p++ = 0xB6;
2986 p = doAMode_M(p, i->Ain.LoadEX.dst, i->Ain.LoadEX.src);
2987 goto done;
2988 }
2989 if (i->Ain.LoadEX.szSmall == 2 && !i->Ain.LoadEX.syned) {
2990 /* movzwq */
2991 *p++ = rexAMode_M(i->Ain.LoadEX.dst, i->Ain.LoadEX.src);
2992 *p++ = 0x0F;
2993 *p++ = 0xB7;
2994 p = doAMode_M(p, i->Ain.LoadEX.dst, i->Ain.LoadEX.src);
2995 goto done;
2996 }
2997 if (i->Ain.LoadEX.szSmall == 4 && !i->Ain.LoadEX.syned) {
2998 /* movzlq */
2999 /* This isn't really an existing AMD64 instruction per se.
3000 Rather, we have to do a 32-bit load. Because a 32-bit
3001 write implicitly clears the upper 32 bits of the target
3002 register, we get what we want. */
3003 *p++ = clearWBit(
3004 rexAMode_M(i->Ain.LoadEX.dst, i->Ain.LoadEX.src));
3005 *p++ = 0x8B;
3006 p = doAMode_M(p, i->Ain.LoadEX.dst, i->Ain.LoadEX.src);
3007 goto done;
3008 }
3009 break;
3010
3011 case Ain_Set64:
3012 /* Make the destination register be 1 or 0, depending on whether
3013 the relevant condition holds. Complication: the top 56 bits
3014 of the destination should be forced to zero, but doing 'xorq
3015 %r,%r' kills the flag(s) we are about to read. Sigh. So
3016 start off my moving $0 into the dest. */
3017 reg = iregBits3210(i->Ain.Set64.dst);
3018 vassert(reg < 16);
3019
3020 /* movq $0, %dst */
3021 *p++ = toUChar(reg >= 8 ? 0x49 : 0x48);
3022 *p++ = 0xC7;
3023 *p++ = toUChar(0xC0 + (reg & 7));
3024 p = emit32(p, 0);
3025
3026 /* setb lo8(%dst) */
3027 /* note, 8-bit register rex trickyness. Be careful here. */
3028 *p++ = toUChar(reg >= 8 ? 0x41 : 0x40);
3029 *p++ = 0x0F;
3030 *p++ = toUChar(0x90 + (0x0F & i->Ain.Set64.cond));
3031 *p++ = toUChar(0xC0 + (reg & 7));
3032 goto done;
3033
3034 case Ain_Bsfr64:
3035 *p++ = rexAMode_R(i->Ain.Bsfr64.dst, i->Ain.Bsfr64.src);
3036 *p++ = 0x0F;
3037 if (i->Ain.Bsfr64.isFwds) {
3038 *p++ = 0xBC;
3039 } else {
3040 *p++ = 0xBD;
3041 }
3042 p = doAMode_R(p, i->Ain.Bsfr64.dst, i->Ain.Bsfr64.src);
3043 goto done;
3044
3045 case Ain_MFence:
3046 /* mfence */
3047 *p++ = 0x0F; *p++ = 0xAE; *p++ = 0xF0;
3048 goto done;
3049
3050 case Ain_ACAS:
3051 /* lock */
3052 *p++ = 0xF0;
3053 if (i->Ain.ACAS.sz == 2) *p++ = 0x66;
3054 /* cmpxchg{b,w,l,q} %rbx,mem. Expected-value in %rax, new value
3055 in %rbx. The new-value register is hardwired to be %rbx
3056 since dealing with byte integer registers is too much hassle,
3057 so we force the register operand to %rbx (could equally be
3058 %rcx or %rdx). */
3059 rex = rexAMode_M( hregAMD64_RBX(), i->Ain.ACAS.addr );
3060 if (i->Ain.ACAS.sz != 8)
3061 rex = clearWBit(rex);
3062
3063 *p++ = rex; /* this can emit 0x40, which is pointless. oh well. */
3064 *p++ = 0x0F;
3065 if (i->Ain.ACAS.sz == 1) *p++ = 0xB0; else *p++ = 0xB1;
3066 p = doAMode_M(p, hregAMD64_RBX(), i->Ain.ACAS.addr);
3067 goto done;
3068
3069 case Ain_DACAS:
3070 /* lock */
3071 *p++ = 0xF0;
3072 /* cmpxchg{8,16}b m{64,128}. Expected-value in %rdx:%rax, new
3073 value in %rcx:%rbx. All 4 regs are hardwired in the ISA, so
3074 aren't encoded in the insn. */
3075 rex = rexAMode_M( fake(1), i->Ain.ACAS.addr );
3076 if (i->Ain.ACAS.sz != 8)
3077 rex = clearWBit(rex);
3078 *p++ = rex;
3079 *p++ = 0x0F;
3080 *p++ = 0xC7;
3081 p = doAMode_M(p, fake(1), i->Ain.DACAS.addr);
3082 goto done;
3083
3084 case Ain_A87Free:
3085 vassert(i->Ain.A87Free.nregs > 0 && i->Ain.A87Free.nregs <= 7);
3086 for (j = 0; j < i->Ain.A87Free.nregs; j++) {
3087 p = do_ffree_st(p, 7-j);
3088 }
3089 goto done;
3090
3091 case Ain_A87PushPop:
3092 vassert(i->Ain.A87PushPop.szB == 8 || i->Ain.A87PushPop.szB == 4);
3093 if (i->Ain.A87PushPop.isPush) {
3094 /* Load from memory into %st(0): flds/fldl amode */
3095 *p++ = clearWBit(
3096 rexAMode_M(fake(0), i->Ain.A87PushPop.addr) );
3097 *p++ = i->Ain.A87PushPop.szB == 4 ? 0xD9 : 0xDD;
3098 p = doAMode_M(p, fake(0)/*subopcode*/, i->Ain.A87PushPop.addr);
3099 } else {
3100 /* Dump %st(0) to memory: fstps/fstpl amode */
3101 *p++ = clearWBit(
3102 rexAMode_M(fake(3), i->Ain.A87PushPop.addr) );
3103 *p++ = i->Ain.A87PushPop.szB == 4 ? 0xD9 : 0xDD;
3104 p = doAMode_M(p, fake(3)/*subopcode*/, i->Ain.A87PushPop.addr);
3105 goto done;
3106 }
3107 goto done;
3108
3109 case Ain_A87FpOp:
3110 switch (i->Ain.A87FpOp.op) {
3111 case Afp_SQRT: *p++ = 0xD9; *p++ = 0xFA; break;
3112 case Afp_SIN: *p++ = 0xD9; *p++ = 0xFE; break;
3113 case Afp_COS: *p++ = 0xD9; *p++ = 0xFF; break;
3114 case Afp_TAN: *p++ = 0xD9; *p++ = 0xF2; break;
3115 case Afp_ROUND: *p++ = 0xD9; *p++ = 0xFC; break;
3116 case Afp_2XM1: *p++ = 0xD9; *p++ = 0xF0; break;
3117 case Afp_SCALE: *p++ = 0xD9; *p++ = 0xFD; break;
3118 case Afp_ATAN: *p++ = 0xD9; *p++ = 0xF3; break;
3119 case Afp_YL2X: *p++ = 0xD9; *p++ = 0xF1; break;
3120 case Afp_YL2XP1: *p++ = 0xD9; *p++ = 0xF9; break;
3121 case Afp_PREM: *p++ = 0xD9; *p++ = 0xF8; break;
3122 case Afp_PREM1: *p++ = 0xD9; *p++ = 0xF5; break;
3123 default: goto bad;
3124 }
3125 goto done;
3126
3127 case Ain_A87LdCW:
3128 *p++ = clearWBit(
3129 rexAMode_M(fake(5), i->Ain.A87LdCW.addr) );
3130 *p++ = 0xD9;
3131 p = doAMode_M(p, fake(5)/*subopcode*/, i->Ain.A87LdCW.addr);
3132 goto done;
3133
3134 case Ain_A87StSW:
3135 *p++ = clearWBit(
3136 rexAMode_M(fake(7), i->Ain.A87StSW.addr) );
3137 *p++ = 0xDD;
3138 p = doAMode_M(p, fake(7)/*subopcode*/, i->Ain.A87StSW.addr);
3139 goto done;
3140
3141 case Ain_Store:
3142 if (i->Ain.Store.sz == 2) {
3143 /* This just goes to show the crazyness of the instruction
3144 set encoding. We have to insert two prefix bytes, but be
3145 careful to avoid a conflict in what the size should be, by
3146 ensuring that REX.W = 0. */
3147 *p++ = 0x66; /* override to 16-bits */
3148 *p++ = clearWBit( rexAMode_M( i->Ain.Store.src, i->Ain.Store.dst) );
3149 *p++ = 0x89;
3150 p = doAMode_M(p, i->Ain.Store.src, i->Ain.Store.dst);
3151 goto done;
3152 }
3153 if (i->Ain.Store.sz == 4) {
3154 *p++ = clearWBit( rexAMode_M( i->Ain.Store.src, i->Ain.Store.dst) );
3155 *p++ = 0x89;
3156 p = doAMode_M(p, i->Ain.Store.src, i->Ain.Store.dst);
3157 goto done;
3158 }
3159 if (i->Ain.Store.sz == 1) {
3160 /* This is one place where it would be wrong to skip emitting
3161 a rex byte of 0x40, since the mere presence of rex changes
3162 the meaning of the byte register access. Be careful. */
3163 *p++ = clearWBit( rexAMode_M( i->Ain.Store.src, i->Ain.Store.dst) );
3164 *p++ = 0x88;
3165 p = doAMode_M(p, i->Ain.Store.src, i->Ain.Store.dst);
3166 goto done;
3167 }
3168 break;
3169
3170 //.. case Xin_FpUnary:
3171 //.. /* gop %src, %dst
3172 //.. --> ffree %st7 ; fld %st(src) ; fop %st(0) ; fstp %st(1+dst)
3173 //.. */
3174 //.. p = do_ffree_st7(p);
3175 //.. p = do_fld_st(p, 0+hregNumber(i->Xin.FpUnary.src));
3176 //.. p = do_fop1_st(p, i->Xin.FpUnary.op);
3177 //.. p = do_fstp_st(p, 1+hregNumber(i->Xin.FpUnary.dst));
3178 //.. goto done;
3179 //..
3180 //.. case Xin_FpBinary:
3181 //.. if (i->Xin.FpBinary.op == Xfp_YL2X
3182 //.. || i->Xin.FpBinary.op == Xfp_YL2XP1) {
3183 //.. /* Have to do this specially. */
3184 //.. /* ffree %st7 ; fld %st(srcL) ;
3185 //.. ffree %st7 ; fld %st(srcR+1) ; fyl2x{p1} ; fstp(1+dst) */
3186 //.. p = do_ffree_st7(p);
3187 //.. p = do_fld_st(p, 0+hregNumber(i->Xin.FpBinary.srcL));
3188 //.. p = do_ffree_st7(p);
3189 //.. p = do_fld_st(p, 1+hregNumber(i->Xin.FpBinary.srcR));
3190 //.. *p++ = 0xD9;
3191 //.. *p++ = i->Xin.FpBinary.op==Xfp_YL2X ? 0xF1 : 0xF9;
3192 //.. p = do_fstp_st(p, 1+hregNumber(i->Xin.FpBinary.dst));
3193 //.. goto done;
3194 //.. }
3195 //.. if (i->Xin.FpBinary.op == Xfp_ATAN) {
3196 //.. /* Have to do this specially. */
3197 //.. /* ffree %st7 ; fld %st(srcL) ;
3198 //.. ffree %st7 ; fld %st(srcR+1) ; fpatan ; fstp(1+dst) */
3199 //.. p = do_ffree_st7(p);
3200 //.. p = do_fld_st(p, 0+hregNumber(i->Xin.FpBinary.srcL));
3201 //.. p = do_ffree_st7(p);
3202 //.. p = do_fld_st(p, 1+hregNumber(i->Xin.FpBinary.srcR));
3203 //.. *p++ = 0xD9; *p++ = 0xF3;
3204 //.. p = do_fstp_st(p, 1+hregNumber(i->Xin.FpBinary.dst));
3205 //.. goto done;
3206 //.. }
3207 //.. if (i->Xin.FpBinary.op == Xfp_PREM
3208 //.. || i->Xin.FpBinary.op == Xfp_PREM1
3209 //.. || i->Xin.FpBinary.op == Xfp_SCALE) {
3210 //.. /* Have to do this specially. */
3211 //.. /* ffree %st7 ; fld %st(srcR) ;
3212 //.. ffree %st7 ; fld %st(srcL+1) ; fprem/fprem1/fscale ; fstp(2+dst) ;
3213 //.. fincstp ; ffree %st7 */
3214 //.. p = do_ffree_st7(p);
3215 //.. p = do_fld_st(p, 0+hregNumber(i->Xin.FpBinary.srcR));
3216 //.. p = do_ffree_st7(p);
3217 //.. p = do_fld_st(p, 1+hregNumber(i->Xin.FpBinary.srcL));
3218 //.. *p++ = 0xD9;
3219 //.. switch (i->Xin.FpBinary.op) {
3220 //.. case Xfp_PREM: *p++ = 0xF8; break;
3221 //.. case Xfp_PREM1: *p++ = 0xF5; break;
3222 //.. case Xfp_SCALE: *p++ = 0xFD; break;
3223 //.. default: vpanic("emitAMD64Instr(FpBinary,PREM/PREM1/SCALE)");
3224 //.. }
3225 //.. p = do_fstp_st(p, 2+hregNumber(i->Xin.FpBinary.dst));
3226 //.. *p++ = 0xD9; *p++ = 0xF7;
3227 //.. p = do_ffree_st7(p);
3228 //.. goto done;
3229 //.. }
3230 //.. /* General case */
3231 //.. /* gop %srcL, %srcR, %dst
3232 //.. --> ffree %st7 ; fld %st(srcL) ; fop %st(1+srcR) ; fstp %st(1+dst)
3233 //.. */
3234 //.. p = do_ffree_st7(p);
3235 //.. p = do_fld_st(p, 0+hregNumber(i->Xin.FpBinary.srcL));
3236 //.. p = do_fop2_st(p, i->Xin.FpBinary.op,
3237 //.. 1+hregNumber(i->Xin.FpBinary.srcR));
3238 //.. p = do_fstp_st(p, 1+hregNumber(i->Xin.FpBinary.dst));
3239 //.. goto done;
3240 //..
3241 //.. case Xin_FpLdSt:
3242 //.. vassert(i->Xin.FpLdSt.sz == 4 || i->Xin.FpLdSt.sz == 8);
3243 //.. if (i->Xin.FpLdSt.isLoad) {
3244 //.. /* Load from memory into %fakeN.
3245 //.. --> ffree %st(7) ; fld{s/l} amode ; fstp st(N+1)
3246 //.. */
3247 //.. p = do_ffree_st7(p);
3248 //.. *p++ = i->Xin.FpLdSt.sz==4 ? 0xD9 : 0xDD;
3249 //.. p = doAMode_M(p, fake(0)/*subopcode*/, i->Xin.FpLdSt.addr);
3250 //.. p = do_fstp_st(p, 1+hregNumber(i->Xin.FpLdSt.reg));
3251 //.. goto done;
3252 //.. } else {
3253 //.. /* Store from %fakeN into memory.
3254 //.. --> ffree %st(7) ; fld st(N) ; fstp{l|s} amode
3255 //.. */
3256 //.. p = do_ffree_st7(p);
3257 //.. p = do_fld_st(p, 0+hregNumber(i->Xin.FpLdSt.reg));
3258 //.. *p++ = i->Xin.FpLdSt.sz==4 ? 0xD9 : 0xDD;
3259 //.. p = doAMode_M(p, fake(3)/*subopcode*/, i->Xin.FpLdSt.addr);
3260 //.. goto done;
3261 //.. }
3262 //.. break;
3263 //..
3264 //.. case Xin_FpLdStI:
3265 //.. if (i->Xin.FpLdStI.isLoad) {
3266 //.. /* Load from memory into %fakeN, converting from an int.
3267 //.. --> ffree %st(7) ; fild{w/l/ll} amode ; fstp st(N+1)
3268 //.. */
3269 //.. switch (i->Xin.FpLdStI.sz) {
3270 //.. case 8: opc = 0xDF; subopc_imm = 5; break;
3271 //.. case 4: opc = 0xDB; subopc_imm = 0; break;
3272 //.. case 2: vassert(0); opc = 0xDF; subopc_imm = 0; break;
3273 //.. default: vpanic("emitAMD64Instr(Xin_FpLdStI-load)");
3274 //.. }
3275 //.. p = do_ffree_st7(p);
3276 //.. *p++ = opc;
3277 //.. p = doAMode_M(p, fake(subopc_imm)/*subopcode*/, i->Xin.FpLdStI.addr);
3278 //.. p = do_fstp_st(p, 1+hregNumber(i->Xin.FpLdStI.reg));
3279 //.. goto done;
3280 //.. } else {
3281 //.. /* Store from %fakeN into memory, converting to an int.
3282 //.. --> ffree %st(7) ; fld st(N) ; fistp{w/l/ll} amode
3283 //.. */
3284 //.. switch (i->Xin.FpLdStI.sz) {
3285 //.. case 8: opc = 0xDF; subopc_imm = 7; break;
3286 //.. case 4: opc = 0xDB; subopc_imm = 3; break;
3287 //.. case 2: opc = 0xDF; subopc_imm = 3; break;
3288 //.. default: vpanic("emitAMD64Instr(Xin_FpLdStI-store)");
3289 //.. }
3290 //.. p = do_ffree_st7(p);
3291 //.. p = do_fld_st(p, 0+hregNumber(i->Xin.FpLdStI.reg));
3292 //.. *p++ = opc;
3293 //.. p = doAMode_M(p, fake(subopc_imm)/*subopcode*/, i->Xin.FpLdStI.addr);
3294 //.. goto done;
3295 //.. }
3296 //.. break;
3297 //..
3298 //.. case Xin_Fp64to32:
3299 //.. /* ffree %st7 ; fld %st(src) */
3300 //.. p = do_ffree_st7(p);
3301 //.. p = do_fld_st(p, 0+fregNo(i->Xin.Fp64to32.src));
3302 //.. /* subl $4, %esp */
3303 //.. *p++ = 0x83; *p++ = 0xEC; *p++ = 0x04;
3304 //.. /* fstps (%esp) */
3305 //.. *p++ = 0xD9; *p++ = 0x1C; *p++ = 0x24;
3306 //.. /* flds (%esp) */
3307 //.. *p++ = 0xD9; *p++ = 0x04; *p++ = 0x24;
3308 //.. /* addl $4, %esp */
3309 //.. *p++ = 0x83; *p++ = 0xC4; *p++ = 0x04;
3310 //.. /* fstp %st(1+dst) */
3311 //.. p = do_fstp_st(p, 1+fregNo(i->Xin.Fp64to32.dst));
3312 //.. goto done;
3313 //..
3314 //.. case Xin_FpCMov:
3315 //.. /* jmp fwds if !condition */
3316 //.. *p++ = 0x70 + (i->Xin.FpCMov.cond ^ 1);
3317 //.. *p++ = 0; /* # of bytes in the next bit, which we don't know yet */
3318 //.. ptmp = p;
3319 //..
3320 //.. /* ffree %st7 ; fld %st(src) ; fstp %st(1+dst) */
3321 //.. p = do_ffree_st7(p);
3322 //.. p = do_fld_st(p, 0+fregNo(i->Xin.FpCMov.src));
3323 //.. p = do_fstp_st(p, 1+fregNo(i->Xin.FpCMov.dst));
3324 //..
3325 //.. /* Fill in the jump offset. */
3326 //.. *(ptmp-1) = p - ptmp;
3327 //.. goto done;
3328
3329 case Ain_LdMXCSR:
3330 *p++ = clearWBit(rexAMode_M( fake(0), i->Ain.LdMXCSR.addr));
3331 *p++ = 0x0F;
3332 *p++ = 0xAE;
3333 p = doAMode_M(p, fake(2)/*subopcode*/, i->Ain.LdMXCSR.addr);
3334 goto done;
3335
3336 //.. case Xin_FpStSW_AX:
3337 //.. /* note, this emits fnstsw %ax, not fstsw %ax */
3338 //.. *p++ = 0xDF;
3339 //.. *p++ = 0xE0;
3340 //.. goto done;
3341
3342 case Ain_SseUComIS:
3343 /* ucomi[sd] %srcL, %srcR ; pushfq ; popq %dst */
3344 /* ucomi[sd] %srcL, %srcR */
3345 if (i->Ain.SseUComIS.sz == 8) {
3346 *p++ = 0x66;
3347 } else {
3348 goto bad;
3349 vassert(i->Ain.SseUComIS.sz == 4);
3350 }
3351 *p++ = clearWBit (
3352 rexAMode_R( vreg2ireg(i->Ain.SseUComIS.srcL),
3353 vreg2ireg(i->Ain.SseUComIS.srcR) ));
3354 *p++ = 0x0F;
3355 *p++ = 0x2E;
3356 p = doAMode_R(p, vreg2ireg(i->Ain.SseUComIS.srcL),
3357 vreg2ireg(i->Ain.SseUComIS.srcR) );
3358 /* pushfq */
3359 *p++ = 0x9C;
3360 /* popq %dst */
3361 *p++ = toUChar(0x40 + (1 & iregBit3(i->Ain.SseUComIS.dst)));
3362 *p++ = toUChar(0x58 + iregBits210(i->Ain.SseUComIS.dst));
3363 goto done;
3364
3365 case Ain_SseSI2SF:
3366 /* cvssi2s[sd] %src, %dst */
3367 rex = rexAMode_R( vreg2ireg(i->Ain.SseSI2SF.dst),
3368 i->Ain.SseSI2SF.src );
3369 *p++ = toUChar(i->Ain.SseSI2SF.szD==4 ? 0xF3 : 0xF2);
3370 *p++ = toUChar(i->Ain.SseSI2SF.szS==4 ? clearWBit(rex) : rex);
3371 *p++ = 0x0F;
3372 *p++ = 0x2A;
3373 p = doAMode_R( p, vreg2ireg(i->Ain.SseSI2SF.dst),
3374 i->Ain.SseSI2SF.src );
3375 goto done;
3376
3377 case Ain_SseSF2SI:
3378 /* cvss[sd]2si %src, %dst */
3379 rex = rexAMode_R( i->Ain.SseSF2SI.dst,
3380 vreg2ireg(i->Ain.SseSF2SI.src) );
3381 *p++ = toUChar(i->Ain.SseSF2SI.szS==4 ? 0xF3 : 0xF2);
3382 *p++ = toUChar(i->Ain.SseSF2SI.szD==4 ? clearWBit(rex) : rex);
3383 *p++ = 0x0F;
3384 *p++ = 0x2D;
3385 p = doAMode_R( p, i->Ain.SseSF2SI.dst,
3386 vreg2ireg(i->Ain.SseSF2SI.src) );
3387 goto done;
3388
3389 case Ain_SseSDSS:
3390 /* cvtsd2ss/cvtss2sd %src, %dst */
3391 *p++ = toUChar(i->Ain.SseSDSS.from64 ? 0xF2 : 0xF3);
3392 *p++ = clearWBit(
3393 rexAMode_R( vreg2ireg(i->Ain.SseSDSS.dst),
3394 vreg2ireg(i->Ain.SseSDSS.src) ));
3395 *p++ = 0x0F;
3396 *p++ = 0x5A;
3397 p = doAMode_R( p, vreg2ireg(i->Ain.SseSDSS.dst),
3398 vreg2ireg(i->Ain.SseSDSS.src) );
3399 goto done;
3400
3401 //..
3402 //.. case Xin_FpCmp:
3403 //.. /* gcmp %fL, %fR, %dst
3404 //.. -> ffree %st7; fpush %fL ; fucomp %(fR+1) ;
3405 //.. fnstsw %ax ; movl %eax, %dst
3406 //.. */
3407 //.. /* ffree %st7 */
3408 //.. p = do_ffree_st7(p);
3409 //.. /* fpush %fL */
3410 //.. p = do_fld_st(p, 0+fregNo(i->Xin.FpCmp.srcL));
3411 //.. /* fucomp %(fR+1) */
3412 //.. *p++ = 0xDD;
3413 //.. *p++ = 0xE8 + (7 & (1+fregNo(i->Xin.FpCmp.srcR)));
3414 //.. /* fnstsw %ax */
3415 //.. *p++ = 0xDF;
3416 //.. *p++ = 0xE0;
3417 //.. /* movl %eax, %dst */
3418 //.. *p++ = 0x89;
3419 //.. p = doAMode_R(p, hregAMD64_EAX(), i->Xin.FpCmp.dst);
3420 //.. goto done;
3421 //..
3422 //.. case Xin_SseConst: {
3423 //.. UShort con = i->Xin.SseConst.con;
3424 //.. p = push_word_from_tags(p, (con >> 12) & 0xF);
3425 //.. p = push_word_from_tags(p, (con >> 8) & 0xF);
3426 //.. p = push_word_from_tags(p, (con >> 4) & 0xF);
3427 //.. p = push_word_from_tags(p, con & 0xF);
3428 //.. /* movl (%esp), %xmm-dst */
3429 //.. *p++ = 0x0F;
3430 //.. *p++ = 0x10;
3431 //.. *p++ = 0x04 + 8 * (7 & vregNo(i->Xin.SseConst.dst));
3432 //.. *p++ = 0x24;
3433 //.. /* addl $16, %esp */
3434 //.. *p++ = 0x83;
3435 //.. *p++ = 0xC4;
3436 //.. *p++ = 0x10;
3437 //.. goto done;
3438 //.. }
3439
3440 case Ain_SseLdSt:
3441 if (i->Ain.SseLdSt.sz == 8) {
3442 *p++ = 0xF2;
3443 } else
3444 if (i->Ain.SseLdSt.sz == 4) {
3445 *p++ = 0xF3;
3446 } else
3447 if (i->Ain.SseLdSt.sz != 16) {
3448 vassert(0);
3449 }
3450 *p++ = clearWBit(
3451 rexAMode_M( vreg2ireg(i->Ain.SseLdSt.reg), i->Ain.SseLdSt.addr));
3452 *p++ = 0x0F;
3453 *p++ = toUChar(i->Ain.SseLdSt.isLoad ? 0x10 : 0x11);
3454 p = doAMode_M(p, vreg2ireg(i->Ain.SseLdSt.reg), i->Ain.SseLdSt.addr);
3455 goto done;
3456
3457 case Ain_SseLdzLO:
3458 vassert(i->Ain.SseLdzLO.sz == 4 || i->Ain.SseLdzLO.sz == 8);
3459 /* movs[sd] amode, %xmm-dst */
3460 *p++ = toUChar(i->Ain.SseLdzLO.sz==4 ? 0xF3 : 0xF2);
3461 *p++ = clearWBit(
3462 rexAMode_M(vreg2ireg(i->Ain.SseLdzLO.reg),
3463 i->Ain.SseLdzLO.addr));
3464 *p++ = 0x0F;
3465 *p++ = 0x10;
3466 p = doAMode_M(p, vreg2ireg(i->Ain.SseLdzLO.reg),
3467 i->Ain.SseLdzLO.addr);
3468 goto done;
3469
3470 case Ain_Sse32Fx4:
3471 xtra = 0;
3472 *p++ = clearWBit(
3473 rexAMode_R( vreg2ireg(i->Ain.Sse32Fx4.dst),
3474 vreg2ireg(i->Ain.Sse32Fx4.src) ));
3475 *p++ = 0x0F;
3476 switch (i->Ain.Sse32Fx4.op) {
3477 case Asse_ADDF: *p++ = 0x58; break;
3478 case Asse_DIVF: *p++ = 0x5E; break;
3479 case Asse_MAXF: *p++ = 0x5F; break;
3480 case Asse_MINF: *p++ = 0x5D; break;
3481 case Asse_MULF: *p++ = 0x59; break;
3482 case Asse_RCPF: *p++ = 0x53; break;
3483 case Asse_RSQRTF: *p++ = 0x52; break;
3484 case Asse_SQRTF: *p++ = 0x51; break;
3485 case Asse_SUBF: *p++ = 0x5C; break;
3486 case Asse_CMPEQF: *p++ = 0xC2; xtra = 0x100; break;
3487 case Asse_CMPLTF: *p++ = 0xC2; xtra = 0x101; break;
3488 case Asse_CMPLEF: *p++ = 0xC2; xtra = 0x102; break;
3489 case Asse_CMPUNF: *p++ = 0xC2; xtra = 0x103; break;
3490 default: goto bad;
3491 }
3492 p = doAMode_R(p, vreg2ireg(i->Ain.Sse32Fx4.dst),
3493 vreg2ireg(i->Ain.Sse32Fx4.src) );
3494 if (xtra & 0x100)
3495 *p++ = toUChar(xtra & 0xFF);
3496 goto done;
3497
3498 case Ain_Sse64Fx2:
3499 xtra = 0;
3500 *p++ = 0x66;
3501 *p++ = clearWBit(
3502 rexAMode_R( vreg2ireg(i->Ain.Sse64Fx2.dst),
3503 vreg2ireg(i->Ain.Sse64Fx2.src) ));
3504 *p++ = 0x0F;
3505 switch (i->Ain.Sse64Fx2.op) {
3506 case Asse_ADDF: *p++ = 0x58; break;
3507 case Asse_DIVF: *p++ = 0x5E; break;
3508 case Asse_MAXF: *p++ = 0x5F; break;
3509 case Asse_MINF: *p++ = 0x5D; break;
3510 case Asse_MULF: *p++ = 0x59; break;
3511 //.. case Xsse_RCPF: *p++ = 0x53; break;
3512 //.. case Xsse_RSQRTF: *p++ = 0x52; break;
3513 case Asse_SQRTF: *p++ = 0x51; break;
3514 case Asse_SUBF: *p++ = 0x5C; break;
3515 case Asse_CMPEQF: *p++ = 0xC2; xtra = 0x100; break;
3516 case Asse_CMPLTF: *p++ = 0xC2; xtra = 0x101; break;
3517 case Asse_CMPLEF: *p++ = 0xC2; xtra = 0x102; break;
3518 case Asse_CMPUNF: *p++ = 0xC2; xtra = 0x103; break;
3519 default: goto bad;
3520 }
3521 p = doAMode_R(p, vreg2ireg(i->Ain.Sse64Fx2.dst),
3522 vreg2ireg(i->Ain.Sse64Fx2.src) );
3523 if (xtra & 0x100)
3524 *p++ = toUChar(xtra & 0xFF);
3525 goto done;
3526
3527 case Ain_Sse32FLo:
3528 xtra = 0;
3529 *p++ = 0xF3;
3530 *p++ = clearWBit(
3531 rexAMode_R( vreg2ireg(i->Ain.Sse32FLo.dst),
3532 vreg2ireg(i->Ain.Sse32FLo.src) ));
3533 *p++ = 0x0F;
3534 switch (i->Ain.Sse32FLo.op) {
3535 case Asse_ADDF: *p++ = 0x58; break;
3536 case Asse_DIVF: *p++ = 0x5E; break;
3537 case Asse_MAXF: *p++ = 0x5F; break;
3538 case Asse_MINF: *p++ = 0x5D; break;
3539 case Asse_MULF: *p++ = 0x59; break;
3540 case Asse_RCPF: *p++ = 0x53; break;
3541 case Asse_RSQRTF: *p++ = 0x52; break;
3542 case Asse_SQRTF: *p++ = 0x51; break;
3543 case Asse_SUBF: *p++ = 0x5C; break;
3544 case Asse_CMPEQF: *p++ = 0xC2; xtra = 0x100; break;
3545 case Asse_CMPLTF: *p++ = 0xC2; xtra = 0x101; break;
3546 case Asse_CMPLEF: *p++ = 0xC2; xtra = 0x102; break;
3547 case Asse_CMPUNF: *p++ = 0xC2; xtra = 0x103; break;
3548 default: goto bad;
3549 }
3550 p = doAMode_R(p, vreg2ireg(i->Ain.Sse32FLo.dst),
3551 vreg2ireg(i->Ain.Sse32FLo.src) );
3552 if (xtra & 0x100)
3553 *p++ = toUChar(xtra & 0xFF);
3554 goto done;
3555
3556 case Ain_Sse64FLo:
3557 xtra = 0;
3558 *p++ = 0xF2;
3559 *p++ = clearWBit(
3560 rexAMode_R( vreg2ireg(i->Ain.Sse64FLo.dst),
3561 vreg2ireg(i->Ain.Sse64FLo.src) ));
3562 *p++ = 0x0F;
3563 switch (i->Ain.Sse64FLo.op) {
3564 case Asse_ADDF: *p++ = 0x58; break;
3565 case Asse_DIVF: *p++ = 0x5E; break;
3566 case Asse_MAXF: *p++ = 0x5F; break;
3567 case Asse_MINF: *p++ = 0x5D; break;
3568 case Asse_MULF: *p++ = 0x59; break;
3569 //.. case Xsse_RCPF: *p++ = 0x53; break;
3570 //.. case Xsse_RSQRTF: *p++ = 0x52; break;
3571 case Asse_SQRTF: *p++ = 0x51; break;
3572 case Asse_SUBF: *p++ = 0x5C; break;
3573 case Asse_CMPEQF: *p++ = 0xC2; xtra = 0x100; break;
3574 case Asse_CMPLTF: *p++ = 0xC2; xtra = 0x101; break;
3575 case Asse_CMPLEF: *p++ = 0xC2; xtra = 0x102; break;
3576 case Asse_CMPUNF: *p++ = 0xC2; xtra = 0x103; break;
3577 default: goto bad;
3578 }
3579 p = doAMode_R(p, vreg2ireg(i->Ain.Sse64FLo.dst),
3580 vreg2ireg(i->Ain.Sse64FLo.src) );
3581 if (xtra & 0x100)
3582 *p++ = toUChar(xtra & 0xFF);
3583 goto done;
3584
3585 case Ain_SseReRg:
3586 # define XX(_n) *p++ = (_n)
3587
3588 rex = clearWBit(
3589 rexAMode_R( vreg2ireg(i->Ain.SseReRg.dst),
3590 vreg2ireg(i->Ain.SseReRg.src) ));
3591
3592 switch (i->Ain.SseReRg.op) {
3593 case Asse_MOV: /*movups*/ XX(rex); XX(0x0F); XX(0x10); break;
3594 case Asse_OR: XX(rex); XX(0x0F); XX(0x56); break;
3595 case Asse_XOR: XX(rex); XX(0x0F); XX(0x57); break;
3596 case Asse_AND: XX(rex); XX(0x0F); XX(0x54); break;
3597 case Asse_ANDN: XX(rex); XX(0x0F); XX(0x55); break;
3598 case Asse_PACKSSD: XX(0x66); XX(rex); XX(0x0F); XX(0x6B); break;
3599 case Asse_PACKSSW: XX(0x66); XX(rex); XX(0x0F); XX(0x63); break;
3600 case Asse_PACKUSW: XX(0x66); XX(rex); XX(0x0F); XX(0x67); break;
3601 case Asse_ADD8: XX(0x66); XX(rex); XX(0x0F); XX(0xFC); break;
3602 case Asse_ADD16: XX(0x66); XX(rex); XX(0x0F); XX(0xFD); break;
3603 case Asse_ADD32: XX(0x66); XX(rex); XX(0x0F); XX(0xFE); break;
3604 case Asse_ADD64: XX(0x66); XX(rex); XX(0x0F); XX(0xD4); break;
3605 case Asse_QADD8S: XX(0x66); XX(rex); XX(0x0F); XX(0xEC); break;
3606 case Asse_QADD16S: XX(0x66); XX(rex); XX(0x0F); XX(0xED); break;
3607 case Asse_QADD8U: XX(0x66); XX(rex); XX(0x0F); XX(0xDC); break;
3608 case Asse_QADD16U: XX(0x66); XX(rex); XX(0x0F); XX(0xDD); break;
3609 case Asse_AVG8U: XX(0x66); XX(rex); XX(0x0F); XX(0xE0); break;
3610 case Asse_AVG16U: XX(0x66); XX(rex); XX(0x0F); XX(0xE3); break;
3611 case Asse_CMPEQ8: XX(0x66); XX(rex); XX(0x0F); XX(0x74); break;
3612 case Asse_CMPEQ16: XX(0x66); XX(rex); XX(0x0F); XX(0x75); break;
3613 case Asse_CMPEQ32: XX(0x66); XX(rex); XX(0x0F); XX(0x76); break;
3614 case Asse_CMPGT8S: XX(0x66); XX(rex); XX(0x0F); XX(0x64); break;
3615 case Asse_CMPGT16S: XX(0x66); XX(rex); XX(0x0F); XX(0x65); break;
3616 case Asse_CMPGT32S: XX(0x66); XX(rex); XX(0x0F); XX(0x66); break;
3617 case Asse_MAX16S: XX(0x66); XX(rex); XX(0x0F); XX(0xEE); break;
3618 case Asse_MAX8U: XX(0x66); XX(rex); XX(0x0F); XX(0xDE); break;
3619 case Asse_MIN16S: XX(0x66); XX(rex); XX(0x0F); XX(0xEA); break;
3620 case Asse_MIN8U: XX(0x66); XX(rex); XX(0x0F); XX(0xDA); break;
3621 case Asse_MULHI16U: XX(0x66); XX(rex); XX(0x0F); XX(0xE4); break;
3622 case Asse_MULHI16S: XX(0x66); XX(rex); XX(0x0F); XX(0xE5); break;
3623 case Asse_MUL16: XX(0x66); XX(rex); XX(0x0F); XX(0xD5); break;
3624 case Asse_SHL16: XX(0x66); XX(rex); XX(0x0F); XX(0xF1); break;
3625 case Asse_SHL32: XX(0x66); XX(rex); XX(0x0F); XX(0xF2); break;
3626 case Asse_SHL64: XX(0x66); XX(rex); XX(0x0F); XX(0xF3); break;
3627 case Asse_SAR16: XX(0x66); XX(rex); XX(0x0F); XX(0xE1); break;
3628 case Asse_SAR32: XX(0x66); XX(rex); XX(0x0F); XX(0xE2); break;
3629 case Asse_SHR16: XX(0x66); XX(rex); XX(0x0F); XX(0xD1); break;
3630 case Asse_SHR32: XX(0x66); XX(rex); XX(0x0F); XX(0xD2); break;
3631 case Asse_SHR64: XX(0x66); XX(rex); XX(0x0F); XX(0xD3); break;
3632 case Asse_SUB8: XX(0x66); XX(rex); XX(0x0F); XX(0xF8); break;
3633 case Asse_SUB16: XX(0x66); XX(rex); XX(0x0F); XX(0xF9); break;
3634 case Asse_SUB32: XX(0x66); XX(rex); XX(0x0F); XX(0xFA); break;
3635 case Asse_SUB64: XX(0x66); XX(rex); XX(0x0F); XX(0xFB); break;
3636 case Asse_QSUB8S: XX(0x66); XX(rex); XX(0x0F); XX(0xE8); break;
3637 case Asse_QSUB16S: XX(0x66); XX(rex); XX(0x0F); XX(0xE9); break;
3638 case Asse_QSUB8U: XX(0x66); XX(rex); XX(0x0F); XX(0xD8); break;
3639 case Asse_QSUB16U: XX(0x66); XX(rex); XX(0x0F); XX(0xD9); break;
3640 case Asse_UNPCKHB: XX(0x66); XX(rex); XX(0x0F); XX(0x68); break;
3641 case Asse_UNPCKHW: XX(0x66); XX(rex); XX(0x0F); XX(0x69); break;
3642 case Asse_UNPCKHD: XX(0x66); XX(rex); XX(0x0F); XX(0x6A); break;
3643 case Asse_UNPCKHQ: XX(0x66); XX(rex); XX(0x0F); XX(0x6D); break;
3644 case Asse_UNPCKLB: XX(0x66); XX(rex); XX(0x0F); XX(0x60); break;
3645 case Asse_UNPCKLW: XX(0x66); XX(rex); XX(0x0F); XX(0x61); break;
3646 case Asse_UNPCKLD: XX(0x66); XX(rex); XX(0x0F); XX(0x62); break;
3647 case Asse_UNPCKLQ: XX(0x66); XX(rex); XX(0x0F); XX(0x6C); break;
3648 default: goto bad;
3649 }
3650 p = doAMode_R(p, vreg2ireg(i->Ain.SseReRg.dst),
3651 vreg2ireg(i->Ain.SseReRg.src) );
3652 # undef XX
3653 goto done;
3654
3655 case Ain_SseCMov:
3656 /* jmp fwds if !condition */
3657 *p++ = toUChar(0x70 + (i->Ain.SseCMov.cond ^ 1));
3658 *p++ = 0; /* # of bytes in the next bit, which we don't know yet */
3659 ptmp = p;
3660
3661 /* movaps %src, %dst */
3662 *p++ = clearWBit(
3663 rexAMode_R( vreg2ireg(i->Ain.SseCMov.dst),
3664 vreg2ireg(i->Ain.SseCMov.src) ));
3665 *p++ = 0x0F;
3666 *p++ = 0x28;
3667 p = doAMode_R(p, vreg2ireg(i->Ain.SseCMov.dst),
3668 vreg2ireg(i->Ain.SseCMov.src) );
3669
3670 /* Fill in the jump offset. */
3671 *(ptmp-1) = toUChar(p - ptmp);
3672 goto done;
3673
3674 case Ain_SseShuf:
3675 *p++ = 0x66;
3676 *p++ = clearWBit(
3677 rexAMode_R( vreg2ireg(i->Ain.SseShuf.dst),
3678 vreg2ireg(i->Ain.SseShuf.src) ));
3679 *p++ = 0x0F;
3680 *p++ = 0x70;
3681 p = doAMode_R(p, vreg2ireg(i->Ain.SseShuf.dst),
3682 vreg2ireg(i->Ain.SseShuf.src) );
3683 *p++ = (UChar)(i->Ain.SseShuf.order);
3684 goto done;
3685
3686 default:
3687 goto bad;
3688 }
3689
3690 bad:
3691 ppAMD64Instr(i, mode64);
3692 vpanic("emit_AMD64Instr");
3693 /*NOTREACHED*/
3694
3695 done:
3696 vassert(p - &buf[0] <= 32);
3697 return p - &buf[0];
3698
3699 # undef fake
3700 }
3701
3702 /*---------------------------------------------------------------*/
3703 /*--- end host_amd64_defs.c ---*/
3704 /*---------------------------------------------------------------*/
3705