[GlobalISel] Tail call memcpy/memmove/memset even in the presence of copies
Differentail revision: https://reviews.llvm.org/D105382
diff --git a/llvm/lib/CodeGen/GlobalISel/LegalizerHelper.cpp b/llvm/lib/CodeGen/GlobalISel/LegalizerHelper.cpp
index edd7969..72d4fbe 100644
--- a/llvm/lib/CodeGen/GlobalISel/LegalizerHelper.cpp
+++ b/llvm/lib/CodeGen/GlobalISel/LegalizerHelper.cpp
@@ -480,8 +480,9 @@
/// True if an instruction is in tail position in its caller. Intended for
/// legalizing libcalls as tail calls when possible.
-static bool isLibCallInTailPosition(const TargetInstrInfo &TII,
- MachineInstr &MI) {
+static bool isLibCallInTailPosition(MachineInstr &MI,
+ const TargetInstrInfo &TII,
+ MachineRegisterInfo &MRI) {
MachineBasicBlock &MBB = *MI.getParent();
const Function &F = MBB.getParent()->getFunction();
@@ -500,8 +501,47 @@
CallerAttrs.hasAttribute(AttributeList::ReturnIndex, Attribute::SExt))
return false;
- // Only tail call if the following instruction is a standard return.
+ // Only tail call if the following instruction is a standard return or if we
+ // have a `thisreturn` callee, and a sequence like:
+ //
+ // G_MEMCPY %0, %1, %2
+ // $x0 = COPY %0
+ // RET_ReallyLR implicit $x0
auto Next = next_nodbg(MI.getIterator(), MBB.instr_end());
+ if (Next != MBB.instr_end() && Next->isCopy()) {
+ switch (MI.getOpcode()) {
+ default:
+ llvm_unreachable("unsupported opcode");
+ case TargetOpcode::G_BZERO:
+ return false;
+ case TargetOpcode::G_MEMCPY:
+ case TargetOpcode::G_MEMMOVE:
+ case TargetOpcode::G_MEMSET:
+ break;
+ }
+
+ Register VReg = MI.getOperand(0).getReg();
+ if (!VReg.isVirtual() || VReg != Next->getOperand(1).getReg())
+ return false;
+
+ Register PReg = Next->getOperand(0).getReg();
+ if (!PReg.isPhysical())
+ return false;
+
+ auto Ret = next_nodbg(Next, MBB.instr_end());
+ if (Ret == MBB.instr_end() || !Ret->isReturn())
+ return false;
+
+ if (Ret->getNumImplicitOperands() != 1)
+ return false;
+
+ if (PReg != Ret->getOperand(0).getReg())
+ return false;
+
+ // Skip over the COPY that we just validated.
+ Next = Ret;
+ }
+
if (Next == MBB.instr_end() || TII.isTailCall(*Next) || !Next->isReturn())
return false;
@@ -607,7 +647,7 @@
Info.Callee = MachineOperand::CreateES(Name);
Info.OrigRet = CallLowering::ArgInfo({0}, Type::getVoidTy(Ctx), 0);
Info.IsTailCall = MI.getOperand(MI.getNumOperands() - 1).getImm() &&
- isLibCallInTailPosition(MIRBuilder.getTII(), MI);
+ isLibCallInTailPosition(MI, MIRBuilder.getTII(), MRI);
std::copy(Args.begin(), Args.end(), std::back_inserter(Info.OrigArgs));
if (!CLI.lowerCall(MIRBuilder, Info))
@@ -623,7 +663,8 @@
// isLibCallInTailPosition.
do {
MachineInstr *Next = MI.getNextNode();
- assert(Next && (Next->isReturn() || Next->isDebugInstr()) &&
+ assert(Next &&
+ (Next->isCopy() || Next->isReturn() || Next->isDebugInstr()) &&
"Expected instr following MI to be return or debug inst?");
// We lowered a tail call, so the call is now the return from the block.
// Delete the old return.