mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-15 04:30:12 +00:00
Rename GR8_ABCD to GR8_ABCD_L and create GR8_ABCD_H, and use these
to precisely describe the h-register subreg register classes. Thanks to Jakob Stoklund Olesen for spotting this and for the initial patch! Also, make getStoreRegOpcode and getLoadRegOpcode aware of the needs of h registers. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@70211 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
6241762c5a
commit
4af325d1b4
@ -1645,7 +1645,7 @@ X86InstrInfo::InsertBranch(MachineBasicBlock &MBB, MachineBasicBlock *TBB,
|
|||||||
|
|
||||||
/// isHReg - Test if the given register is a physical h register.
|
/// isHReg - Test if the given register is a physical h register.
|
||||||
static bool isHReg(unsigned Reg) {
|
static bool isHReg(unsigned Reg) {
|
||||||
return Reg == X86::AH || Reg == X86::BH || Reg == X86::CH || Reg == X86::DH;
|
return X86::GR8_ABCD_HRegClass.contains(Reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool X86InstrInfo::copyRegToReg(MachineBasicBlock &MBB,
|
bool X86InstrInfo::copyRegToReg(MachineBasicBlock &MBB,
|
||||||
@ -1674,7 +1674,7 @@ bool X86InstrInfo::copyRegToReg(MachineBasicBlock &MBB,
|
|||||||
} else if (CommonRC == &X86::GR16RegClass) {
|
} else if (CommonRC == &X86::GR16RegClass) {
|
||||||
Opc = X86::MOV16rr;
|
Opc = X86::MOV16rr;
|
||||||
} else if (CommonRC == &X86::GR8RegClass) {
|
} else if (CommonRC == &X86::GR8RegClass) {
|
||||||
// Copying two or from a physical H register on x86-64 requires a NOREX
|
// Copying to or from a physical H register on x86-64 requires a NOREX
|
||||||
// move. Otherwise use a normal move.
|
// move. Otherwise use a normal move.
|
||||||
if ((isHReg(DestReg) || isHReg(SrcReg)) &&
|
if ((isHReg(DestReg) || isHReg(SrcReg)) &&
|
||||||
TM.getSubtarget<X86Subtarget>().is64Bit())
|
TM.getSubtarget<X86Subtarget>().is64Bit())
|
||||||
@ -1687,7 +1687,12 @@ bool X86InstrInfo::copyRegToReg(MachineBasicBlock &MBB,
|
|||||||
Opc = X86::MOV32rr;
|
Opc = X86::MOV32rr;
|
||||||
} else if (CommonRC == &X86::GR16_ABCDRegClass) {
|
} else if (CommonRC == &X86::GR16_ABCDRegClass) {
|
||||||
Opc = X86::MOV16rr;
|
Opc = X86::MOV16rr;
|
||||||
} else if (CommonRC == &X86::GR8_ABCDRegClass) {
|
} else if (CommonRC == &X86::GR8_ABCD_LRegClass) {
|
||||||
|
Opc = X86::MOV8rr;
|
||||||
|
} else if (CommonRC == &X86::GR8_ABCD_HRegClass) {
|
||||||
|
if (TM.getSubtarget<X86Subtarget>().is64Bit())
|
||||||
|
Opc = X86::MOV8rr_NOREX;
|
||||||
|
else
|
||||||
Opc = X86::MOV8rr;
|
Opc = X86::MOV8rr;
|
||||||
} else if (CommonRC == &X86::GR64_NOREXRegClass) {
|
} else if (CommonRC == &X86::GR64_NOREXRegClass) {
|
||||||
Opc = X86::MOV64rr;
|
Opc = X86::MOV64rr;
|
||||||
@ -1791,8 +1796,10 @@ bool X86InstrInfo::copyRegToReg(MachineBasicBlock &MBB,
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned getStoreRegOpcode(const TargetRegisterClass *RC,
|
static unsigned getStoreRegOpcode(unsigned SrcReg,
|
||||||
bool isStackAligned) {
|
const TargetRegisterClass *RC,
|
||||||
|
bool isStackAligned,
|
||||||
|
TargetMachine &TM) {
|
||||||
unsigned Opc = 0;
|
unsigned Opc = 0;
|
||||||
if (RC == &X86::GR64RegClass) {
|
if (RC == &X86::GR64RegClass) {
|
||||||
Opc = X86::MOV64mr;
|
Opc = X86::MOV64mr;
|
||||||
@ -1801,6 +1808,12 @@ static unsigned getStoreRegOpcode(const TargetRegisterClass *RC,
|
|||||||
} else if (RC == &X86::GR16RegClass) {
|
} else if (RC == &X86::GR16RegClass) {
|
||||||
Opc = X86::MOV16mr;
|
Opc = X86::MOV16mr;
|
||||||
} else if (RC == &X86::GR8RegClass) {
|
} else if (RC == &X86::GR8RegClass) {
|
||||||
|
// Copying to or from a physical H register on x86-64 requires a NOREX
|
||||||
|
// move. Otherwise use a normal move.
|
||||||
|
if (isHReg(SrcReg) &&
|
||||||
|
TM.getSubtarget<X86Subtarget>().is64Bit())
|
||||||
|
Opc = X86::MOV8mr_NOREX;
|
||||||
|
else
|
||||||
Opc = X86::MOV8mr;
|
Opc = X86::MOV8mr;
|
||||||
} else if (RC == &X86::GR64_ABCDRegClass) {
|
} else if (RC == &X86::GR64_ABCDRegClass) {
|
||||||
Opc = X86::MOV64mr;
|
Opc = X86::MOV64mr;
|
||||||
@ -1808,7 +1821,12 @@ static unsigned getStoreRegOpcode(const TargetRegisterClass *RC,
|
|||||||
Opc = X86::MOV32mr;
|
Opc = X86::MOV32mr;
|
||||||
} else if (RC == &X86::GR16_ABCDRegClass) {
|
} else if (RC == &X86::GR16_ABCDRegClass) {
|
||||||
Opc = X86::MOV16mr;
|
Opc = X86::MOV16mr;
|
||||||
} else if (RC == &X86::GR8_ABCDRegClass) {
|
} else if (RC == &X86::GR8_ABCD_LRegClass) {
|
||||||
|
Opc = X86::MOV8mr;
|
||||||
|
} else if (RC == &X86::GR8_ABCD_HRegClass) {
|
||||||
|
if (TM.getSubtarget<X86Subtarget>().is64Bit())
|
||||||
|
Opc = X86::MOV8mr_NOREX;
|
||||||
|
else
|
||||||
Opc = X86::MOV8mr;
|
Opc = X86::MOV8mr;
|
||||||
} else if (RC == &X86::GR64_NOREXRegClass) {
|
} else if (RC == &X86::GR64_NOREXRegClass) {
|
||||||
Opc = X86::MOV64mr;
|
Opc = X86::MOV64mr;
|
||||||
@ -1848,7 +1866,7 @@ void X86InstrInfo::storeRegToStackSlot(MachineBasicBlock &MBB,
|
|||||||
const MachineFunction &MF = *MBB.getParent();
|
const MachineFunction &MF = *MBB.getParent();
|
||||||
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
||||||
RI.needsStackRealignment(MF);
|
RI.needsStackRealignment(MF);
|
||||||
unsigned Opc = getStoreRegOpcode(RC, isAligned);
|
unsigned Opc = getStoreRegOpcode(SrcReg, RC, isAligned, TM);
|
||||||
DebugLoc DL = DebugLoc::getUnknownLoc();
|
DebugLoc DL = DebugLoc::getUnknownLoc();
|
||||||
if (MI != MBB.end()) DL = MI->getDebugLoc();
|
if (MI != MBB.end()) DL = MI->getDebugLoc();
|
||||||
addFrameReference(BuildMI(MBB, MI, DL, get(Opc)), FrameIdx)
|
addFrameReference(BuildMI(MBB, MI, DL, get(Opc)), FrameIdx)
|
||||||
@ -1862,7 +1880,7 @@ void X86InstrInfo::storeRegToAddr(MachineFunction &MF, unsigned SrcReg,
|
|||||||
SmallVectorImpl<MachineInstr*> &NewMIs) const {
|
SmallVectorImpl<MachineInstr*> &NewMIs) const {
|
||||||
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
||||||
RI.needsStackRealignment(MF);
|
RI.needsStackRealignment(MF);
|
||||||
unsigned Opc = getStoreRegOpcode(RC, isAligned);
|
unsigned Opc = getStoreRegOpcode(SrcReg, RC, isAligned, TM);
|
||||||
DebugLoc DL = DebugLoc::getUnknownLoc();
|
DebugLoc DL = DebugLoc::getUnknownLoc();
|
||||||
MachineInstrBuilder MIB = BuildMI(MF, DL, get(Opc));
|
MachineInstrBuilder MIB = BuildMI(MF, DL, get(Opc));
|
||||||
for (unsigned i = 0, e = Addr.size(); i != e; ++i)
|
for (unsigned i = 0, e = Addr.size(); i != e; ++i)
|
||||||
@ -1871,8 +1889,10 @@ void X86InstrInfo::storeRegToAddr(MachineFunction &MF, unsigned SrcReg,
|
|||||||
NewMIs.push_back(MIB);
|
NewMIs.push_back(MIB);
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned getLoadRegOpcode(const TargetRegisterClass *RC,
|
static unsigned getLoadRegOpcode(unsigned DestReg,
|
||||||
bool isStackAligned) {
|
const TargetRegisterClass *RC,
|
||||||
|
bool isStackAligned,
|
||||||
|
const TargetMachine &TM) {
|
||||||
unsigned Opc = 0;
|
unsigned Opc = 0;
|
||||||
if (RC == &X86::GR64RegClass) {
|
if (RC == &X86::GR64RegClass) {
|
||||||
Opc = X86::MOV64rm;
|
Opc = X86::MOV64rm;
|
||||||
@ -1881,6 +1901,12 @@ static unsigned getLoadRegOpcode(const TargetRegisterClass *RC,
|
|||||||
} else if (RC == &X86::GR16RegClass) {
|
} else if (RC == &X86::GR16RegClass) {
|
||||||
Opc = X86::MOV16rm;
|
Opc = X86::MOV16rm;
|
||||||
} else if (RC == &X86::GR8RegClass) {
|
} else if (RC == &X86::GR8RegClass) {
|
||||||
|
// Copying to or from a physical H register on x86-64 requires a NOREX
|
||||||
|
// move. Otherwise use a normal move.
|
||||||
|
if (isHReg(DestReg) &&
|
||||||
|
TM.getSubtarget<X86Subtarget>().is64Bit())
|
||||||
|
Opc = X86::MOV8rm_NOREX;
|
||||||
|
else
|
||||||
Opc = X86::MOV8rm;
|
Opc = X86::MOV8rm;
|
||||||
} else if (RC == &X86::GR64_ABCDRegClass) {
|
} else if (RC == &X86::GR64_ABCDRegClass) {
|
||||||
Opc = X86::MOV64rm;
|
Opc = X86::MOV64rm;
|
||||||
@ -1888,7 +1914,12 @@ static unsigned getLoadRegOpcode(const TargetRegisterClass *RC,
|
|||||||
Opc = X86::MOV32rm;
|
Opc = X86::MOV32rm;
|
||||||
} else if (RC == &X86::GR16_ABCDRegClass) {
|
} else if (RC == &X86::GR16_ABCDRegClass) {
|
||||||
Opc = X86::MOV16rm;
|
Opc = X86::MOV16rm;
|
||||||
} else if (RC == &X86::GR8_ABCDRegClass) {
|
} else if (RC == &X86::GR8_ABCD_LRegClass) {
|
||||||
|
Opc = X86::MOV8rm;
|
||||||
|
} else if (RC == &X86::GR8_ABCD_HRegClass) {
|
||||||
|
if (TM.getSubtarget<X86Subtarget>().is64Bit())
|
||||||
|
Opc = X86::MOV8rm_NOREX;
|
||||||
|
else
|
||||||
Opc = X86::MOV8rm;
|
Opc = X86::MOV8rm;
|
||||||
} else if (RC == &X86::GR64_NOREXRegClass) {
|
} else if (RC == &X86::GR64_NOREXRegClass) {
|
||||||
Opc = X86::MOV64rm;
|
Opc = X86::MOV64rm;
|
||||||
@ -1928,7 +1959,7 @@ void X86InstrInfo::loadRegFromStackSlot(MachineBasicBlock &MBB,
|
|||||||
const MachineFunction &MF = *MBB.getParent();
|
const MachineFunction &MF = *MBB.getParent();
|
||||||
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
||||||
RI.needsStackRealignment(MF);
|
RI.needsStackRealignment(MF);
|
||||||
unsigned Opc = getLoadRegOpcode(RC, isAligned);
|
unsigned Opc = getLoadRegOpcode(DestReg, RC, isAligned, TM);
|
||||||
DebugLoc DL = DebugLoc::getUnknownLoc();
|
DebugLoc DL = DebugLoc::getUnknownLoc();
|
||||||
if (MI != MBB.end()) DL = MI->getDebugLoc();
|
if (MI != MBB.end()) DL = MI->getDebugLoc();
|
||||||
addFrameReference(BuildMI(MBB, MI, DL, get(Opc), DestReg), FrameIdx);
|
addFrameReference(BuildMI(MBB, MI, DL, get(Opc), DestReg), FrameIdx);
|
||||||
@ -1940,7 +1971,7 @@ void X86InstrInfo::loadRegFromAddr(MachineFunction &MF, unsigned DestReg,
|
|||||||
SmallVectorImpl<MachineInstr*> &NewMIs) const {
|
SmallVectorImpl<MachineInstr*> &NewMIs) const {
|
||||||
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
||||||
RI.needsStackRealignment(MF);
|
RI.needsStackRealignment(MF);
|
||||||
unsigned Opc = getLoadRegOpcode(RC, isAligned);
|
unsigned Opc = getLoadRegOpcode(DestReg, RC, isAligned, TM);
|
||||||
DebugLoc DL = DebugLoc::getUnknownLoc();
|
DebugLoc DL = DebugLoc::getUnknownLoc();
|
||||||
MachineInstrBuilder MIB = BuildMI(MF, DL, get(Opc), DestReg);
|
MachineInstrBuilder MIB = BuildMI(MF, DL, get(Opc), DestReg);
|
||||||
for (unsigned i = 0, e = Addr.size(); i != e; ++i)
|
for (unsigned i = 0, e = Addr.size(); i != e; ++i)
|
||||||
@ -2455,9 +2486,8 @@ X86InstrInfo::unfoldMemoryOperand(SelectionDAG &DAG, SDNode *N,
|
|||||||
MVT VT = *RC->vt_begin();
|
MVT VT = *RC->vt_begin();
|
||||||
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
||||||
RI.needsStackRealignment(MF);
|
RI.needsStackRealignment(MF);
|
||||||
Load = DAG.getTargetNode(getLoadRegOpcode(RC, isAligned), dl,
|
Load = DAG.getTargetNode(getLoadRegOpcode(0, RC, isAligned, TM), dl,
|
||||||
VT, MVT::Other,
|
VT, MVT::Other, &AddrOps[0], AddrOps.size());
|
||||||
&AddrOps[0], AddrOps.size());
|
|
||||||
NewNodes.push_back(Load);
|
NewNodes.push_back(Load);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2489,8 +2519,10 @@ X86InstrInfo::unfoldMemoryOperand(SelectionDAG &DAG, SDNode *N,
|
|||||||
AddrOps.push_back(Chain);
|
AddrOps.push_back(Chain);
|
||||||
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
bool isAligned = (RI.getStackAlignment() >= 16) ||
|
||||||
RI.needsStackRealignment(MF);
|
RI.needsStackRealignment(MF);
|
||||||
SDNode *Store = DAG.getTargetNode(getStoreRegOpcode(DstRC, isAligned), dl,
|
SDNode *Store = DAG.getTargetNode(getStoreRegOpcode(0, DstRC,
|
||||||
MVT::Other, &AddrOps[0], AddrOps.size());
|
isAligned, TM),
|
||||||
|
dl, MVT::Other,
|
||||||
|
&AddrOps[0], AddrOps.size());
|
||||||
NewNodes.push_back(Store);
|
NewNodes.push_back(Store);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -784,9 +784,9 @@ def MOV32mr : I<0x89, MRMDestMem, (outs), (ins i32mem:$dst, GR32:$src),
|
|||||||
"mov{l}\t{$src, $dst|$dst, $src}",
|
"mov{l}\t{$src, $dst|$dst, $src}",
|
||||||
[(store GR32:$src, addr:$dst)]>;
|
[(store GR32:$src, addr:$dst)]>;
|
||||||
|
|
||||||
// Versions of MOV8rr and MOV8mr that use i8mem_NOREX and GR8_NOREX so that they
|
// Versions of MOV8rr, MOV8mr, and MOV8rm that use i8mem_NOREX and GR8_NOREX so
|
||||||
// can be used for copying and storing h registers, which can't be encoded when
|
// that they can be used for copying and storing h registers, which can't be
|
||||||
// a REX prefix is present.
|
// encoded when a REX prefix is present.
|
||||||
let neverHasSideEffects = 1 in
|
let neverHasSideEffects = 1 in
|
||||||
def MOV8rr_NOREX : I<0x88, MRMDestReg,
|
def MOV8rr_NOREX : I<0x88, MRMDestReg,
|
||||||
(outs GR8_NOREX:$dst), (ins GR8_NOREX:$src),
|
(outs GR8_NOREX:$dst), (ins GR8_NOREX:$src),
|
||||||
@ -794,6 +794,10 @@ def MOV8rr_NOREX : I<0x88, MRMDestReg,
|
|||||||
def MOV8mr_NOREX : I<0x88, MRMDestMem,
|
def MOV8mr_NOREX : I<0x88, MRMDestMem,
|
||||||
(outs), (ins i8mem_NOREX:$dst, GR8_NOREX:$src),
|
(outs), (ins i8mem_NOREX:$dst, GR8_NOREX:$src),
|
||||||
"mov{b}\t{$src, $dst|$dst, $src} # NOREX", []>;
|
"mov{b}\t{$src, $dst|$dst, $src} # NOREX", []>;
|
||||||
|
let canFoldAsLoad = 1, isReMaterializable = 1, mayHaveSideEffects = 1 in
|
||||||
|
def MOV8rm_NOREX : I<0x8A, MRMSrcMem,
|
||||||
|
(outs GR8_NOREX:$dst), (ins i8mem_NOREX:$src),
|
||||||
|
"mov{b}\t{$src, $dst|$dst, $src} # NOREX", []>;
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
// Fixed-Register Multiplication and Division Instructions...
|
// Fixed-Register Multiplication and Division Instructions...
|
||||||
|
@ -461,21 +461,24 @@ def GR64 : RegisterClass<"X86", [i64], 64,
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// GR8_ABCD, GR16_ABCD, GR32_ABCD, GR64_ABCD - Subclasses of GR8, GR16, GR32,
|
// GR8_ABCD_L, GR8_ABCD_H, GR16_ABCD, GR32_ABCD, GR64_ABCD - Subclasses of
|
||||||
// and GR64 which contain just the "a" "b", "c", and "d" registers. On x86-32,
|
// GR8, GR16, GR32, and GR64 which contain just the "a" "b", "c", and "d"
|
||||||
// GR16_ABCD and GR32_ABCD are classes for registers that support 8-bit subreg
|
// registers. On x86-32, GR16_ABCD and GR32_ABCD are classes for registers
|
||||||
// operations. On x86-64, GR16_ABCD, GR32_ABCD, and GR64_ABCD are classes for
|
// that support 8-bit subreg operations. On x86-64, GR16_ABCD, GR32_ABCD,
|
||||||
// registers that support 8-bit h-register operations.
|
// and GR64_ABCD are classes for registers that support 8-bit h-register
|
||||||
def GR8_ABCD : RegisterClass<"X86", [i8], 8, [AL, CL, DL, BL]> {
|
// operations.
|
||||||
|
def GR8_ABCD_L : RegisterClass<"X86", [i8], 8, [AL, CL, DL, BL]> {
|
||||||
|
}
|
||||||
|
def GR8_ABCD_H : RegisterClass<"X86", [i8], 8, [AH, CH, DH, BH]> {
|
||||||
}
|
}
|
||||||
def GR16_ABCD : RegisterClass<"X86", [i16], 16, [AX, CX, DX, BX]> {
|
def GR16_ABCD : RegisterClass<"X86", [i16], 16, [AX, CX, DX, BX]> {
|
||||||
let SubRegClassList = [GR8_ABCD, GR8_ABCD];
|
let SubRegClassList = [GR8_ABCD_L, GR8_ABCD_H];
|
||||||
}
|
}
|
||||||
def GR32_ABCD : RegisterClass<"X86", [i32], 32, [EAX, ECX, EDX, EBX]> {
|
def GR32_ABCD : RegisterClass<"X86", [i32], 32, [EAX, ECX, EDX, EBX]> {
|
||||||
let SubRegClassList = [GR8_ABCD, GR8_ABCD, GR16_ABCD];
|
let SubRegClassList = [GR8_ABCD_L, GR8_ABCD_H, GR16_ABCD];
|
||||||
}
|
}
|
||||||
def GR64_ABCD : RegisterClass<"X86", [i64], 64, [RAX, RCX, RDX, RBX]> {
|
def GR64_ABCD : RegisterClass<"X86", [i64], 64, [RAX, RCX, RDX, RBX]> {
|
||||||
let SubRegClassList = [GR8_ABCD, GR8_ABCD, GR16_ABCD, GR32_ABCD];
|
let SubRegClassList = [GR8_ABCD_L, GR8_ABCD_H, GR16_ABCD, GR32_ABCD];
|
||||||
}
|
}
|
||||||
|
|
||||||
// GR8_NOREX, GR16_NOREX, GR32_NOREX, GR64_NOREX - Subclasses of
|
// GR8_NOREX, GR16_NOREX, GR32_NOREX, GR64_NOREX - Subclasses of
|
||||||
|
Loading…
Reference in New Issue
Block a user