aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--lib/Target/X86/X86MCCodeEmitter.cpp111
1 files changed, 108 insertions, 3 deletions
diff --git a/lib/Target/X86/X86MCCodeEmitter.cpp b/lib/Target/X86/X86MCCodeEmitter.cpp
index f2a38cc..8b0cb3e 100644
--- a/lib/Target/X86/X86MCCodeEmitter.cpp
+++ b/lib/Target/X86/X86MCCodeEmitter.cpp
@@ -411,7 +411,7 @@ EncodeInstruction(const MCInst &MI, raw_ostream &OS) const {
EmitMemModRMByte(MI, CurOp,
GetX86RegNum(MI.getOperand(CurOp + X86AddrNumOperands)),
0, OS);
- CurOp += X86AddrNumOperands + 1;
+ CurOp += X86AddrNumOperands + 1;
if (CurOp != NumOps)
EmitConstant(MI.getOperand(CurOp++).getImm(),
X86II::getSizeOfImm(TSFlags), OS);
@@ -450,11 +450,116 @@ EncodeInstruction(const MCInst &MI, raw_ostream &OS) const {
X86II::getSizeOfImm(TSFlags), OS);
break;
}
-
+
+ case X86II::MRM0r: case X86II::MRM1r:
+ case X86II::MRM2r: case X86II::MRM3r:
+ case X86II::MRM4r: case X86II::MRM5r:
+ case X86II::MRM6r: case X86II::MRM7r: {
+ EmitByte(BaseOpcode, OS);
+
+ // Special handling of lfence, mfence, monitor, and mwait.
+ // FIXME: This is terrible, they should get proper encoding bits in TSFlags.
+ if (Opcode == X86::LFENCE || Opcode == X86::MFENCE ||
+ Opcode == X86::MONITOR || Opcode == X86::MWAIT) {
+ EmitByte(ModRMByte(3, (TSFlags & X86II::FormMask)-X86II::MRM0r, 0), OS);
+
+ switch (Opcode) {
+ default: break;
+ case X86::MONITOR: EmitByte(0xC8, OS); break;
+ case X86::MWAIT: EmitByte(0xC9, OS); break;
+ }
+ } else {
+ EmitRegModRMByte(MI.getOperand(CurOp++),
+ (TSFlags & X86II::FormMask)-X86II::MRM0r,
+ OS);
+ }
+
+ if (CurOp == NumOps)
+ break;
+
+ const MCOperand &MO1 = MI.getOperand(CurOp++);
+ if (MO1.isImm()) {
+ EmitConstant(MO1.getImm(), X86II::getSizeOfImm(TSFlags), OS);
+ break;
+ }
+
+ assert(0 && "relo unimpl");
+#if 0
+ unsigned rt = Is64BitMode ? X86::reloc_pcrel_word
+ : (IsPIC ? X86::reloc_picrel_word : X86::reloc_absolute_word);
+ if (Opcode == X86::MOV64ri32)
+ rt = X86::reloc_absolute_word_sext; // FIXME: add X86II flag?
+ if (MO1.isGlobal()) {
+ bool Indirect = gvNeedsNonLazyPtr(MO1, TM);
+ emitGlobalAddress(MO1.getGlobal(), rt, MO1.getOffset(), 0,
+ Indirect);
+ } else if (MO1.isSymbol())
+ emitExternalSymbolAddress(MO1.getSymbolName(), rt);
+ else if (MO1.isCPI())
+ emitConstPoolAddress(MO1.getIndex(), rt);
+ else if (MO1.isJTI())
+ emitJumpTableAddress(MO1.getIndex(), rt);
+ break;
+#endif
+ }
+ case X86II::MRM0m: case X86II::MRM1m:
+ case X86II::MRM2m: case X86II::MRM3m:
+ case X86II::MRM4m: case X86II::MRM5m:
+ case X86II::MRM6m: case X86II::MRM7m: {
+ intptr_t PCAdj = 0;
+ if (CurOp + X86AddrNumOperands != NumOps) {
+ if (MI.getOperand(CurOp+X86AddrNumOperands).isImm())
+ PCAdj = X86II::getSizeOfImm(TSFlags);
+ else
+ PCAdj = 4;
+ }
+
+ EmitByte(BaseOpcode, OS);
+ EmitMemModRMByte(MI, CurOp, (TSFlags & X86II::FormMask)-X86II::MRM0m,
+ PCAdj, OS);
+ CurOp += X86AddrNumOperands;
+
+ if (CurOp == NumOps)
+ break;
+
+ const MCOperand &MO = MI.getOperand(CurOp++);
+ if (MO.isImm()) {
+ EmitConstant(MO.getImm(), X86II::getSizeOfImm(TSFlags), OS);
+ break;
+ }
+
+ assert(0 && "relo not handled");
+#if 0
+ unsigned rt = Is64BitMode ? X86::reloc_pcrel_word
+ : (IsPIC ? X86::reloc_picrel_word : X86::reloc_absolute_word);
+ if (Opcode == X86::MOV64mi32)
+ rt = X86::reloc_absolute_word_sext; // FIXME: add X86II flag?
+ if (MO.isGlobal()) {
+ bool Indirect = gvNeedsNonLazyPtr(MO, TM);
+ emitGlobalAddress(MO.getGlobal(), rt, MO.getOffset(), 0,
+ Indirect);
+ } else if (MO.isSymbol())
+ emitExternalSymbolAddress(MO.getSymbolName(), rt);
+ else if (MO.isCPI())
+ emitConstPoolAddress(MO.getIndex(), rt);
+ else if (MO.isJTI())
+ emitJumpTableAddress(MO.getIndex(), rt);
+#endif
+ break;
+ }
+
+ case X86II::MRMInitReg:
+ EmitByte(BaseOpcode, OS);
+ // Duplicate register, used by things like MOV8r0 (aka xor reg,reg).
+ EmitRegModRMByte(MI.getOperand(CurOp),
+ GetX86RegNum(MI.getOperand(CurOp)), OS);
+ ++CurOp;
+ break;
}
#ifndef NDEBUG
- if (!Desc.isVariadic() && CurOp != NumOps) {
+ // FIXME: Verify.
+ if (/*!Desc.isVariadic() &&*/ CurOp != NumOps) {
errs() << "Cannot encode all operands of: ";
MI.dump();
errs() << '\n';