mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2025-01-27 06:54:30 +00:00
instead of migrating it to the MC instruction encoder, just
rip out the implementation of X86InstrInfo::GetInstSizeInBytes. The code being ripped out just implemented a copy and hacked up version of the (old) instruction encoder, and is buggy and terrible in other ways. Since "GetInstSizeInBytes" is really only there to support the JIT's "NeedsExactSize" hook (which noone is using), just rip out the code. I will rip out the NeedsExactSize hook next. This resolves rdar://7617809 - switch X86InstrInfo::GetInstSizeInBytes to use X86MCCodeEmitter llvm-svn: 109149
This commit is contained in:
parent
f53de95e64
commit
367aa754b1
@ -146,6 +146,103 @@ bool Emitter<CodeEmitter>::runOnMachineFunction(MachineFunction &MF) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// determineREX - Determine if the MachineInstr has to be encoded with a X86-64
|
||||
/// REX prefix which specifies 1) 64-bit instructions, 2) non-default operand
|
||||
/// size, and 3) use of X86-64 extended registers.
|
||||
static unsigned determineREX(const MachineInstr &MI) {
|
||||
unsigned REX = 0;
|
||||
const TargetInstrDesc &Desc = MI.getDesc();
|
||||
|
||||
// Pseudo instructions do not need REX prefix byte.
|
||||
if ((Desc.TSFlags & X86II::FormMask) == X86II::Pseudo)
|
||||
return 0;
|
||||
if (Desc.TSFlags & X86II::REX_W)
|
||||
REX |= 1 << 3;
|
||||
|
||||
unsigned NumOps = Desc.getNumOperands();
|
||||
if (NumOps) {
|
||||
bool isTwoAddr = NumOps > 1 &&
|
||||
Desc.getOperandConstraint(1, TOI::TIED_TO) != -1;
|
||||
|
||||
// If it accesses SPL, BPL, SIL, or DIL, then it requires a 0x40 REX prefix.
|
||||
unsigned i = isTwoAddr ? 1 : 0;
|
||||
for (unsigned e = NumOps; i != e; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (MO.isReg()) {
|
||||
unsigned Reg = MO.getReg();
|
||||
if (X86InstrInfo::isX86_64NonExtLowByteReg(Reg))
|
||||
REX |= 0x40;
|
||||
}
|
||||
}
|
||||
|
||||
switch (Desc.TSFlags & X86II::FormMask) {
|
||||
case X86II::MRMInitReg:
|
||||
if (X86InstrInfo::isX86_64ExtendedReg(MI.getOperand(0)))
|
||||
REX |= (1 << 0) | (1 << 2);
|
||||
break;
|
||||
case X86II::MRMSrcReg: {
|
||||
if (X86InstrInfo::isX86_64ExtendedReg(MI.getOperand(0)))
|
||||
REX |= 1 << 2;
|
||||
i = isTwoAddr ? 2 : 1;
|
||||
for (unsigned e = NumOps; i != e; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (X86InstrInfo::isX86_64ExtendedReg(MO))
|
||||
REX |= 1 << 0;
|
||||
}
|
||||
break;
|
||||
}
|
||||
case X86II::MRMSrcMem: {
|
||||
if (X86InstrInfo::isX86_64ExtendedReg(MI.getOperand(0)))
|
||||
REX |= 1 << 2;
|
||||
unsigned Bit = 0;
|
||||
i = isTwoAddr ? 2 : 1;
|
||||
for (; i != NumOps; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (MO.isReg()) {
|
||||
if (X86InstrInfo::isX86_64ExtendedReg(MO))
|
||||
REX |= 1 << Bit;
|
||||
Bit++;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
case X86II::MRM0m: case X86II::MRM1m:
|
||||
case X86II::MRM2m: case X86II::MRM3m:
|
||||
case X86II::MRM4m: case X86II::MRM5m:
|
||||
case X86II::MRM6m: case X86II::MRM7m:
|
||||
case X86II::MRMDestMem: {
|
||||
unsigned e = (isTwoAddr ? X86::AddrNumOperands+1 : X86::AddrNumOperands);
|
||||
i = isTwoAddr ? 1 : 0;
|
||||
if (NumOps > e && X86InstrInfo::isX86_64ExtendedReg(MI.getOperand(e)))
|
||||
REX |= 1 << 2;
|
||||
unsigned Bit = 0;
|
||||
for (; i != e; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (MO.isReg()) {
|
||||
if (X86InstrInfo::isX86_64ExtendedReg(MO))
|
||||
REX |= 1 << Bit;
|
||||
Bit++;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
default: {
|
||||
if (X86InstrInfo::isX86_64ExtendedReg(MI.getOperand(0)))
|
||||
REX |= 1 << 0;
|
||||
i = isTwoAddr ? 2 : 1;
|
||||
for (unsigned e = NumOps; i != e; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (X86InstrInfo::isX86_64ExtendedReg(MO))
|
||||
REX |= 1 << 2;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return REX;
|
||||
}
|
||||
|
||||
|
||||
/// emitPCRelativeBlockAddress - This method keeps track of the information
|
||||
/// necessary to resolve the address of this block later and emits a dummy
|
||||
/// value.
|
||||
@ -569,7 +666,7 @@ void Emitter<CodeEmitter>::emitInstruction(const MachineInstr &MI,
|
||||
|
||||
// Handle REX prefix.
|
||||
if (Is64BitMode) {
|
||||
if (unsigned REX = X86InstrInfo::determineREX(MI))
|
||||
if (unsigned REX = determineREX(MI))
|
||||
MCE.emitByte(0x40 | REX);
|
||||
}
|
||||
|
||||
|
@ -2945,558 +2945,9 @@ bool X86InstrInfo::isX86_64ExtendedReg(unsigned RegNo) {
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
/// determineREX - Determine if the MachineInstr has to be encoded with a X86-64
|
||||
/// REX prefix which specifies 1) 64-bit instructions, 2) non-default operand
|
||||
/// size, and 3) use of X86-64 extended registers.
|
||||
unsigned X86InstrInfo::determineREX(const MachineInstr &MI) {
|
||||
unsigned REX = 0;
|
||||
const TargetInstrDesc &Desc = MI.getDesc();
|
||||
|
||||
// Pseudo instructions do not need REX prefix byte.
|
||||
if ((Desc.TSFlags & X86II::FormMask) == X86II::Pseudo)
|
||||
return 0;
|
||||
if (Desc.TSFlags & X86II::REX_W)
|
||||
REX |= 1 << 3;
|
||||
|
||||
unsigned NumOps = Desc.getNumOperands();
|
||||
if (NumOps) {
|
||||
bool isTwoAddr = NumOps > 1 &&
|
||||
Desc.getOperandConstraint(1, TOI::TIED_TO) != -1;
|
||||
|
||||
// If it accesses SPL, BPL, SIL, or DIL, then it requires a 0x40 REX prefix.
|
||||
unsigned i = isTwoAddr ? 1 : 0;
|
||||
for (unsigned e = NumOps; i != e; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (MO.isReg()) {
|
||||
unsigned Reg = MO.getReg();
|
||||
if (isX86_64NonExtLowByteReg(Reg))
|
||||
REX |= 0x40;
|
||||
}
|
||||
}
|
||||
|
||||
switch (Desc.TSFlags & X86II::FormMask) {
|
||||
case X86II::MRMInitReg:
|
||||
if (isX86_64ExtendedReg(MI.getOperand(0)))
|
||||
REX |= (1 << 0) | (1 << 2);
|
||||
break;
|
||||
case X86II::MRMSrcReg: {
|
||||
if (isX86_64ExtendedReg(MI.getOperand(0)))
|
||||
REX |= 1 << 2;
|
||||
i = isTwoAddr ? 2 : 1;
|
||||
for (unsigned e = NumOps; i != e; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (isX86_64ExtendedReg(MO))
|
||||
REX |= 1 << 0;
|
||||
}
|
||||
break;
|
||||
}
|
||||
case X86II::MRMSrcMem: {
|
||||
if (isX86_64ExtendedReg(MI.getOperand(0)))
|
||||
REX |= 1 << 2;
|
||||
unsigned Bit = 0;
|
||||
i = isTwoAddr ? 2 : 1;
|
||||
for (; i != NumOps; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (MO.isReg()) {
|
||||
if (isX86_64ExtendedReg(MO))
|
||||
REX |= 1 << Bit;
|
||||
Bit++;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
case X86II::MRM0m: case X86II::MRM1m:
|
||||
case X86II::MRM2m: case X86II::MRM3m:
|
||||
case X86II::MRM4m: case X86II::MRM5m:
|
||||
case X86II::MRM6m: case X86II::MRM7m:
|
||||
case X86II::MRMDestMem: {
|
||||
unsigned e = (isTwoAddr ? X86::AddrNumOperands+1 : X86::AddrNumOperands);
|
||||
i = isTwoAddr ? 1 : 0;
|
||||
if (NumOps > e && isX86_64ExtendedReg(MI.getOperand(e)))
|
||||
REX |= 1 << 2;
|
||||
unsigned Bit = 0;
|
||||
for (; i != e; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (MO.isReg()) {
|
||||
if (isX86_64ExtendedReg(MO))
|
||||
REX |= 1 << Bit;
|
||||
Bit++;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
default: {
|
||||
if (isX86_64ExtendedReg(MI.getOperand(0)))
|
||||
REX |= 1 << 0;
|
||||
i = isTwoAddr ? 2 : 1;
|
||||
for (unsigned e = NumOps; i != e; ++i) {
|
||||
const MachineOperand& MO = MI.getOperand(i);
|
||||
if (isX86_64ExtendedReg(MO))
|
||||
REX |= 1 << 2;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return REX;
|
||||
}
|
||||
|
||||
/// sizePCRelativeBlockAddress - This method returns the size of a PC
|
||||
/// relative block address instruction
|
||||
///
|
||||
static unsigned sizePCRelativeBlockAddress() {
|
||||
return 4;
|
||||
}
|
||||
|
||||
/// sizeGlobalAddress - Give the size of the emission of this global address
|
||||
///
|
||||
static unsigned sizeGlobalAddress(bool dword) {
|
||||
return dword ? 8 : 4;
|
||||
}
|
||||
|
||||
/// sizeConstPoolAddress - Give the size of the emission of this constant
|
||||
/// pool address
|
||||
///
|
||||
static unsigned sizeConstPoolAddress(bool dword) {
|
||||
return dword ? 8 : 4;
|
||||
}
|
||||
|
||||
/// sizeExternalSymbolAddress - Give the size of the emission of this external
|
||||
/// symbol
|
||||
///
|
||||
static unsigned sizeExternalSymbolAddress(bool dword) {
|
||||
return dword ? 8 : 4;
|
||||
}
|
||||
|
||||
/// sizeJumpTableAddress - Give the size of the emission of this jump
|
||||
/// table address
|
||||
///
|
||||
static unsigned sizeJumpTableAddress(bool dword) {
|
||||
return dword ? 8 : 4;
|
||||
}
|
||||
|
||||
static unsigned sizeConstant(unsigned Size) {
|
||||
return Size;
|
||||
}
|
||||
|
||||
static unsigned sizeRegModRMByte(){
|
||||
return 1;
|
||||
}
|
||||
|
||||
static unsigned sizeSIBByte(){
|
||||
return 1;
|
||||
}
|
||||
|
||||
static unsigned getDisplacementFieldSize(const MachineOperand *RelocOp) {
|
||||
unsigned FinalSize = 0;
|
||||
// If this is a simple integer displacement that doesn't require a relocation.
|
||||
if (!RelocOp) {
|
||||
FinalSize += sizeConstant(4);
|
||||
return FinalSize;
|
||||
}
|
||||
|
||||
// Otherwise, this is something that requires a relocation.
|
||||
if (RelocOp->isGlobal()) {
|
||||
FinalSize += sizeGlobalAddress(false);
|
||||
} else if (RelocOp->isCPI()) {
|
||||
FinalSize += sizeConstPoolAddress(false);
|
||||
} else if (RelocOp->isJTI()) {
|
||||
FinalSize += sizeJumpTableAddress(false);
|
||||
} else {
|
||||
llvm_unreachable("Unknown value to relocate!");
|
||||
}
|
||||
return FinalSize;
|
||||
}
|
||||
|
||||
static unsigned getMemModRMByteSize(const MachineInstr &MI, unsigned Op,
|
||||
bool IsPIC, bool Is64BitMode) {
|
||||
const MachineOperand &Op3 = MI.getOperand(Op+3);
|
||||
int DispVal = 0;
|
||||
const MachineOperand *DispForReloc = 0;
|
||||
unsigned FinalSize = 0;
|
||||
|
||||
// Figure out what sort of displacement we have to handle here.
|
||||
if (Op3.isGlobal()) {
|
||||
DispForReloc = &Op3;
|
||||
} else if (Op3.isCPI()) {
|
||||
if (Is64BitMode || IsPIC) {
|
||||
DispForReloc = &Op3;
|
||||
} else {
|
||||
DispVal = 1;
|
||||
}
|
||||
} else if (Op3.isJTI()) {
|
||||
if (Is64BitMode || IsPIC) {
|
||||
DispForReloc = &Op3;
|
||||
} else {
|
||||
DispVal = 1;
|
||||
}
|
||||
} else {
|
||||
DispVal = 1;
|
||||
}
|
||||
|
||||
const MachineOperand &Base = MI.getOperand(Op);
|
||||
const MachineOperand &IndexReg = MI.getOperand(Op+2);
|
||||
|
||||
unsigned BaseReg = Base.getReg();
|
||||
|
||||
// Is a SIB byte needed?
|
||||
if ((!Is64BitMode || DispForReloc || BaseReg != 0) &&
|
||||
IndexReg.getReg() == 0 &&
|
||||
(BaseReg == 0 || X86RegisterInfo::getX86RegNum(BaseReg) != N86::ESP)) {
|
||||
if (BaseReg == 0) { // Just a displacement?
|
||||
// Emit special case [disp32] encoding
|
||||
++FinalSize;
|
||||
FinalSize += getDisplacementFieldSize(DispForReloc);
|
||||
} else {
|
||||
unsigned BaseRegNo = X86RegisterInfo::getX86RegNum(BaseReg);
|
||||
if (!DispForReloc && DispVal == 0 && BaseRegNo != N86::EBP) {
|
||||
// Emit simple indirect register encoding... [EAX] f.e.
|
||||
++FinalSize;
|
||||
// Be pessimistic and assume it's a disp32, not a disp8
|
||||
} else {
|
||||
// Emit the most general non-SIB encoding: [REG+disp32]
|
||||
++FinalSize;
|
||||
FinalSize += getDisplacementFieldSize(DispForReloc);
|
||||
}
|
||||
}
|
||||
|
||||
} else { // We need a SIB byte, so start by outputting the ModR/M byte first
|
||||
assert(IndexReg.getReg() != X86::ESP &&
|
||||
IndexReg.getReg() != X86::RSP && "Cannot use ESP as index reg!");
|
||||
|
||||
bool ForceDisp32 = false;
|
||||
if (BaseReg == 0 || DispForReloc) {
|
||||
// Emit the normal disp32 encoding.
|
||||
++FinalSize;
|
||||
ForceDisp32 = true;
|
||||
} else {
|
||||
++FinalSize;
|
||||
}
|
||||
|
||||
FinalSize += sizeSIBByte();
|
||||
|
||||
// Do we need to output a displacement?
|
||||
if (DispVal != 0 || ForceDisp32) {
|
||||
FinalSize += getDisplacementFieldSize(DispForReloc);
|
||||
}
|
||||
}
|
||||
return FinalSize;
|
||||
}
|
||||
|
||||
|
||||
static unsigned GetInstSizeWithDesc(const MachineInstr &MI,
|
||||
const TargetInstrDesc *Desc,
|
||||
bool IsPIC, bool Is64BitMode) {
|
||||
|
||||
unsigned Opcode = Desc->Opcode;
|
||||
unsigned FinalSize = 0;
|
||||
|
||||
// Emit the lock opcode prefix as needed.
|
||||
if (Desc->TSFlags & X86II::LOCK) ++FinalSize;
|
||||
|
||||
// Emit segment override opcode prefix as needed.
|
||||
switch (Desc->TSFlags & X86II::SegOvrMask) {
|
||||
case X86II::FS:
|
||||
case X86II::GS:
|
||||
++FinalSize;
|
||||
break;
|
||||
default: llvm_unreachable("Invalid segment!");
|
||||
case 0: break; // No segment override!
|
||||
}
|
||||
|
||||
// Emit the repeat opcode prefix as needed.
|
||||
if ((Desc->TSFlags & X86II::Op0Mask) == X86II::REP) ++FinalSize;
|
||||
|
||||
// Emit the operand size opcode prefix as needed.
|
||||
if (Desc->TSFlags & X86II::OpSize) ++FinalSize;
|
||||
|
||||
// Emit the address size opcode prefix as needed.
|
||||
if (Desc->TSFlags & X86II::AdSize) ++FinalSize;
|
||||
|
||||
bool Need0FPrefix = false;
|
||||
switch (Desc->TSFlags & X86II::Op0Mask) {
|
||||
case X86II::TB: // Two-byte opcode prefix
|
||||
case X86II::T8: // 0F 38
|
||||
case X86II::TA: // 0F 3A
|
||||
Need0FPrefix = true;
|
||||
break;
|
||||
case X86II::TF: // F2 0F 38
|
||||
++FinalSize;
|
||||
Need0FPrefix = true;
|
||||
break;
|
||||
case X86II::REP: break; // already handled.
|
||||
case X86II::XS: // F3 0F
|
||||
++FinalSize;
|
||||
Need0FPrefix = true;
|
||||
break;
|
||||
case X86II::XD: // F2 0F
|
||||
++FinalSize;
|
||||
Need0FPrefix = true;
|
||||
break;
|
||||
case X86II::D8: case X86II::D9: case X86II::DA: case X86II::DB:
|
||||
case X86II::DC: case X86II::DD: case X86II::DE: case X86II::DF:
|
||||
++FinalSize;
|
||||
break; // Two-byte opcode prefix
|
||||
default: llvm_unreachable("Invalid prefix!");
|
||||
case 0: break; // No prefix!
|
||||
}
|
||||
|
||||
if (Is64BitMode) {
|
||||
// REX prefix
|
||||
unsigned REX = X86InstrInfo::determineREX(MI);
|
||||
if (REX)
|
||||
++FinalSize;
|
||||
}
|
||||
|
||||
// 0x0F escape code must be emitted just before the opcode.
|
||||
if (Need0FPrefix)
|
||||
++FinalSize;
|
||||
|
||||
switch (Desc->TSFlags & X86II::Op0Mask) {
|
||||
case X86II::T8: // 0F 38
|
||||
++FinalSize;
|
||||
break;
|
||||
case X86II::TA: // 0F 3A
|
||||
++FinalSize;
|
||||
break;
|
||||
case X86II::TF: // F2 0F 38
|
||||
++FinalSize;
|
||||
break;
|
||||
}
|
||||
|
||||
// If this is a two-address instruction, skip one of the register operands.
|
||||
unsigned NumOps = Desc->getNumOperands();
|
||||
unsigned CurOp = 0;
|
||||
if (NumOps > 1 && Desc->getOperandConstraint(1, TOI::TIED_TO) != -1)
|
||||
CurOp++;
|
||||
else if (NumOps > 2 && Desc->getOperandConstraint(NumOps-1, TOI::TIED_TO)== 0)
|
||||
// Skip the last source operand that is tied_to the dest reg. e.g. LXADD32
|
||||
--NumOps;
|
||||
|
||||
switch (Desc->TSFlags & X86II::FormMask) {
|
||||
default: llvm_unreachable("Unknown FormMask value in X86 MachineCodeEmitter!");
|
||||
case X86II::Pseudo:
|
||||
// Remember the current PC offset, this is the PIC relocation
|
||||
// base address.
|
||||
switch (Opcode) {
|
||||
default:
|
||||
break;
|
||||
case TargetOpcode::INLINEASM: {
|
||||
const MachineFunction *MF = MI.getParent()->getParent();
|
||||
const TargetInstrInfo &TII = *MF->getTarget().getInstrInfo();
|
||||
FinalSize += TII.getInlineAsmLength(MI.getOperand(0).getSymbolName(),
|
||||
*MF->getTarget().getMCAsmInfo());
|
||||
break;
|
||||
}
|
||||
case TargetOpcode::PROLOG_LABEL:
|
||||
case TargetOpcode::EH_LABEL:
|
||||
case TargetOpcode::DBG_VALUE:
|
||||
break;
|
||||
case TargetOpcode::IMPLICIT_DEF:
|
||||
case TargetOpcode::KILL:
|
||||
break;
|
||||
case X86::MOVPC32r: {
|
||||
// This emits the "call" portion of this pseudo instruction.
|
||||
++FinalSize;
|
||||
FinalSize += sizeConstant(X86II::getSizeOfImm(Desc->TSFlags));
|
||||
break;
|
||||
}
|
||||
}
|
||||
CurOp = NumOps;
|
||||
break;
|
||||
case X86II::RawFrm:
|
||||
++FinalSize;
|
||||
|
||||
if (CurOp != NumOps) {
|
||||
const MachineOperand &MO = MI.getOperand(CurOp++);
|
||||
if (MO.isMBB()) {
|
||||
FinalSize += sizePCRelativeBlockAddress();
|
||||
} else if (MO.isGlobal()) {
|
||||
FinalSize += sizeGlobalAddress(false);
|
||||
} else if (MO.isSymbol()) {
|
||||
FinalSize += sizeExternalSymbolAddress(false);
|
||||
} else if (MO.isImm()) {
|
||||
FinalSize += sizeConstant(X86II::getSizeOfImm(Desc->TSFlags));
|
||||
} else {
|
||||
llvm_unreachable("Unknown RawFrm operand!");
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case X86II::AddRegFrm:
|
||||
++FinalSize;
|
||||
++CurOp;
|
||||
|
||||
if (CurOp != NumOps) {
|
||||
const MachineOperand &MO1 = MI.getOperand(CurOp++);
|
||||
unsigned Size = X86II::getSizeOfImm(Desc->TSFlags);
|
||||
if (MO1.isImm())
|
||||
FinalSize += sizeConstant(Size);
|
||||
else {
|
||||
bool dword = false;
|
||||
if (Opcode == X86::MOV64ri)
|
||||
dword = true;
|
||||
if (MO1.isGlobal()) {
|
||||
FinalSize += sizeGlobalAddress(dword);
|
||||
} else if (MO1.isSymbol())
|
||||
FinalSize += sizeExternalSymbolAddress(dword);
|
||||
else if (MO1.isCPI())
|
||||
FinalSize += sizeConstPoolAddress(dword);
|
||||
else if (MO1.isJTI())
|
||||
FinalSize += sizeJumpTableAddress(dword);
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case X86II::MRMDestReg: {
|
||||
++FinalSize;
|
||||
FinalSize += sizeRegModRMByte();
|
||||
CurOp += 2;
|
||||
if (CurOp != NumOps) {
|
||||
++CurOp;
|
||||
FinalSize += sizeConstant(X86II::getSizeOfImm(Desc->TSFlags));
|
||||
}
|
||||
break;
|
||||
}
|
||||
case X86II::MRMDestMem: {
|
||||
++FinalSize;
|
||||
FinalSize += getMemModRMByteSize(MI, CurOp, IsPIC, Is64BitMode);
|
||||
CurOp += X86::AddrNumOperands + 1;
|
||||
if (CurOp != NumOps) {
|
||||
++CurOp;
|
||||
FinalSize += sizeConstant(X86II::getSizeOfImm(Desc->TSFlags));
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case X86II::MRMSrcReg:
|
||||
++FinalSize;
|
||||
FinalSize += sizeRegModRMByte();
|
||||
CurOp += 2;
|
||||
if (CurOp != NumOps) {
|
||||
++CurOp;
|
||||
FinalSize += sizeConstant(X86II::getSizeOfImm(Desc->TSFlags));
|
||||
}
|
||||
break;
|
||||
|
||||
case X86II::MRMSrcMem: {
|
||||
++FinalSize;
|
||||
FinalSize += getMemModRMByteSize(MI, CurOp+1, IsPIC, Is64BitMode);
|
||||
CurOp += X86::AddrNumOperands + 1;
|
||||
if (CurOp != NumOps) {
|
||||
++CurOp;
|
||||
FinalSize += sizeConstant(X86II::getSizeOfImm(Desc->TSFlags));
|
||||
}
|
||||
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:
|
||||
++FinalSize;
|
||||
if (Desc->getOpcode() == X86::LFENCE ||
|
||||
Desc->getOpcode() == X86::MFENCE) {
|
||||
// Special handling of lfence and mfence;
|
||||
FinalSize += sizeRegModRMByte();
|
||||
} else if (Desc->getOpcode() == X86::MONITOR ||
|
||||
Desc->getOpcode() == X86::MWAIT) {
|
||||
// Special handling of monitor and mwait.
|
||||
FinalSize += sizeRegModRMByte() + 1; // +1 for the opcode.
|
||||
} else {
|
||||
++CurOp;
|
||||
FinalSize += sizeRegModRMByte();
|
||||
}
|
||||
|
||||
if (CurOp != NumOps) {
|
||||
const MachineOperand &MO1 = MI.getOperand(CurOp++);
|
||||
unsigned Size = X86II::getSizeOfImm(Desc->TSFlags);
|
||||
if (MO1.isImm())
|
||||
FinalSize += sizeConstant(Size);
|
||||
else {
|
||||
bool dword = false;
|
||||
if (Opcode == X86::MOV64ri32)
|
||||
dword = true;
|
||||
if (MO1.isGlobal()) {
|
||||
FinalSize += sizeGlobalAddress(dword);
|
||||
} else if (MO1.isSymbol())
|
||||
FinalSize += sizeExternalSymbolAddress(dword);
|
||||
else if (MO1.isCPI())
|
||||
FinalSize += sizeConstPoolAddress(dword);
|
||||
else if (MO1.isJTI())
|
||||
FinalSize += sizeJumpTableAddress(dword);
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case X86II::MRM0m: case X86II::MRM1m:
|
||||
case X86II::MRM2m: case X86II::MRM3m:
|
||||
case X86II::MRM4m: case X86II::MRM5m:
|
||||
case X86II::MRM6m: case X86II::MRM7m: {
|
||||
|
||||
++FinalSize;
|
||||
FinalSize += getMemModRMByteSize(MI, CurOp, IsPIC, Is64BitMode);
|
||||
CurOp += X86::AddrNumOperands;
|
||||
|
||||
if (CurOp != NumOps) {
|
||||
const MachineOperand &MO = MI.getOperand(CurOp++);
|
||||
unsigned Size = X86II::getSizeOfImm(Desc->TSFlags);
|
||||
if (MO.isImm())
|
||||
FinalSize += sizeConstant(Size);
|
||||
else {
|
||||
bool dword = false;
|
||||
if (Opcode == X86::MOV64mi32)
|
||||
dword = true;
|
||||
if (MO.isGlobal()) {
|
||||
FinalSize += sizeGlobalAddress(dword);
|
||||
} else if (MO.isSymbol())
|
||||
FinalSize += sizeExternalSymbolAddress(dword);
|
||||
else if (MO.isCPI())
|
||||
FinalSize += sizeConstPoolAddress(dword);
|
||||
else if (MO.isJTI())
|
||||
FinalSize += sizeJumpTableAddress(dword);
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case X86II::MRM_C1:
|
||||
case X86II::MRM_C8:
|
||||
case X86II::MRM_C9:
|
||||
case X86II::MRM_E8:
|
||||
case X86II::MRM_F0:
|
||||
FinalSize += 2;
|
||||
break;
|
||||
}
|
||||
|
||||
case X86II::MRMInitReg:
|
||||
++FinalSize;
|
||||
// Duplicate register, used by things like MOV8r0 (aka xor reg,reg).
|
||||
FinalSize += sizeRegModRMByte();
|
||||
++CurOp;
|
||||
break;
|
||||
}
|
||||
|
||||
if (!Desc->isVariadic() && CurOp != NumOps) {
|
||||
std::string msg;
|
||||
raw_string_ostream Msg(msg);
|
||||
Msg << "Cannot determine size: " << MI;
|
||||
report_fatal_error(Msg.str());
|
||||
}
|
||||
|
||||
|
||||
return FinalSize;
|
||||
}
|
||||
|
||||
|
||||
unsigned X86InstrInfo::GetInstSizeInBytes(const MachineInstr *MI) const {
|
||||
const TargetInstrDesc &Desc = MI->getDesc();
|
||||
bool IsPIC = TM.getRelocationModel() == Reloc::PIC_;
|
||||
bool Is64BitMode = TM.getSubtargetImpl()->is64Bit();
|
||||
unsigned Size = GetInstSizeWithDesc(*MI, &Desc, IsPIC, Is64BitMode);
|
||||
if (Desc.getOpcode() == X86::MOVPC32r)
|
||||
Size += GetInstSizeWithDesc(*MI, &get(X86::POP32r), IsPIC, Is64BitMode);
|
||||
return Size;
|
||||
assert(0 && "X86InstrInfo::GetInstSizeInBytes isn't implemented");
|
||||
abort();
|
||||
}
|
||||
|
||||
/// getGlobalBaseReg - Return a virtual register initialized with the
|
||||
|
@ -820,7 +820,6 @@ public:
|
||||
if (!MO.isReg()) return false;
|
||||
return isX86_64ExtendedReg(MO.getReg());
|
||||
}
|
||||
static unsigned determineREX(const MachineInstr &MI);
|
||||
|
||||
/// isX86_64ExtendedReg - Is the MachineOperand a x86-64 extended (r8 or
|
||||
/// higher) register? e.g. r8, xmm8, xmm13, etc.
|
||||
|
@ -630,31 +630,6 @@ TEST_F(JITTest, AvailableExternallyFunctionIsntCompiled) {
|
||||
<< " not 7 from the IR version.";
|
||||
}
|
||||
|
||||
TEST_F(JITTest, NeedsExactSizeWithManyGlobals) {
|
||||
// PR5291: When the JMM needed the exact size of function bodies before
|
||||
// starting to emit them, the JITEmitter would modify a set while iterating
|
||||
// over it.
|
||||
TheJIT->DisableLazyCompilation(true);
|
||||
RJMM->setSizeRequired(true);
|
||||
|
||||
LoadAssembly("@A = global i32 42 "
|
||||
"@B = global i32* @A "
|
||||
"@C = global i32** @B "
|
||||
"@D = global i32*** @C "
|
||||
"@E = global i32**** @D "
|
||||
"@F = global i32***** @E "
|
||||
"@G = global i32****** @F "
|
||||
"@H = global i32******* @G "
|
||||
"@I = global i32******** @H "
|
||||
"define i32********* @test() { "
|
||||
" ret i32********* @I "
|
||||
"}");
|
||||
Function *testIR = M->getFunction("test");
|
||||
int32_t********* (*test)() = reinterpret_cast<int32_t*********(*)()>(
|
||||
(intptr_t)TheJIT->getPointerToFunction(testIR));
|
||||
EXPECT_EQ(42, *********test());
|
||||
}
|
||||
|
||||
TEST_F(JITTest, EscapedLazyStubStillCallable) {
|
||||
TheJIT->DisableLazyCompilation(false);
|
||||
LoadAssembly("define internal i32 @stubbed() { "
|
||||
|
Loading…
x
Reference in New Issue
Block a user