mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-08-14 15:28:20 +00:00
Implement MEMCPY natively in terms of rep movs*
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@19468 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@@ -50,7 +50,6 @@ namespace {
|
|||||||
|
|
||||||
computeRegisterProperties();
|
computeRegisterProperties();
|
||||||
|
|
||||||
setOperationUnsupported(ISD::MEMCPY, MVT::Other);
|
|
||||||
setOperationUnsupported(ISD::MEMMOVE, MVT::Other);
|
setOperationUnsupported(ISD::MEMMOVE, MVT::Other);
|
||||||
|
|
||||||
setOperationUnsupported(ISD::MUL, MVT::i8);
|
setOperationUnsupported(ISD::MUL, MVT::i8);
|
||||||
@@ -2088,6 +2087,51 @@ void ISel::Select(SDOperand N) {
|
|||||||
BuildMI(BB, Opcode, 0);
|
BuildMI(BB, Opcode, 0);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
case ISD::MEMCPY:
|
||||||
|
Select(N.getOperand(0)); // Select the chain.
|
||||||
|
unsigned Align =
|
||||||
|
(unsigned)cast<ConstantSDNode>(Node->getOperand(4))->getValue();
|
||||||
|
if (Align == 0) Align = 1;
|
||||||
|
|
||||||
|
// Turn the byte code into # iterations
|
||||||
|
unsigned CountReg;
|
||||||
|
unsigned Opcode;
|
||||||
|
switch (Align & 3) {
|
||||||
|
case 2: // WORD aligned
|
||||||
|
CountReg = MakeReg(MVT::i32);
|
||||||
|
if (ConstantSDNode *I = dyn_cast<ConstantSDNode>(Node->getOperand(3))) {
|
||||||
|
BuildMI(BB, X86::MOV32ri, 1, CountReg).addImm(I->getValue()/2);
|
||||||
|
} else {
|
||||||
|
unsigned ByteReg = SelectExpr(Node->getOperand(3));
|
||||||
|
BuildMI(BB, X86::SHR32ri, 2, CountReg).addReg(ByteReg).addImm(1);
|
||||||
|
}
|
||||||
|
Opcode = X86::REP_MOVSW;
|
||||||
|
break;
|
||||||
|
case 0: // DWORD aligned
|
||||||
|
CountReg = MakeReg(MVT::i32);
|
||||||
|
if (ConstantSDNode *I = dyn_cast<ConstantSDNode>(Node->getOperand(3))) {
|
||||||
|
BuildMI(BB, X86::MOV32ri, 1, CountReg).addImm(I->getValue()/4);
|
||||||
|
} else {
|
||||||
|
unsigned ByteReg = SelectExpr(Node->getOperand(3));
|
||||||
|
BuildMI(BB, X86::SHR32ri, 2, CountReg).addReg(ByteReg).addImm(2);
|
||||||
|
}
|
||||||
|
Opcode = X86::REP_MOVSD;
|
||||||
|
break;
|
||||||
|
default: // BYTE aligned
|
||||||
|
CountReg = SelectExpr(Node->getOperand(3));
|
||||||
|
Opcode = X86::REP_MOVSB;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// No matter what the alignment is, we put the source in ESI, the
|
||||||
|
// destination in EDI, and the count in ECX.
|
||||||
|
unsigned TmpReg1 = SelectExpr(Node->getOperand(1));
|
||||||
|
unsigned TmpReg2 = SelectExpr(Node->getOperand(2));
|
||||||
|
BuildMI(BB, X86::MOV32rr, 1, X86::ECX).addReg(CountReg);
|
||||||
|
BuildMI(BB, X86::MOV32rr, 1, X86::EDI).addReg(TmpReg1);
|
||||||
|
BuildMI(BB, X86::MOV32rr, 1, X86::ESI).addReg(TmpReg2);
|
||||||
|
BuildMI(BB, Opcode, 0);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
assert(0 && "Should not be reached!");
|
assert(0 && "Should not be reached!");
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user