blob: 21a100f59a6bc903be92fd1bbcb93f15baab329c [file] [log] [blame]
Ben Murdoche0cee9b2011-05-25 10:26:03 +01001// Copyright 2011 the V8 project authors. All rights reserved.
Steve Blocka7e24c12009-10-30 11:49:00 +00002// Redistribution and use in source and binary forms, with or without
3// modification, are permitted provided that the following conditions are
4// met:
5//
6// * Redistributions of source code must retain the above copyright
7// notice, this list of conditions and the following disclaimer.
8// * Redistributions in binary form must reproduce the above
9// copyright notice, this list of conditions and the following
10// disclaimer in the documentation and/or other materials provided
11// with the distribution.
12// * Neither the name of Google Inc. nor the names of its
13// contributors may be used to endorse or promote products derived
14// from this software without specific prior written permission.
15//
16// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28#include <assert.h>
29#include <stdio.h>
30#include <stdarg.h>
31
32#include "v8.h"
Leon Clarkef7060e22010-06-03 12:02:55 +010033
34#if defined(V8_TARGET_ARCH_X64)
35
Steve Blocka7e24c12009-10-30 11:49:00 +000036#include "disasm.h"
37
38namespace disasm {
39
40enum OperandType {
41 UNSET_OP_ORDER = 0,
42 // Operand size decides between 16, 32 and 64 bit operands.
43 REG_OPER_OP_ORDER = 1, // Register destination, operand source.
44 OPER_REG_OP_ORDER = 2, // Operand destination, register source.
45 // Fixed 8-bit operands.
46 BYTE_SIZE_OPERAND_FLAG = 4,
47 BYTE_REG_OPER_OP_ORDER = REG_OPER_OP_ORDER | BYTE_SIZE_OPERAND_FLAG,
48 BYTE_OPER_REG_OP_ORDER = OPER_REG_OP_ORDER | BYTE_SIZE_OPERAND_FLAG
49};
50
51//------------------------------------------------------------------
52// Tables
53//------------------------------------------------------------------
54struct ByteMnemonic {
55 int b; // -1 terminates, otherwise must be in range (0..255)
56 OperandType op_order_;
57 const char* mnem;
58};
59
60
61static ByteMnemonic two_operands_instr[] = {
62 { 0x00, BYTE_OPER_REG_OP_ORDER, "add" },
63 { 0x01, OPER_REG_OP_ORDER, "add" },
64 { 0x02, BYTE_REG_OPER_OP_ORDER, "add" },
65 { 0x03, REG_OPER_OP_ORDER, "add" },
66 { 0x08, BYTE_OPER_REG_OP_ORDER, "or" },
67 { 0x09, OPER_REG_OP_ORDER, "or" },
68 { 0x0A, BYTE_REG_OPER_OP_ORDER, "or" },
69 { 0x0B, REG_OPER_OP_ORDER, "or" },
70 { 0x10, BYTE_OPER_REG_OP_ORDER, "adc" },
71 { 0x11, OPER_REG_OP_ORDER, "adc" },
72 { 0x12, BYTE_REG_OPER_OP_ORDER, "adc" },
73 { 0x13, REG_OPER_OP_ORDER, "adc" },
74 { 0x18, BYTE_OPER_REG_OP_ORDER, "sbb" },
75 { 0x19, OPER_REG_OP_ORDER, "sbb" },
76 { 0x1A, BYTE_REG_OPER_OP_ORDER, "sbb" },
77 { 0x1B, REG_OPER_OP_ORDER, "sbb" },
78 { 0x20, BYTE_OPER_REG_OP_ORDER, "and" },
79 { 0x21, OPER_REG_OP_ORDER, "and" },
80 { 0x22, BYTE_REG_OPER_OP_ORDER, "and" },
81 { 0x23, REG_OPER_OP_ORDER, "and" },
82 { 0x28, BYTE_OPER_REG_OP_ORDER, "sub" },
83 { 0x29, OPER_REG_OP_ORDER, "sub" },
84 { 0x2A, BYTE_REG_OPER_OP_ORDER, "sub" },
85 { 0x2B, REG_OPER_OP_ORDER, "sub" },
86 { 0x30, BYTE_OPER_REG_OP_ORDER, "xor" },
87 { 0x31, OPER_REG_OP_ORDER, "xor" },
88 { 0x32, BYTE_REG_OPER_OP_ORDER, "xor" },
89 { 0x33, REG_OPER_OP_ORDER, "xor" },
90 { 0x38, BYTE_OPER_REG_OP_ORDER, "cmp" },
91 { 0x39, OPER_REG_OP_ORDER, "cmp" },
92 { 0x3A, BYTE_REG_OPER_OP_ORDER, "cmp" },
93 { 0x3B, REG_OPER_OP_ORDER, "cmp" },
94 { 0x63, REG_OPER_OP_ORDER, "movsxlq" },
95 { 0x84, BYTE_REG_OPER_OP_ORDER, "test" },
96 { 0x85, REG_OPER_OP_ORDER, "test" },
97 { 0x86, BYTE_REG_OPER_OP_ORDER, "xchg" },
98 { 0x87, REG_OPER_OP_ORDER, "xchg" },
99 { 0x88, BYTE_OPER_REG_OP_ORDER, "mov" },
100 { 0x89, OPER_REG_OP_ORDER, "mov" },
101 { 0x8A, BYTE_REG_OPER_OP_ORDER, "mov" },
102 { 0x8B, REG_OPER_OP_ORDER, "mov" },
103 { 0x8D, REG_OPER_OP_ORDER, "lea" },
104 { -1, UNSET_OP_ORDER, "" }
105};
106
107
108static ByteMnemonic zero_operands_instr[] = {
109 { 0xC3, UNSET_OP_ORDER, "ret" },
110 { 0xC9, UNSET_OP_ORDER, "leave" },
111 { 0xF4, UNSET_OP_ORDER, "hlt" },
112 { 0xCC, UNSET_OP_ORDER, "int3" },
113 { 0x60, UNSET_OP_ORDER, "pushad" },
114 { 0x61, UNSET_OP_ORDER, "popad" },
115 { 0x9C, UNSET_OP_ORDER, "pushfd" },
116 { 0x9D, UNSET_OP_ORDER, "popfd" },
117 { 0x9E, UNSET_OP_ORDER, "sahf" },
118 { 0x99, UNSET_OP_ORDER, "cdq" },
119 { 0x9B, UNSET_OP_ORDER, "fwait" },
Leon Clarked91b9f72010-01-27 17:25:45 +0000120 { 0xA4, UNSET_OP_ORDER, "movs" },
121 { 0xA5, UNSET_OP_ORDER, "movs" },
122 { 0xA6, UNSET_OP_ORDER, "cmps" },
123 { 0xA7, UNSET_OP_ORDER, "cmps" },
Steve Blocka7e24c12009-10-30 11:49:00 +0000124 { -1, UNSET_OP_ORDER, "" }
125};
126
127
128static ByteMnemonic call_jump_instr[] = {
129 { 0xE8, UNSET_OP_ORDER, "call" },
130 { 0xE9, UNSET_OP_ORDER, "jmp" },
131 { -1, UNSET_OP_ORDER, "" }
132};
133
134
135static ByteMnemonic short_immediate_instr[] = {
136 { 0x05, UNSET_OP_ORDER, "add" },
137 { 0x0D, UNSET_OP_ORDER, "or" },
138 { 0x15, UNSET_OP_ORDER, "adc" },
139 { 0x1D, UNSET_OP_ORDER, "sbb" },
140 { 0x25, UNSET_OP_ORDER, "and" },
141 { 0x2D, UNSET_OP_ORDER, "sub" },
142 { 0x35, UNSET_OP_ORDER, "xor" },
143 { 0x3D, UNSET_OP_ORDER, "cmp" },
144 { -1, UNSET_OP_ORDER, "" }
145};
146
147
148static const char* conditional_code_suffix[] = {
149 "o", "no", "c", "nc", "z", "nz", "na", "a",
150 "s", "ns", "pe", "po", "l", "ge", "le", "g"
151};
152
153
154enum InstructionType {
155 NO_INSTR,
156 ZERO_OPERANDS_INSTR,
157 TWO_OPERANDS_INSTR,
158 JUMP_CONDITIONAL_SHORT_INSTR,
159 REGISTER_INSTR,
160 PUSHPOP_INSTR, // Has implicit 64-bit operand size.
161 MOVE_REG_INSTR,
162 CALL_JUMP_INSTR,
163 SHORT_IMMEDIATE_INSTR
164};
165
166
Leon Clarked91b9f72010-01-27 17:25:45 +0000167enum Prefixes {
168 ESCAPE_PREFIX = 0x0F,
169 OPERAND_SIZE_OVERRIDE_PREFIX = 0x66,
170 ADDRESS_SIZE_OVERRIDE_PREFIX = 0x67,
171 REPNE_PREFIX = 0xF2,
172 REP_PREFIX = 0xF3,
173 REPEQ_PREFIX = REP_PREFIX
174};
175
176
Steve Blocka7e24c12009-10-30 11:49:00 +0000177struct InstructionDesc {
178 const char* mnem;
179 InstructionType type;
180 OperandType op_order_;
181 bool byte_size_operation; // Fixed 8-bit operation.
182};
183
184
185class InstructionTable {
186 public:
187 InstructionTable();
188 const InstructionDesc& Get(byte x) const {
189 return instructions_[x];
190 }
191
192 private:
193 InstructionDesc instructions_[256];
194 void Clear();
195 void Init();
196 void CopyTable(ByteMnemonic bm[], InstructionType type);
197 void SetTableRange(InstructionType type, byte start, byte end, bool byte_size,
198 const char* mnem);
199 void AddJumpConditionalShort();
200};
201
202
203InstructionTable::InstructionTable() {
204 Clear();
205 Init();
206}
207
208
209void InstructionTable::Clear() {
210 for (int i = 0; i < 256; i++) {
211 instructions_[i].mnem = "(bad)";
212 instructions_[i].type = NO_INSTR;
213 instructions_[i].op_order_ = UNSET_OP_ORDER;
214 instructions_[i].byte_size_operation = false;
215 }
216}
217
218
219void InstructionTable::Init() {
220 CopyTable(two_operands_instr, TWO_OPERANDS_INSTR);
221 CopyTable(zero_operands_instr, ZERO_OPERANDS_INSTR);
222 CopyTable(call_jump_instr, CALL_JUMP_INSTR);
223 CopyTable(short_immediate_instr, SHORT_IMMEDIATE_INSTR);
224 AddJumpConditionalShort();
225 SetTableRange(PUSHPOP_INSTR, 0x50, 0x57, false, "push");
226 SetTableRange(PUSHPOP_INSTR, 0x58, 0x5F, false, "pop");
227 SetTableRange(MOVE_REG_INSTR, 0xB8, 0xBF, false, "mov");
228}
229
230
231void InstructionTable::CopyTable(ByteMnemonic bm[], InstructionType type) {
232 for (int i = 0; bm[i].b >= 0; i++) {
233 InstructionDesc* id = &instructions_[bm[i].b];
234 id->mnem = bm[i].mnem;
235 OperandType op_order = bm[i].op_order_;
236 id->op_order_ =
237 static_cast<OperandType>(op_order & ~BYTE_SIZE_OPERAND_FLAG);
Steve Blockd0582a62009-12-15 09:54:21 +0000238 ASSERT_EQ(NO_INSTR, id->type); // Information not already entered
Steve Blocka7e24c12009-10-30 11:49:00 +0000239 id->type = type;
240 id->byte_size_operation = ((op_order & BYTE_SIZE_OPERAND_FLAG) != 0);
241 }
242}
243
244
245void InstructionTable::SetTableRange(InstructionType type,
246 byte start,
247 byte end,
248 bool byte_size,
249 const char* mnem) {
250 for (byte b = start; b <= end; b++) {
251 InstructionDesc* id = &instructions_[b];
Steve Blockd0582a62009-12-15 09:54:21 +0000252 ASSERT_EQ(NO_INSTR, id->type); // Information not already entered
Steve Blocka7e24c12009-10-30 11:49:00 +0000253 id->mnem = mnem;
254 id->type = type;
255 id->byte_size_operation = byte_size;
256 }
257}
258
259
260void InstructionTable::AddJumpConditionalShort() {
261 for (byte b = 0x70; b <= 0x7F; b++) {
262 InstructionDesc* id = &instructions_[b];
Steve Blockd0582a62009-12-15 09:54:21 +0000263 ASSERT_EQ(NO_INSTR, id->type); // Information not already entered
Steve Blocka7e24c12009-10-30 11:49:00 +0000264 id->mnem = NULL; // Computed depending on condition code.
265 id->type = JUMP_CONDITIONAL_SHORT_INSTR;
266 }
267}
268
269
270static InstructionTable instruction_table;
271
272static InstructionDesc cmov_instructions[16] = {
273 {"cmovo", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
274 {"cmovno", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
275 {"cmovc", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
276 {"cmovnc", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
277 {"cmovz", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
278 {"cmovnz", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
279 {"cmovna", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
280 {"cmova", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
281 {"cmovs", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
282 {"cmovns", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
283 {"cmovpe", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
284 {"cmovpo", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
285 {"cmovl", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
286 {"cmovge", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
287 {"cmovle", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false},
288 {"cmovg", TWO_OPERANDS_INSTR, REG_OPER_OP_ORDER, false}
289};
290
291//------------------------------------------------------------------------------
292// DisassemblerX64 implementation.
293
294enum UnimplementedOpcodeAction {
295 CONTINUE_ON_UNIMPLEMENTED_OPCODE,
296 ABORT_ON_UNIMPLEMENTED_OPCODE
297};
298
299// A new DisassemblerX64 object is created to disassemble each instruction.
300// The object can only disassemble a single instruction.
301class DisassemblerX64 {
302 public:
303 DisassemblerX64(const NameConverter& converter,
304 UnimplementedOpcodeAction unimplemented_action =
305 ABORT_ON_UNIMPLEMENTED_OPCODE)
306 : converter_(converter),
307 tmp_buffer_pos_(0),
308 abort_on_unimplemented_(
309 unimplemented_action == ABORT_ON_UNIMPLEMENTED_OPCODE),
310 rex_(0),
311 operand_size_(0),
312 group_1_prefix_(0),
313 byte_size_operand_(false) {
314 tmp_buffer_[0] = '\0';
315 }
316
317 virtual ~DisassemblerX64() {
318 }
319
320 // Writes one disassembled instruction into 'buffer' (0-terminated).
321 // Returns the length of the disassembled machine instruction in bytes.
322 int InstructionDecode(v8::internal::Vector<char> buffer, byte* instruction);
323
324 private:
325 enum OperandSize {
326 BYTE_SIZE = 0,
327 WORD_SIZE = 1,
328 DOUBLEWORD_SIZE = 2,
329 QUADWORD_SIZE = 3
330 };
331
332 const NameConverter& converter_;
333 v8::internal::EmbeddedVector<char, 128> tmp_buffer_;
334 unsigned int tmp_buffer_pos_;
335 bool abort_on_unimplemented_;
336 // Prefixes parsed
337 byte rex_;
338 byte operand_size_; // 0x66 or (if no group 3 prefix is present) 0x0.
339 byte group_1_prefix_; // 0xF2, 0xF3, or (if no group 1 prefix is present) 0.
340 // Byte size operand override.
341 bool byte_size_operand_;
342
343 void setRex(byte rex) {
344 ASSERT_EQ(0x40, rex & 0xF0);
345 rex_ = rex;
346 }
347
348 bool rex() { return rex_ != 0; }
349
350 bool rex_b() { return (rex_ & 0x01) != 0; }
351
352 // Actual number of base register given the low bits and the rex.b state.
353 int base_reg(int low_bits) { return low_bits | ((rex_ & 0x01) << 3); }
354
355 bool rex_x() { return (rex_ & 0x02) != 0; }
356
357 bool rex_r() { return (rex_ & 0x04) != 0; }
358
359 bool rex_w() { return (rex_ & 0x08) != 0; }
360
361 OperandSize operand_size() {
362 if (byte_size_operand_) return BYTE_SIZE;
363 if (rex_w()) return QUADWORD_SIZE;
364 if (operand_size_ != 0) return WORD_SIZE;
365 return DOUBLEWORD_SIZE;
366 }
367
368 char operand_size_code() {
369 return "bwlq"[operand_size()];
370 }
371
372 const char* NameOfCPURegister(int reg) const {
373 return converter_.NameOfCPURegister(reg);
374 }
375
376 const char* NameOfByteCPURegister(int reg) const {
377 return converter_.NameOfByteCPURegister(reg);
378 }
379
380 const char* NameOfXMMRegister(int reg) const {
381 return converter_.NameOfXMMRegister(reg);
382 }
383
384 const char* NameOfAddress(byte* addr) const {
385 return converter_.NameOfAddress(addr);
386 }
387
388 // Disassembler helper functions.
389 void get_modrm(byte data,
390 int* mod,
391 int* regop,
392 int* rm) {
393 *mod = (data >> 6) & 3;
394 *regop = ((data & 0x38) >> 3) | (rex_r() ? 8 : 0);
395 *rm = (data & 7) | (rex_b() ? 8 : 0);
396 }
397
398 void get_sib(byte data,
399 int* scale,
400 int* index,
401 int* base) {
402 *scale = (data >> 6) & 3;
403 *index = ((data >> 3) & 7) | (rex_x() ? 8 : 0);
404 *base = (data & 7) | (rex_b() ? 8 : 0);
405 }
406
407 typedef const char* (DisassemblerX64::*RegisterNameMapping)(int reg) const;
408
409 int PrintRightOperandHelper(byte* modrmp,
410 RegisterNameMapping register_name);
411 int PrintRightOperand(byte* modrmp);
412 int PrintRightByteOperand(byte* modrmp);
Steve Blockd0582a62009-12-15 09:54:21 +0000413 int PrintRightXMMOperand(byte* modrmp);
Steve Blocka7e24c12009-10-30 11:49:00 +0000414 int PrintOperands(const char* mnem,
415 OperandType op_order,
416 byte* data);
417 int PrintImmediate(byte* data, OperandSize size);
418 int PrintImmediateOp(byte* data);
419 const char* TwoByteMnemonic(byte opcode);
420 int TwoByteOpcodeInstruction(byte* data);
Steve Blockd0582a62009-12-15 09:54:21 +0000421 int F6F7Instruction(byte* data);
Steve Blocka7e24c12009-10-30 11:49:00 +0000422 int ShiftInstruction(byte* data);
423 int JumpShort(byte* data);
424 int JumpConditional(byte* data);
425 int JumpConditionalShort(byte* data);
426 int SetCC(byte* data);
427 int FPUInstruction(byte* data);
Steve Blockd0582a62009-12-15 09:54:21 +0000428 int MemoryFPUInstruction(int escape_opcode, int regop, byte* modrm_start);
429 int RegisterFPUInstruction(int escape_opcode, byte modrm_byte);
Steve Blocka7e24c12009-10-30 11:49:00 +0000430 void AppendToBuffer(const char* format, ...);
431
432 void UnimplementedInstruction() {
433 if (abort_on_unimplemented_) {
434 CHECK(false);
435 } else {
436 AppendToBuffer("'Unimplemented Instruction'");
437 }
438 }
439};
440
441
442void DisassemblerX64::AppendToBuffer(const char* format, ...) {
443 v8::internal::Vector<char> buf = tmp_buffer_ + tmp_buffer_pos_;
444 va_list args;
445 va_start(args, format);
446 int result = v8::internal::OS::VSNPrintF(buf, format, args);
447 va_end(args);
448 tmp_buffer_pos_ += result;
449}
450
451
452int DisassemblerX64::PrintRightOperandHelper(
453 byte* modrmp,
454 RegisterNameMapping register_name) {
455 int mod, regop, rm;
456 get_modrm(*modrmp, &mod, &regop, &rm);
457 switch (mod) {
458 case 0:
459 if ((rm & 7) == 5) {
460 int32_t disp = *reinterpret_cast<int32_t*>(modrmp + 1);
461 AppendToBuffer("[0x%x]", disp);
462 return 5;
463 } else if ((rm & 7) == 4) {
464 // Codes for SIB byte.
465 byte sib = *(modrmp + 1);
466 int scale, index, base;
467 get_sib(sib, &scale, &index, &base);
468 if (index == 4 && (base & 7) == 4 && scale == 0 /*times_1*/) {
469 // index == rsp means no index. Only use sib byte with no index for
470 // rsp and r12 base.
Steve Block8defd9f2010-07-08 12:39:36 +0100471 AppendToBuffer("[%s]", NameOfCPURegister(base));
Steve Blocka7e24c12009-10-30 11:49:00 +0000472 return 2;
473 } else if (base == 5) {
474 // base == rbp means no base register (when mod == 0).
475 int32_t disp = *reinterpret_cast<int32_t*>(modrmp + 2);
476 AppendToBuffer("[%s*%d+0x%x]",
Steve Block8defd9f2010-07-08 12:39:36 +0100477 NameOfCPURegister(index),
Steve Blocka7e24c12009-10-30 11:49:00 +0000478 1 << scale, disp);
479 return 6;
480 } else if (index != 4 && base != 5) {
481 // [base+index*scale]
482 AppendToBuffer("[%s+%s*%d]",
Steve Block8defd9f2010-07-08 12:39:36 +0100483 NameOfCPURegister(base),
484 NameOfCPURegister(index),
Steve Blocka7e24c12009-10-30 11:49:00 +0000485 1 << scale);
486 return 2;
487 } else {
488 UnimplementedInstruction();
489 return 1;
490 }
491 } else {
Steve Block8defd9f2010-07-08 12:39:36 +0100492 AppendToBuffer("[%s]", NameOfCPURegister(rm));
Steve Blocka7e24c12009-10-30 11:49:00 +0000493 return 1;
494 }
495 break;
496 case 1: // fall through
497 case 2:
498 if ((rm & 7) == 4) {
499 byte sib = *(modrmp + 1);
500 int scale, index, base;
501 get_sib(sib, &scale, &index, &base);
502 int disp = (mod == 2) ? *reinterpret_cast<int32_t*>(modrmp + 2)
503 : *reinterpret_cast<char*>(modrmp + 2);
504 if (index == 4 && (base & 7) == 4 && scale == 0 /*times_1*/) {
505 if (-disp > 0) {
Steve Block8defd9f2010-07-08 12:39:36 +0100506 AppendToBuffer("[%s-0x%x]", NameOfCPURegister(base), -disp);
Steve Blocka7e24c12009-10-30 11:49:00 +0000507 } else {
Steve Block8defd9f2010-07-08 12:39:36 +0100508 AppendToBuffer("[%s+0x%x]", NameOfCPURegister(base), disp);
Steve Blocka7e24c12009-10-30 11:49:00 +0000509 }
510 } else {
511 if (-disp > 0) {
512 AppendToBuffer("[%s+%s*%d-0x%x]",
Steve Block8defd9f2010-07-08 12:39:36 +0100513 NameOfCPURegister(base),
514 NameOfCPURegister(index),
Steve Blocka7e24c12009-10-30 11:49:00 +0000515 1 << scale,
516 -disp);
517 } else {
518 AppendToBuffer("[%s+%s*%d+0x%x]",
Steve Block8defd9f2010-07-08 12:39:36 +0100519 NameOfCPURegister(base),
520 NameOfCPURegister(index),
Steve Blocka7e24c12009-10-30 11:49:00 +0000521 1 << scale,
522 disp);
523 }
524 }
525 return mod == 2 ? 6 : 3;
526 } else {
527 // No sib.
528 int disp = (mod == 2) ? *reinterpret_cast<int32_t*>(modrmp + 1)
529 : *reinterpret_cast<char*>(modrmp + 1);
530 if (-disp > 0) {
Steve Block8defd9f2010-07-08 12:39:36 +0100531 AppendToBuffer("[%s-0x%x]", NameOfCPURegister(rm), -disp);
Steve Blocka7e24c12009-10-30 11:49:00 +0000532 } else {
Steve Block8defd9f2010-07-08 12:39:36 +0100533 AppendToBuffer("[%s+0x%x]", NameOfCPURegister(rm), disp);
Steve Blocka7e24c12009-10-30 11:49:00 +0000534 }
535 return (mod == 2) ? 5 : 2;
536 }
537 break;
538 case 3:
539 AppendToBuffer("%s", (this->*register_name)(rm));
540 return 1;
541 default:
542 UnimplementedInstruction();
543 return 1;
544 }
545 UNREACHABLE();
546}
547
548
549int DisassemblerX64::PrintImmediate(byte* data, OperandSize size) {
550 int64_t value;
551 int count;
552 switch (size) {
553 case BYTE_SIZE:
554 value = *data;
555 count = 1;
556 break;
557 case WORD_SIZE:
558 value = *reinterpret_cast<int16_t*>(data);
559 count = 2;
560 break;
561 case DOUBLEWORD_SIZE:
562 value = *reinterpret_cast<uint32_t*>(data);
563 count = 4;
564 break;
565 case QUADWORD_SIZE:
566 value = *reinterpret_cast<int32_t*>(data);
567 count = 4;
568 break;
569 default:
570 UNREACHABLE();
571 value = 0; // Initialize variables on all paths to satisfy the compiler.
572 count = 0;
573 }
574 AppendToBuffer("%" V8_PTR_PREFIX "x", value);
575 return count;
576}
577
578
579int DisassemblerX64::PrintRightOperand(byte* modrmp) {
580 return PrintRightOperandHelper(modrmp,
581 &DisassemblerX64::NameOfCPURegister);
582}
583
584
585int DisassemblerX64::PrintRightByteOperand(byte* modrmp) {
586 return PrintRightOperandHelper(modrmp,
587 &DisassemblerX64::NameOfByteCPURegister);
588}
589
590
Steve Blockd0582a62009-12-15 09:54:21 +0000591int DisassemblerX64::PrintRightXMMOperand(byte* modrmp) {
592 return PrintRightOperandHelper(modrmp,
593 &DisassemblerX64::NameOfXMMRegister);
594}
595
596
Steve Blocka7e24c12009-10-30 11:49:00 +0000597// Returns number of bytes used including the current *data.
598// Writes instruction's mnemonic, left and right operands to 'tmp_buffer_'.
599int DisassemblerX64::PrintOperands(const char* mnem,
600 OperandType op_order,
601 byte* data) {
602 byte modrm = *data;
603 int mod, regop, rm;
604 get_modrm(modrm, &mod, &regop, &rm);
605 int advance = 0;
606 const char* register_name =
607 byte_size_operand_ ? NameOfByteCPURegister(regop)
608 : NameOfCPURegister(regop);
609 switch (op_order) {
610 case REG_OPER_OP_ORDER: {
611 AppendToBuffer("%s%c %s,",
612 mnem,
613 operand_size_code(),
614 register_name);
615 advance = byte_size_operand_ ? PrintRightByteOperand(data)
616 : PrintRightOperand(data);
617 break;
618 }
619 case OPER_REG_OP_ORDER: {
620 AppendToBuffer("%s%c ", mnem, operand_size_code());
621 advance = byte_size_operand_ ? PrintRightByteOperand(data)
622 : PrintRightOperand(data);
623 AppendToBuffer(",%s", register_name);
624 break;
625 }
626 default:
627 UNREACHABLE();
628 break;
629 }
630 return advance;
631}
632
633
634// Returns number of bytes used by machine instruction, including *data byte.
635// Writes immediate instructions to 'tmp_buffer_'.
636int DisassemblerX64::PrintImmediateOp(byte* data) {
637 bool byte_size_immediate = (*data & 0x02) != 0;
638 byte modrm = *(data + 1);
639 int mod, regop, rm;
640 get_modrm(modrm, &mod, &regop, &rm);
641 const char* mnem = "Imm???";
642 switch (regop) {
643 case 0:
644 mnem = "add";
645 break;
646 case 1:
647 mnem = "or";
648 break;
649 case 2:
650 mnem = "adc";
651 break;
652 case 4:
653 mnem = "and";
654 break;
655 case 5:
656 mnem = "sub";
657 break;
658 case 6:
659 mnem = "xor";
660 break;
661 case 7:
662 mnem = "cmp";
663 break;
664 default:
665 UnimplementedInstruction();
666 }
667 AppendToBuffer("%s%c ", mnem, operand_size_code());
668 int count = PrintRightOperand(data + 1);
669 AppendToBuffer(",0x");
670 OperandSize immediate_size = byte_size_immediate ? BYTE_SIZE : operand_size();
671 count += PrintImmediate(data + 1 + count, immediate_size);
672 return 1 + count;
673}
674
675
676// Returns number of bytes used, including *data.
Steve Blockd0582a62009-12-15 09:54:21 +0000677int DisassemblerX64::F6F7Instruction(byte* data) {
678 ASSERT(*data == 0xF7 || *data == 0xF6);
Steve Blocka7e24c12009-10-30 11:49:00 +0000679 byte modrm = *(data + 1);
680 int mod, regop, rm;
681 get_modrm(modrm, &mod, &regop, &rm);
682 if (mod == 3 && regop != 0) {
683 const char* mnem = NULL;
684 switch (regop) {
685 case 2:
686 mnem = "not";
687 break;
688 case 3:
689 mnem = "neg";
690 break;
691 case 4:
692 mnem = "mul";
693 break;
694 case 7:
695 mnem = "idiv";
696 break;
697 default:
698 UnimplementedInstruction();
699 }
700 AppendToBuffer("%s%c %s",
701 mnem,
702 operand_size_code(),
703 NameOfCPURegister(rm));
704 return 2;
Steve Blocka7e24c12009-10-30 11:49:00 +0000705 } else if (regop == 0) {
706 AppendToBuffer("test%c ", operand_size_code());
Steve Blockd0582a62009-12-15 09:54:21 +0000707 int count = PrintRightOperand(data + 1); // Use name of 64-bit register.
708 AppendToBuffer(",0x");
709 count += PrintImmediate(data + 1 + count, operand_size());
710 return 1 + count;
Steve Blocka7e24c12009-10-30 11:49:00 +0000711 } else {
712 UnimplementedInstruction();
713 return 2;
714 }
715}
716
717
718int DisassemblerX64::ShiftInstruction(byte* data) {
719 byte op = *data & (~1);
720 if (op != 0xD0 && op != 0xD2 && op != 0xC0) {
721 UnimplementedInstruction();
722 return 1;
723 }
724 byte modrm = *(data + 1);
725 int mod, regop, rm;
726 get_modrm(modrm, &mod, &regop, &rm);
727 regop &= 0x7; // The REX.R bit does not affect the operation.
728 int imm8 = -1;
729 int num_bytes = 2;
730 if (mod != 3) {
731 UnimplementedInstruction();
732 return num_bytes;
733 }
734 const char* mnem = NULL;
735 switch (regop) {
736 case 0:
737 mnem = "rol";
738 break;
739 case 1:
740 mnem = "ror";
741 break;
742 case 2:
743 mnem = "rcl";
744 break;
745 case 3:
746 mnem = "rcr";
747 break;
748 case 4:
749 mnem = "shl";
750 break;
751 case 5:
752 mnem = "shr";
753 break;
754 case 7:
755 mnem = "sar";
756 break;
757 default:
758 UnimplementedInstruction();
759 return num_bytes;
760 }
Steve Blockd0582a62009-12-15 09:54:21 +0000761 ASSERT_NE(NULL, mnem);
Steve Blocka7e24c12009-10-30 11:49:00 +0000762 if (op == 0xD0) {
763 imm8 = 1;
764 } else if (op == 0xC0) {
765 imm8 = *(data + 2);
766 num_bytes = 3;
767 }
768 AppendToBuffer("%s%c %s,",
769 mnem,
770 operand_size_code(),
771 byte_size_operand_ ? NameOfByteCPURegister(rm)
772 : NameOfCPURegister(rm));
773 if (op == 0xD2) {
774 AppendToBuffer("cl");
775 } else {
776 AppendToBuffer("%d", imm8);
777 }
778 return num_bytes;
779}
780
781
782// Returns number of bytes used, including *data.
783int DisassemblerX64::JumpShort(byte* data) {
Steve Blockd0582a62009-12-15 09:54:21 +0000784 ASSERT_EQ(0xEB, *data);
Steve Blocka7e24c12009-10-30 11:49:00 +0000785 byte b = *(data + 1);
786 byte* dest = data + static_cast<int8_t>(b) + 2;
787 AppendToBuffer("jmp %s", NameOfAddress(dest));
788 return 2;
789}
790
791
792// Returns number of bytes used, including *data.
793int DisassemblerX64::JumpConditional(byte* data) {
Steve Blockd0582a62009-12-15 09:54:21 +0000794 ASSERT_EQ(0x0F, *data);
Steve Blocka7e24c12009-10-30 11:49:00 +0000795 byte cond = *(data + 1) & 0x0F;
796 byte* dest = data + *reinterpret_cast<int32_t*>(data + 2) + 6;
797 const char* mnem = conditional_code_suffix[cond];
798 AppendToBuffer("j%s %s", mnem, NameOfAddress(dest));
799 return 6; // includes 0x0F
800}
801
802
803// Returns number of bytes used, including *data.
804int DisassemblerX64::JumpConditionalShort(byte* data) {
805 byte cond = *data & 0x0F;
806 byte b = *(data + 1);
807 byte* dest = data + static_cast<int8_t>(b) + 2;
808 const char* mnem = conditional_code_suffix[cond];
809 AppendToBuffer("j%s %s", mnem, NameOfAddress(dest));
810 return 2;
811}
812
813
814// Returns number of bytes used, including *data.
815int DisassemblerX64::SetCC(byte* data) {
Steve Blockd0582a62009-12-15 09:54:21 +0000816 ASSERT_EQ(0x0F, *data);
Steve Blocka7e24c12009-10-30 11:49:00 +0000817 byte cond = *(data + 1) & 0x0F;
818 const char* mnem = conditional_code_suffix[cond];
819 AppendToBuffer("set%s%c ", mnem, operand_size_code());
820 PrintRightByteOperand(data + 2);
821 return 3; // includes 0x0F
822}
823
824
825// Returns number of bytes used, including *data.
826int DisassemblerX64::FPUInstruction(byte* data) {
Steve Blockd0582a62009-12-15 09:54:21 +0000827 byte escape_opcode = *data;
828 ASSERT_EQ(0xD8, escape_opcode & 0xF8);
829 byte modrm_byte = *(data+1);
830
831 if (modrm_byte >= 0xC0) {
832 return RegisterFPUInstruction(escape_opcode, modrm_byte);
833 } else {
834 return MemoryFPUInstruction(escape_opcode, modrm_byte, data+1);
Steve Blocka7e24c12009-10-30 11:49:00 +0000835 }
Steve Blockd0582a62009-12-15 09:54:21 +0000836}
837
838int DisassemblerX64::MemoryFPUInstruction(int escape_opcode,
839 int modrm_byte,
840 byte* modrm_start) {
841 const char* mnem = "?";
842 int regop = (modrm_byte >> 3) & 0x7; // reg/op field of modrm byte.
843 switch (escape_opcode) {
844 case 0xD9: switch (regop) {
845 case 0: mnem = "fld_s"; break;
846 case 3: mnem = "fstp_s"; break;
847 case 7: mnem = "fstcw"; break;
848 default: UnimplementedInstruction();
849 }
850 break;
851
852 case 0xDB: switch (regop) {
853 case 0: mnem = "fild_s"; break;
854 case 1: mnem = "fisttp_s"; break;
855 case 2: mnem = "fist_s"; break;
856 case 3: mnem = "fistp_s"; break;
857 default: UnimplementedInstruction();
858 }
859 break;
860
861 case 0xDD: switch (regop) {
862 case 0: mnem = "fld_d"; break;
863 case 3: mnem = "fstp_d"; break;
864 default: UnimplementedInstruction();
865 }
866 break;
867
868 case 0xDF: switch (regop) {
869 case 5: mnem = "fild_d"; break;
870 case 7: mnem = "fistp_d"; break;
871 default: UnimplementedInstruction();
872 }
873 break;
874
875 default: UnimplementedInstruction();
876 }
877 AppendToBuffer("%s ", mnem);
878 int count = PrintRightOperand(modrm_start);
879 return count + 1;
880}
881
882int DisassemblerX64::RegisterFPUInstruction(int escape_opcode,
883 byte modrm_byte) {
884 bool has_register = false; // Is the FPU register encoded in modrm_byte?
885 const char* mnem = "?";
886
887 switch (escape_opcode) {
888 case 0xD8:
889 UnimplementedInstruction();
890 break;
891
892 case 0xD9:
893 switch (modrm_byte & 0xF8) {
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100894 case 0xC0:
895 mnem = "fld";
896 has_register = true;
897 break;
Steve Blockd0582a62009-12-15 09:54:21 +0000898 case 0xC8:
899 mnem = "fxch";
900 has_register = true;
901 break;
902 default:
903 switch (modrm_byte) {
904 case 0xE0: mnem = "fchs"; break;
905 case 0xE1: mnem = "fabs"; break;
906 case 0xE4: mnem = "ftst"; break;
907 case 0xE8: mnem = "fld1"; break;
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100908 case 0xEB: mnem = "fldpi"; break;
Ben Murdochb0fe1622011-05-05 13:52:32 +0100909 case 0xED: mnem = "fldln2"; break;
Steve Blockd0582a62009-12-15 09:54:21 +0000910 case 0xEE: mnem = "fldz"; break;
Ben Murdochb0fe1622011-05-05 13:52:32 +0100911 case 0xF1: mnem = "fyl2x"; break;
Steve Blockd0582a62009-12-15 09:54:21 +0000912 case 0xF5: mnem = "fprem1"; break;
913 case 0xF7: mnem = "fincstp"; break;
914 case 0xF8: mnem = "fprem"; break;
915 case 0xFE: mnem = "fsin"; break;
916 case 0xFF: mnem = "fcos"; break;
917 default: UnimplementedInstruction();
918 }
919 }
920 break;
921
922 case 0xDA:
923 if (modrm_byte == 0xE9) {
924 mnem = "fucompp";
925 } else {
926 UnimplementedInstruction();
927 }
928 break;
929
930 case 0xDB:
931 if ((modrm_byte & 0xF8) == 0xE8) {
932 mnem = "fucomi";
933 has_register = true;
934 } else if (modrm_byte == 0xE2) {
935 mnem = "fclex";
936 } else {
937 UnimplementedInstruction();
938 }
939 break;
940
941 case 0xDC:
942 has_register = true;
943 switch (modrm_byte & 0xF8) {
944 case 0xC0: mnem = "fadd"; break;
945 case 0xE8: mnem = "fsub"; break;
946 case 0xC8: mnem = "fmul"; break;
947 case 0xF8: mnem = "fdiv"; break;
948 default: UnimplementedInstruction();
949 }
950 break;
951
952 case 0xDD:
953 has_register = true;
954 switch (modrm_byte & 0xF8) {
955 case 0xC0: mnem = "ffree"; break;
956 case 0xD8: mnem = "fstp"; break;
957 default: UnimplementedInstruction();
958 }
959 break;
960
961 case 0xDE:
962 if (modrm_byte == 0xD9) {
963 mnem = "fcompp";
964 } else {
965 has_register = true;
966 switch (modrm_byte & 0xF8) {
967 case 0xC0: mnem = "faddp"; break;
968 case 0xE8: mnem = "fsubp"; break;
969 case 0xC8: mnem = "fmulp"; break;
970 case 0xF8: mnem = "fdivp"; break;
971 default: UnimplementedInstruction();
972 }
973 }
974 break;
975
976 case 0xDF:
977 if (modrm_byte == 0xE0) {
978 mnem = "fnstsw_ax";
979 } else if ((modrm_byte & 0xF8) == 0xE8) {
980 mnem = "fucomip";
981 has_register = true;
982 }
983 break;
984
985 default: UnimplementedInstruction();
986 }
987
988 if (has_register) {
989 AppendToBuffer("%s st%d", mnem, modrm_byte & 0x7);
990 } else {
991 AppendToBuffer("%s", mnem);
992 }
Steve Blocka7e24c12009-10-30 11:49:00 +0000993 return 2;
994}
995
996
Steve Blockd0582a62009-12-15 09:54:21 +0000997
Steve Blocka7e24c12009-10-30 11:49:00 +0000998// Handle all two-byte opcodes, which start with 0x0F.
999// These instructions may be affected by an 0x66, 0xF2, or 0xF3 prefix.
1000// We do not use any three-byte opcodes, which start with 0x0F38 or 0x0F3A.
1001int DisassemblerX64::TwoByteOpcodeInstruction(byte* data) {
1002 byte opcode = *(data + 1);
1003 byte* current = data + 2;
1004 // At return, "current" points to the start of the next instruction.
1005 const char* mnemonic = TwoByteMnemonic(opcode);
Andrei Popescu402d9372010-02-26 13:31:12 +00001006 if (operand_size_ == 0x66) {
1007 // 0x66 0x0F prefix.
Steve Blocka7e24c12009-10-30 11:49:00 +00001008 int mod, regop, rm;
Steve Block6ded16b2010-05-10 14:33:55 +01001009 if (opcode == 0x3A) {
1010 byte third_byte = *current;
1011 current = data + 3;
1012 if (third_byte == 0x17) {
1013 get_modrm(*current, &mod, &regop, &rm);
1014 AppendToBuffer("extractps "); // reg/m32, xmm, imm8
1015 current += PrintRightOperand(current);
1016 AppendToBuffer(", %s, %d", NameOfCPURegister(regop), (*current) & 3);
1017 current += 1;
1018 } else {
1019 UnimplementedInstruction();
1020 }
Steve Blocka7e24c12009-10-30 11:49:00 +00001021 } else {
Steve Block6ded16b2010-05-10 14:33:55 +01001022 get_modrm(*current, &mod, &regop, &rm);
1023 if (opcode == 0x6E) {
1024 AppendToBuffer("mov%c %s,",
1025 rex_w() ? 'q' : 'd',
1026 NameOfXMMRegister(regop));
1027 current += PrintRightOperand(current);
Steve Block1e0659c2011-05-24 12:43:12 +01001028 } else if (opcode == 0x6F) {
1029 AppendToBuffer("movdqa %s,",
1030 NameOfXMMRegister(regop));
1031 current += PrintRightOperand(current);
Steve Block6ded16b2010-05-10 14:33:55 +01001032 } else if (opcode == 0x7E) {
Ben Murdochbb769b22010-08-11 14:56:33 +01001033 AppendToBuffer("mov%c ",
1034 rex_w() ? 'q' : 'd');
1035 current += PrintRightOperand(current);
1036 AppendToBuffer(", %s", NameOfXMMRegister(regop));
Steve Block1e0659c2011-05-24 12:43:12 +01001037 } else if (opcode == 0x7F) {
1038 AppendToBuffer("movdqa ");
1039 current += PrintRightOperand(current);
1040 AppendToBuffer(", %s", NameOfXMMRegister(regop));
Steve Block6ded16b2010-05-10 14:33:55 +01001041 } else {
1042 const char* mnemonic = "?";
Ben Murdoche0cee9b2011-05-25 10:26:03 +01001043 if (opcode == 0x50) {
1044 mnemonic = "movmskpd";
1045 } else if (opcode == 0x54) {
1046 mnemonic = "andpd";
1047 } else if (opcode == 0x56) {
1048 mnemonic = "orpd";
1049 } else if (opcode == 0x57) {
Steve Block6ded16b2010-05-10 14:33:55 +01001050 mnemonic = "xorpd";
1051 } else if (opcode == 0x2E) {
Steve Block6ded16b2010-05-10 14:33:55 +01001052 mnemonic = "ucomisd";
Steve Block8defd9f2010-07-08 12:39:36 +01001053 } else if (opcode == 0x2F) {
1054 mnemonic = "comisd";
Steve Block6ded16b2010-05-10 14:33:55 +01001055 } else {
1056 UnimplementedInstruction();
1057 }
1058 AppendToBuffer("%s %s,", mnemonic, NameOfXMMRegister(regop));
1059 current += PrintRightXMMOperand(current);
1060 }
Steve Blocka7e24c12009-10-30 11:49:00 +00001061 }
1062 } else if (group_1_prefix_ == 0xF2) {
1063 // Beginning of instructions with prefix 0xF2.
1064
1065 if (opcode == 0x11 || opcode == 0x10) {
1066 // MOVSD: Move scalar double-precision fp to/from/between XMM registers.
1067 AppendToBuffer("movsd ");
1068 int mod, regop, rm;
1069 get_modrm(*current, &mod, &regop, &rm);
1070 if (opcode == 0x11) {
1071 current += PrintRightOperand(current);
1072 AppendToBuffer(",%s", NameOfXMMRegister(regop));
1073 } else {
1074 AppendToBuffer("%s,", NameOfXMMRegister(regop));
1075 current += PrintRightOperand(current);
1076 }
1077 } else if (opcode == 0x2A) {
1078 // CVTSI2SD: integer to XMM double conversion.
1079 int mod, regop, rm;
1080 get_modrm(*current, &mod, &regop, &rm);
Steve Block8defd9f2010-07-08 12:39:36 +01001081 AppendToBuffer("%sd %s,", mnemonic, NameOfXMMRegister(regop));
Steve Blockd0582a62009-12-15 09:54:21 +00001082 current += PrintRightOperand(current);
Kristian Monsen0d5e1162010-09-30 15:31:59 +01001083 } else if (opcode == 0x2C) {
1084 // CVTTSD2SI:
1085 // Convert with truncation scalar double-precision FP to integer.
1086 int mod, regop, rm;
1087 get_modrm(*current, &mod, &regop, &rm);
1088 AppendToBuffer("cvttsd2si%c %s,",
1089 operand_size_code(), NameOfCPURegister(regop));
1090 current += PrintRightXMMOperand(current);
1091 } else if (opcode == 0x2D) {
1092 // CVTSD2SI: Convert scalar double-precision FP to integer.
1093 int mod, regop, rm;
1094 get_modrm(*current, &mod, &regop, &rm);
1095 AppendToBuffer("cvtsd2si%c %s,",
1096 operand_size_code(), NameOfCPURegister(regop));
1097 current += PrintRightXMMOperand(current);
Steve Block6ded16b2010-05-10 14:33:55 +01001098 } else if ((opcode & 0xF8) == 0x58 || opcode == 0x51) {
Steve Blocka7e24c12009-10-30 11:49:00 +00001099 // XMM arithmetic. Mnemonic was retrieved at the start of this function.
1100 int mod, regop, rm;
1101 get_modrm(*current, &mod, &regop, &rm);
Steve Blockd0582a62009-12-15 09:54:21 +00001102 AppendToBuffer("%s %s,", mnemonic, NameOfXMMRegister(regop));
1103 current += PrintRightXMMOperand(current);
Steve Blocka7e24c12009-10-30 11:49:00 +00001104 } else {
1105 UnimplementedInstruction();
1106 }
Steve Block6ded16b2010-05-10 14:33:55 +01001107 } else if (group_1_prefix_ == 0xF3) {
1108 // Instructions with prefix 0xF3.
Steve Block8defd9f2010-07-08 12:39:36 +01001109 if (opcode == 0x11 || opcode == 0x10) {
1110 // MOVSS: Move scalar double-precision fp to/from/between XMM registers.
1111 AppendToBuffer("movss ");
1112 int mod, regop, rm;
1113 get_modrm(*current, &mod, &regop, &rm);
1114 if (opcode == 0x11) {
1115 current += PrintRightOperand(current);
1116 AppendToBuffer(",%s", NameOfXMMRegister(regop));
1117 } else {
1118 AppendToBuffer("%s,", NameOfXMMRegister(regop));
1119 current += PrintRightOperand(current);
1120 }
1121 } else if (opcode == 0x2A) {
1122 // CVTSI2SS: integer to XMM single conversion.
1123 int mod, regop, rm;
1124 get_modrm(*current, &mod, &regop, &rm);
1125 AppendToBuffer("%ss %s,", mnemonic, NameOfXMMRegister(regop));
1126 current += PrintRightOperand(current);
1127 } else if (opcode == 0x2C) {
Kristian Monsen0d5e1162010-09-30 15:31:59 +01001128 // CVTTSS2SI:
1129 // Convert with truncation scalar single-precision FP to dword integer.
Steve Block1e0659c2011-05-24 12:43:12 +01001130 int mod, regop, rm;
1131 get_modrm(*current, &mod, &regop, &rm);
1132 AppendToBuffer("cvttss2si%c %s,",
1133 operand_size_code(), NameOfCPURegister(regop));
1134 current += PrintRightXMMOperand(current);
Steve Block6ded16b2010-05-10 14:33:55 +01001135 } else if (opcode == 0x5A) {
Kristian Monsen0d5e1162010-09-30 15:31:59 +01001136 // CVTSS2SD:
1137 // Convert scalar single-precision FP to scalar double-precision FP.
Steve Block6ded16b2010-05-10 14:33:55 +01001138 int mod, regop, rm;
1139 get_modrm(*current, &mod, &regop, &rm);
1140 AppendToBuffer("cvtss2sd %s,", NameOfXMMRegister(regop));
1141 current += PrintRightXMMOperand(current);
1142 } else {
1143 UnimplementedInstruction();
1144 }
Andrei Popescu402d9372010-02-26 13:31:12 +00001145 } else if (opcode == 0x1F) {
1146 // NOP
1147 int mod, regop, rm;
1148 get_modrm(*current, &mod, &regop, &rm);
1149 current++;
1150 if (regop == 4) { // SIB byte present.
1151 current++;
1152 }
1153 if (mod == 1) { // Byte displacement.
1154 current += 1;
1155 } else if (mod == 2) { // 32-bit displacement.
1156 current += 4;
1157 } // else no immediate displacement.
1158 AppendToBuffer("nop");
1159 } else if (opcode == 0xA2 || opcode == 0x31) {
1160 // RDTSC or CPUID
1161 AppendToBuffer("%s", mnemonic);
1162
1163 } else if ((opcode & 0xF0) == 0x40) {
1164 // CMOVcc: conditional move.
1165 int condition = opcode & 0x0F;
1166 const InstructionDesc& idesc = cmov_instructions[condition];
1167 byte_size_operand_ = idesc.byte_size_operation;
1168 current += PrintOperands(idesc.mnem, idesc.op_order_, current);
1169
1170 } else if ((opcode & 0xF0) == 0x80) {
1171 // Jcc: Conditional jump (branch).
1172 current = data + JumpConditional(data);
1173
1174 } else if (opcode == 0xBE || opcode == 0xBF || opcode == 0xB6 ||
1175 opcode == 0xB7 || opcode == 0xAF) {
1176 // Size-extending moves, IMUL.
1177 current += PrintOperands(mnemonic, REG_OPER_OP_ORDER, current);
1178
1179 } else if ((opcode & 0xF0) == 0x90) {
1180 // SETcc: Set byte on condition. Needs pointer to beginning of instruction.
1181 current = data + SetCC(data);
1182
1183 } else if (opcode == 0xAB || opcode == 0xA5 || opcode == 0xAD) {
1184 // SHLD, SHRD (double-precision shift), BTS (bit set).
1185 AppendToBuffer("%s ", mnemonic);
1186 int mod, regop, rm;
1187 get_modrm(*current, &mod, &regop, &rm);
1188 current += PrintRightOperand(current);
1189 if (opcode == 0xAB) {
1190 AppendToBuffer(",%s", NameOfCPURegister(regop));
1191 } else {
1192 AppendToBuffer(",%s,cl", NameOfCPURegister(regop));
1193 }
Steve Blocka7e24c12009-10-30 11:49:00 +00001194 } else {
1195 UnimplementedInstruction();
1196 }
Steve Blockd0582a62009-12-15 09:54:21 +00001197 return static_cast<int>(current - data);
Steve Blocka7e24c12009-10-30 11:49:00 +00001198}
1199
1200
1201// Mnemonics for two-byte opcode instructions starting with 0x0F.
1202// The argument is the second byte of the two-byte opcode.
1203// Returns NULL if the instruction is not handled here.
1204const char* DisassemblerX64::TwoByteMnemonic(byte opcode) {
1205 switch (opcode) {
1206 case 0x1F:
1207 return "nop";
Steve Block8defd9f2010-07-08 12:39:36 +01001208 case 0x2A: // F2/F3 prefix.
1209 return "cvtsi2s";
Steve Blocka7e24c12009-10-30 11:49:00 +00001210 case 0x31:
1211 return "rdtsc";
Steve Block6ded16b2010-05-10 14:33:55 +01001212 case 0x51: // F2 prefix.
1213 return "sqrtsd";
Steve Blocka7e24c12009-10-30 11:49:00 +00001214 case 0x58: // F2 prefix.
1215 return "addsd";
1216 case 0x59: // F2 prefix.
1217 return "mulsd";
1218 case 0x5C: // F2 prefix.
1219 return "subsd";
1220 case 0x5E: // F2 prefix.
1221 return "divsd";
1222 case 0xA2:
1223 return "cpuid";
1224 case 0xA5:
1225 return "shld";
1226 case 0xAB:
1227 return "bts";
1228 case 0xAD:
1229 return "shrd";
1230 case 0xAF:
1231 return "imul";
1232 case 0xB6:
1233 return "movzxb";
1234 case 0xB7:
1235 return "movzxw";
1236 case 0xBE:
1237 return "movsxb";
1238 case 0xBF:
1239 return "movsxw";
1240 default:
1241 return NULL;
1242 }
1243}
1244
1245
1246// Disassembles the instruction at instr, and writes it into out_buffer.
1247int DisassemblerX64::InstructionDecode(v8::internal::Vector<char> out_buffer,
1248 byte* instr) {
1249 tmp_buffer_pos_ = 0; // starting to write as position 0
1250 byte* data = instr;
1251 bool processed = true; // Will be set to false if the current instruction
1252 // is not in 'instructions' table.
1253 byte current;
1254
1255 // Scan for prefixes.
1256 while (true) {
1257 current = *data;
Leon Clarked91b9f72010-01-27 17:25:45 +00001258 if (current == OPERAND_SIZE_OVERRIDE_PREFIX) { // Group 3 prefix.
Steve Blocka7e24c12009-10-30 11:49:00 +00001259 operand_size_ = current;
1260 } else if ((current & 0xF0) == 0x40) { // REX prefix.
1261 setRex(current);
1262 if (rex_w()) AppendToBuffer("REX.W ");
Leon Clarked91b9f72010-01-27 17:25:45 +00001263 } else if ((current & 0xFE) == 0xF2) { // Group 1 prefix (0xF2 or 0xF3).
Steve Blocka7e24c12009-10-30 11:49:00 +00001264 group_1_prefix_ = current;
1265 } else { // Not a prefix - an opcode.
1266 break;
1267 }
1268 data++;
1269 }
1270
1271 const InstructionDesc& idesc = instruction_table.Get(current);
1272 byte_size_operand_ = idesc.byte_size_operation;
1273 switch (idesc.type) {
1274 case ZERO_OPERANDS_INSTR:
Leon Clarked91b9f72010-01-27 17:25:45 +00001275 if (current >= 0xA4 && current <= 0xA7) {
1276 // String move or compare operations.
1277 if (group_1_prefix_ == REP_PREFIX) {
1278 // REP.
1279 AppendToBuffer("rep ");
1280 }
1281 if (rex_w()) AppendToBuffer("REX.W ");
1282 AppendToBuffer("%s%c", idesc.mnem, operand_size_code());
1283 } else {
1284 AppendToBuffer("%s", idesc.mnem, operand_size_code());
1285 }
Steve Blocka7e24c12009-10-30 11:49:00 +00001286 data++;
1287 break;
1288
1289 case TWO_OPERANDS_INSTR:
1290 data++;
1291 data += PrintOperands(idesc.mnem, idesc.op_order_, data);
1292 break;
1293
1294 case JUMP_CONDITIONAL_SHORT_INSTR:
1295 data += JumpConditionalShort(data);
1296 break;
1297
1298 case REGISTER_INSTR:
1299 AppendToBuffer("%s%c %s",
1300 idesc.mnem,
1301 operand_size_code(),
1302 NameOfCPURegister(base_reg(current & 0x07)));
1303 data++;
1304 break;
1305 case PUSHPOP_INSTR:
1306 AppendToBuffer("%s %s",
1307 idesc.mnem,
1308 NameOfCPURegister(base_reg(current & 0x07)));
1309 data++;
1310 break;
1311 case MOVE_REG_INSTR: {
1312 byte* addr = NULL;
1313 switch (operand_size()) {
1314 case WORD_SIZE:
1315 addr = reinterpret_cast<byte*>(*reinterpret_cast<int16_t*>(data + 1));
1316 data += 3;
1317 break;
1318 case DOUBLEWORD_SIZE:
1319 addr = reinterpret_cast<byte*>(*reinterpret_cast<int32_t*>(data + 1));
1320 data += 5;
1321 break;
1322 case QUADWORD_SIZE:
1323 addr = reinterpret_cast<byte*>(*reinterpret_cast<int64_t*>(data + 1));
1324 data += 9;
1325 break;
1326 default:
1327 UNREACHABLE();
1328 }
1329 AppendToBuffer("mov%c %s,%s",
1330 operand_size_code(),
1331 NameOfCPURegister(base_reg(current & 0x07)),
1332 NameOfAddress(addr));
1333 break;
1334 }
1335
1336 case CALL_JUMP_INSTR: {
1337 byte* addr = data + *reinterpret_cast<int32_t*>(data + 1) + 5;
1338 AppendToBuffer("%s %s", idesc.mnem, NameOfAddress(addr));
1339 data += 5;
1340 break;
1341 }
1342
1343 case SHORT_IMMEDIATE_INSTR: {
1344 byte* addr =
1345 reinterpret_cast<byte*>(*reinterpret_cast<int32_t*>(data + 1));
1346 AppendToBuffer("%s rax, %s", idesc.mnem, NameOfAddress(addr));
1347 data += 5;
1348 break;
1349 }
1350
1351 case NO_INSTR:
1352 processed = false;
1353 break;
1354
1355 default:
1356 UNIMPLEMENTED(); // This type is not implemented.
1357 }
1358
1359 // The first byte didn't match any of the simple opcodes, so we
1360 // need to do special processing on it.
1361 if (!processed) {
1362 switch (*data) {
1363 case 0xC2:
1364 AppendToBuffer("ret 0x%x", *reinterpret_cast<uint16_t*>(data + 1));
1365 data += 3;
1366 break;
1367
1368 case 0x69: // fall through
1369 case 0x6B: {
1370 int mod, regop, rm;
1371 get_modrm(*(data + 1), &mod, &regop, &rm);
1372 int32_t imm = *data == 0x6B ? *(data + 2)
1373 : *reinterpret_cast<int32_t*>(data + 2);
Steve Block6ded16b2010-05-10 14:33:55 +01001374 AppendToBuffer("imul%c %s,%s,0x%x",
1375 operand_size_code(),
1376 NameOfCPURegister(regop),
Steve Blocka7e24c12009-10-30 11:49:00 +00001377 NameOfCPURegister(rm), imm);
1378 data += 2 + (*data == 0x6B ? 1 : 4);
1379 break;
1380 }
1381
Steve Blocka7e24c12009-10-30 11:49:00 +00001382 case 0x81: // fall through
1383 case 0x83: // 0x81 with sign extension bit set
1384 data += PrintImmediateOp(data);
1385 break;
1386
1387 case 0x0F:
1388 data += TwoByteOpcodeInstruction(data);
1389 break;
1390
1391 case 0x8F: {
1392 data++;
1393 int mod, regop, rm;
1394 get_modrm(*data, &mod, &regop, &rm);
1395 if (regop == 0) {
1396 AppendToBuffer("pop ");
1397 data += PrintRightOperand(data);
1398 }
1399 }
1400 break;
1401
1402 case 0xFF: {
1403 data++;
1404 int mod, regop, rm;
1405 get_modrm(*data, &mod, &regop, &rm);
1406 const char* mnem = NULL;
1407 switch (regop) {
1408 case 0:
1409 mnem = "inc";
1410 break;
1411 case 1:
1412 mnem = "dec";
1413 break;
1414 case 2:
1415 mnem = "call";
1416 break;
1417 case 4:
1418 mnem = "jmp";
1419 break;
1420 case 6:
1421 mnem = "push";
1422 break;
1423 default:
1424 mnem = "???";
1425 }
1426 AppendToBuffer(((regop <= 1) ? "%s%c " : "%s "),
1427 mnem,
1428 operand_size_code());
1429 data += PrintRightOperand(data);
1430 }
1431 break;
1432
1433 case 0xC7: // imm32, fall through
1434 case 0xC6: // imm8
1435 {
1436 bool is_byte = *data == 0xC6;
1437 data++;
1438
1439 AppendToBuffer("mov%c ", is_byte ? 'b' : operand_size_code());
1440 data += PrintRightOperand(data);
1441 int32_t imm = is_byte ? *data : *reinterpret_cast<int32_t*>(data);
1442 AppendToBuffer(",0x%x", imm);
1443 data += is_byte ? 1 : 4;
1444 }
1445 break;
1446
1447 case 0x80: {
1448 data++;
1449 AppendToBuffer("cmpb ");
1450 data += PrintRightOperand(data);
1451 int32_t imm = *data;
1452 AppendToBuffer(",0x%x", imm);
1453 data++;
1454 }
1455 break;
1456
1457 case 0x88: // 8bit, fall through
1458 case 0x89: // 32bit
1459 {
1460 bool is_byte = *data == 0x88;
1461 int mod, regop, rm;
1462 data++;
1463 get_modrm(*data, &mod, &regop, &rm);
1464 AppendToBuffer("mov%c ", is_byte ? 'b' : operand_size_code());
1465 data += PrintRightOperand(data);
1466 AppendToBuffer(",%s", NameOfCPURegister(regop));
1467 }
1468 break;
1469
1470 case 0x90:
1471 case 0x91:
1472 case 0x92:
1473 case 0x93:
1474 case 0x94:
1475 case 0x95:
1476 case 0x96:
1477 case 0x97: {
Steve Blockd0582a62009-12-15 09:54:21 +00001478 int reg = (*data & 0x7) | (rex_b() ? 8 : 0);
Steve Blocka7e24c12009-10-30 11:49:00 +00001479 if (reg == 0) {
1480 AppendToBuffer("nop"); // Common name for xchg rax,rax.
1481 } else {
1482 AppendToBuffer("xchg%c rax, %s",
1483 operand_size_code(),
1484 NameOfCPURegister(reg));
1485 }
Steve Blockd0582a62009-12-15 09:54:21 +00001486 data++;
Steve Blocka7e24c12009-10-30 11:49:00 +00001487 }
Steve Blockd0582a62009-12-15 09:54:21 +00001488 break;
Steve Blocka7e24c12009-10-30 11:49:00 +00001489
1490 case 0xFE: {
1491 data++;
1492 int mod, regop, rm;
1493 get_modrm(*data, &mod, &regop, &rm);
Kristian Monsen0d5e1162010-09-30 15:31:59 +01001494 if (regop == 1) {
1495 AppendToBuffer("decb ");
1496 data += PrintRightOperand(data);
Steve Blocka7e24c12009-10-30 11:49:00 +00001497 } else {
1498 UnimplementedInstruction();
1499 }
Steve Blocka7e24c12009-10-30 11:49:00 +00001500 }
1501 break;
1502
1503 case 0x68:
1504 AppendToBuffer("push 0x%x", *reinterpret_cast<int32_t*>(data + 1));
1505 data += 5;
1506 break;
1507
1508 case 0x6A:
1509 AppendToBuffer("push 0x%x", *reinterpret_cast<int8_t*>(data + 1));
1510 data += 2;
1511 break;
1512
1513 case 0xA1: // Fall through.
1514 case 0xA3:
1515 switch (operand_size()) {
1516 case DOUBLEWORD_SIZE: {
1517 const char* memory_location = NameOfAddress(
1518 reinterpret_cast<byte*>(
1519 *reinterpret_cast<int32_t*>(data + 1)));
1520 if (*data == 0xA1) { // Opcode 0xA1
1521 AppendToBuffer("movzxlq rax,(%s)", memory_location);
1522 } else { // Opcode 0xA3
1523 AppendToBuffer("movzxlq (%s),rax", memory_location);
1524 }
1525 data += 5;
1526 break;
1527 }
1528 case QUADWORD_SIZE: {
1529 // New x64 instruction mov rax,(imm_64).
1530 const char* memory_location = NameOfAddress(
1531 *reinterpret_cast<byte**>(data + 1));
1532 if (*data == 0xA1) { // Opcode 0xA1
1533 AppendToBuffer("movq rax,(%s)", memory_location);
1534 } else { // Opcode 0xA3
1535 AppendToBuffer("movq (%s),rax", memory_location);
1536 }
1537 data += 9;
1538 break;
1539 }
1540 default:
1541 UnimplementedInstruction();
1542 data += 2;
1543 }
1544 break;
1545
1546 case 0xA8:
1547 AppendToBuffer("test al,0x%x", *reinterpret_cast<uint8_t*>(data + 1));
1548 data += 2;
1549 break;
1550
1551 case 0xA9: {
1552 int64_t value = 0;
1553 switch (operand_size()) {
1554 case WORD_SIZE:
1555 value = *reinterpret_cast<uint16_t*>(data + 1);
1556 data += 3;
1557 break;
1558 case DOUBLEWORD_SIZE:
1559 value = *reinterpret_cast<uint32_t*>(data + 1);
1560 data += 5;
1561 break;
1562 case QUADWORD_SIZE:
1563 value = *reinterpret_cast<int32_t*>(data + 1);
1564 data += 5;
1565 break;
1566 default:
1567 UNREACHABLE();
1568 }
1569 AppendToBuffer("test%c rax,0x%"V8_PTR_PREFIX"x",
1570 operand_size_code(),
1571 value);
1572 break;
1573 }
1574 case 0xD1: // fall through
1575 case 0xD3: // fall through
1576 case 0xC1:
1577 data += ShiftInstruction(data);
1578 break;
1579 case 0xD0: // fall through
1580 case 0xD2: // fall through
1581 case 0xC0:
1582 byte_size_operand_ = true;
1583 data += ShiftInstruction(data);
1584 break;
1585
1586 case 0xD9: // fall through
1587 case 0xDA: // fall through
1588 case 0xDB: // fall through
1589 case 0xDC: // fall through
1590 case 0xDD: // fall through
1591 case 0xDE: // fall through
1592 case 0xDF:
1593 data += FPUInstruction(data);
1594 break;
1595
1596 case 0xEB:
1597 data += JumpShort(data);
1598 break;
1599
Steve Blockd0582a62009-12-15 09:54:21 +00001600 case 0xF6:
1601 byte_size_operand_ = true; // fall through
Steve Blocka7e24c12009-10-30 11:49:00 +00001602 case 0xF7:
Steve Blockd0582a62009-12-15 09:54:21 +00001603 data += F6F7Instruction(data);
Steve Blocka7e24c12009-10-30 11:49:00 +00001604 break;
1605
1606 default:
1607 UnimplementedInstruction();
1608 data += 1;
1609 }
1610 } // !processed
1611
1612 if (tmp_buffer_pos_ < sizeof tmp_buffer_) {
1613 tmp_buffer_[tmp_buffer_pos_] = '\0';
1614 }
1615
Steve Blockd0582a62009-12-15 09:54:21 +00001616 int instr_len = static_cast<int>(data - instr);
Steve Blocka7e24c12009-10-30 11:49:00 +00001617 ASSERT(instr_len > 0); // Ensure progress.
1618
1619 int outp = 0;
1620 // Instruction bytes.
1621 for (byte* bp = instr; bp < data; bp++) {
1622 outp += v8::internal::OS::SNPrintF(out_buffer + outp, "%02x", *bp);
1623 }
1624 for (int i = 6 - instr_len; i >= 0; i--) {
1625 outp += v8::internal::OS::SNPrintF(out_buffer + outp, " ");
1626 }
1627
1628 outp += v8::internal::OS::SNPrintF(out_buffer + outp, " %s",
1629 tmp_buffer_.start());
1630 return instr_len;
1631}
1632
1633//------------------------------------------------------------------------------
1634
1635
1636static const char* cpu_regs[16] = {
1637 "rax", "rcx", "rdx", "rbx", "rsp", "rbp", "rsi", "rdi",
1638 "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15"
1639};
1640
1641
1642static const char* byte_cpu_regs[16] = {
1643 "al", "cl", "dl", "bl", "spl", "bpl", "sil", "dil",
1644 "r8l", "r9l", "r10l", "r11l", "r12l", "r13l", "r14l", "r15l"
1645};
1646
1647
1648static const char* xmm_regs[16] = {
1649 "xmm0", "xmm1", "xmm2", "xmm3", "xmm4", "xmm5", "xmm6", "xmm7",
1650 "xmm8", "xmm9", "xmm10", "xmm11", "xmm12", "xmm13", "xmm14", "xmm15"
1651};
1652
1653
1654const char* NameConverter::NameOfAddress(byte* addr) const {
1655 static v8::internal::EmbeddedVector<char, 32> tmp_buffer;
1656 v8::internal::OS::SNPrintF(tmp_buffer, "%p", addr);
1657 return tmp_buffer.start();
1658}
1659
1660
1661const char* NameConverter::NameOfConstant(byte* addr) const {
1662 return NameOfAddress(addr);
1663}
1664
1665
1666const char* NameConverter::NameOfCPURegister(int reg) const {
1667 if (0 <= reg && reg < 16)
1668 return cpu_regs[reg];
1669 return "noreg";
1670}
1671
1672
1673const char* NameConverter::NameOfByteCPURegister(int reg) const {
1674 if (0 <= reg && reg < 16)
1675 return byte_cpu_regs[reg];
1676 return "noreg";
1677}
1678
1679
1680const char* NameConverter::NameOfXMMRegister(int reg) const {
1681 if (0 <= reg && reg < 16)
1682 return xmm_regs[reg];
1683 return "noxmmreg";
1684}
1685
1686
1687const char* NameConverter::NameInCode(byte* addr) const {
1688 // X64 does not embed debug strings at the moment.
1689 UNREACHABLE();
1690 return "";
1691}
1692
1693//------------------------------------------------------------------------------
1694
1695Disassembler::Disassembler(const NameConverter& converter)
1696 : converter_(converter) { }
1697
1698Disassembler::~Disassembler() { }
1699
1700
1701int Disassembler::InstructionDecode(v8::internal::Vector<char> buffer,
1702 byte* instruction) {
1703 DisassemblerX64 d(converter_, CONTINUE_ON_UNIMPLEMENTED_OPCODE);
1704 return d.InstructionDecode(buffer, instruction);
1705}
1706
1707
1708// The X64 assembler does not use constant pools.
1709int Disassembler::ConstantPoolSizeAt(byte* instruction) {
1710 return -1;
1711}
1712
1713
1714void Disassembler::Disassemble(FILE* f, byte* begin, byte* end) {
1715 NameConverter converter;
1716 Disassembler d(converter);
1717 for (byte* pc = begin; pc < end;) {
1718 v8::internal::EmbeddedVector<char, 128> buffer;
1719 buffer[0] = '\0';
1720 byte* prev_pc = pc;
1721 pc += d.InstructionDecode(buffer, pc);
1722 fprintf(f, "%p", prev_pc);
1723 fprintf(f, " ");
1724
1725 for (byte* bp = prev_pc; bp < pc; bp++) {
1726 fprintf(f, "%02x", *bp);
1727 }
Steve Blockd0582a62009-12-15 09:54:21 +00001728 for (int i = 6 - static_cast<int>(pc - prev_pc); i >= 0; i--) {
Steve Blocka7e24c12009-10-30 11:49:00 +00001729 fprintf(f, " ");
1730 }
1731 fprintf(f, " %s\n", buffer.start());
1732 }
1733}
1734
1735} // namespace disasm
Leon Clarkef7060e22010-06-03 12:02:55 +01001736
1737#endif // V8_TARGET_ARCH_X64