Index: llvm/lib/Target/PowerPC/GISel/PPCCallLowering.cpp =================================================================== --- llvm/lib/Target/PowerPC/GISel/PPCCallLowering.cpp +++ llvm/lib/Target/PowerPC/GISel/PPCCallLowering.cpp @@ -13,6 +13,7 @@ //===----------------------------------------------------------------------===// #include "PPCCallLowering.h" +#include "PPCCallingConv.h" #include "PPCISelLowering.h" #include "PPCSubtarget.h" #include "PPCTargetMachine.h" @@ -27,6 +28,35 @@ using namespace llvm; +namespace { + +struct OutgoingArgHandler : public CallLowering::OutgoingValueHandler { + OutgoingArgHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI, + MachineInstrBuilder MIB) + : OutgoingValueHandler(MIRBuilder, MRI), MIB(MIB) {} + + void assignValueToReg(Register ValVReg, Register PhysReg, + CCValAssign VA) override { + MIB.addUse(PhysReg, RegState::Implicit); + Register ExtReg = extendRegister(ValVReg, VA); + MIRBuilder.buildCopy(PhysReg, ExtReg); + } + + void assignValueToAddress(Register ValVReg, Register Addr, LLT MemTy, + MachinePointerInfo &MPO, CCValAssign &VA) override { + llvm_unreachable("unimplemented"); + } + + Register getStackAddress(uint64_t Size, int64_t Offset, + MachinePointerInfo &MPO, + ISD::ArgFlagsTy Flags) override { + llvm_unreachable("unimplemented"); + } + + MachineInstrBuilder MIB; +}; +} + PPCCallLowering::PPCCallLowering(const PPCTargetLowering &TLI) : CallLowering(&TLI) {} @@ -34,13 +64,25 @@ const Value *Val, ArrayRef VRegs, FunctionLoweringInfo &FLI, Register SwiftErrorVReg) const { - assert(((Val && !VRegs.empty()) || (!Val && VRegs.empty())) && - "Return value without a vreg"); - if (VRegs.size() > 0) - return false; - - MIRBuilder.buildInstr(PPC::BLR8); - return true; + auto MIB = MIRBuilder.buildInstrNoInsert(PPC::BLR8); + bool Success = true; + MachineFunction &MF = MIRBuilder.getMF(); + const Function &F = MF.getFunction(); + MachineRegisterInfo &MRI = MF.getRegInfo(); + auto &DL = F.getParent()->getDataLayout(); + if (!VRegs.empty()) { + SmallVector SplitArgs; + ArgInfo OrigArg{VRegs, Val->getType(), 0}; + setArgFlags(OrigArg, AttributeList::ReturnIndex, DL, F); + splitToValueTypes(OrigArg, SplitArgs, DL, F.getCallingConv()); + OutgoingValueAssigner ArgAssigner(RetCC_PPC); + OutgoingArgHandler ArgHandler(MIRBuilder, MRI, MIB); + Success = determineAndHandleAssignments(ArgHandler, ArgAssigner, SplitArgs, + MIRBuilder, F.getCallingConv(), + F.isVarArg()); + } + MIRBuilder.insertInstr(MIB); + return Success; } bool PPCCallLowering::lowerCall(MachineIRBuilder &MIRBuilder, Index: llvm/lib/Target/PowerPC/GISel/PPCGenRegisterBankInfo.def =================================================================== --- /dev/null +++ llvm/lib/Target/PowerPC/GISel/PPCGenRegisterBankInfo.def @@ -0,0 +1,51 @@ +//===- PPCGenRegisterBankInfo.def -------------------------------*- C++ -*-==// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// \file +/// This file defines all the static objects used by PPCRegisterBankInfo. +/// \todo This should be generated by TableGen. +//===----------------------------------------------------------------------===// + +namespace llvm { +RegisterBankInfo::PartialMapping PPCGenRegisterBankInfo::PartMappings[]{ + /* StartIdx, Length, RegBank */ + // 0: GPR 64-bit value. + {0, 64, PPC::GPRRegBank}, +}; + +// ValueMappings. +RegisterBankInfo::ValueMapping PPCGenRegisterBankInfo::ValMappings[]{ + /* BreakDown, NumBreakDowns */ + // 0: invalid + {nullptr, 0}, + // 1: GPR 64-bit value. + {&PPCGenRegisterBankInfo::PartMappings[PMI_GPR64 - PMI_Min], 1}, + {&PPCGenRegisterBankInfo::PartMappings[PMI_GPR64 - PMI_Min], 1}, + {&PPCGenRegisterBankInfo::PartMappings[PMI_GPR64 - PMI_Min], 1}, +}; + +// TODO Too simple! +const RegisterBankInfo::ValueMapping * +PPCGenRegisterBankInfo::getValueMapping(PartialMappingIdx RBIdx) { + assert(RBIdx != PartialMappingIdx::PMI_None && "No mapping needed for that"); + + unsigned ValMappingIdx = RBIdx - PMI_Min; + + return &ValMappings[1 + 3 * ValMappingIdx]; +} + +// TODO Too simple! +const RegisterBankInfo::ValueMapping * +PPCGenRegisterBankInfo::getCopyMapping(unsigned DstBankID, unsigned SrcBankID, + unsigned Size) { + assert(DstBankID < PPC::NumRegisterBanks && "Invalid bank ID"); + assert(SrcBankID < PPC::NumRegisterBanks && "Invalid bank ID"); + + return &ValMappings[1]; +} + +} // namespace llvm Index: llvm/lib/Target/PowerPC/GISel/PPCInstructionSelector.cpp =================================================================== --- llvm/lib/Target/PowerPC/GISel/PPCInstructionSelector.cpp +++ llvm/lib/Target/PowerPC/GISel/PPCInstructionSelector.cpp @@ -75,7 +75,50 @@ { } +static const TargetRegisterClass *guessRegClass(unsigned Reg, + MachineRegisterInfo &MRI, + const TargetRegisterInfo &TRI, + const RegisterBankInfo &RBI) { + const RegisterBank *RegBank = RBI.getRegBank(Reg, MRI, TRI); + assert(RegBank && "Can't get reg bank for virtual register"); + + assert((RegBank->getID() == PPC::GPRRegBankID) && "Unsupported reg bank"); + + return &PPC::G8RCRegClass; +} + +static bool selectCopy(MachineInstr &I, const TargetInstrInfo &TII, + MachineRegisterInfo &MRI, const TargetRegisterInfo &TRI, + const RegisterBankInfo &RBI) { + Register DstReg = I.getOperand(0).getReg(); + if (Register::isPhysicalRegister(DstReg)) + return true; + + const TargetRegisterClass *RC = guessRegClass(DstReg, MRI, TRI, RBI); + + // No need to constrain SrcReg. It will get constrained when + // we hit another of its uses or its defs. + // Copies do not have constraints. + if (!RBI.constrainGenericRegister(DstReg, *RC, MRI)) { + LLVM_DEBUG(dbgs() << "Failed to constrain " << TII.getName(I.getOpcode()) + << " operand\n"); + return false; + } + return true; +} + bool PPCInstructionSelector::select(MachineInstr &I) { + auto &MBB = *I.getParent(); + auto &MF = *MBB.getParent(); + auto &MRI = MF.getRegInfo(); + + if (!isPreISelGenericOpcode(I.getOpcode())) { + if (I.isCopy()) + return selectCopy(I, TII, MRI, TRI, RBI); + + return true; + } + if (selectImpl(I, *CoverageInfo)) return true; return false; Index: llvm/lib/Target/PowerPC/GISel/PPCLegalizerInfo.cpp =================================================================== --- llvm/lib/Target/PowerPC/GISel/PPCLegalizerInfo.cpp +++ llvm/lib/Target/PowerPC/GISel/PPCLegalizerInfo.cpp @@ -18,5 +18,10 @@ using namespace LegalizeActions; PPCLegalizerInfo::PPCLegalizerInfo(const PPCSubtarget &ST) { + using namespace TargetOpcode; + const LLT S64 = LLT::scalar(64); + getActionDefinitionsBuilder(G_IMPLICIT_DEF).legalFor({S64}); + getActionDefinitionsBuilder(G_CONSTANT).legalFor({S64}); + getActionDefinitionsBuilder({G_AND, G_OR, G_XOR}).legalFor({S64}); getLegacyLegalizerInfo().computeTables(); } Index: llvm/lib/Target/PowerPC/GISel/PPCRegisterBankInfo.h =================================================================== --- llvm/lib/Target/PowerPC/GISel/PPCRegisterBankInfo.h +++ llvm/lib/Target/PowerPC/GISel/PPCRegisterBankInfo.h @@ -26,6 +26,32 @@ class PPCGenRegisterBankInfo : public RegisterBankInfo { protected: + enum PartialMappingIdx { + PMI_None = -1, + PMI_GPR64 = 1, + PMI_Min = PMI_GPR64, + }; + + static RegisterBankInfo::PartialMapping PartMappings[]; + static RegisterBankInfo::ValueMapping ValMappings[]; + static PartialMappingIdx BankIDToCopyMapIdx[]; + + /// Get the pointer to the ValueMapping representing the RegisterBank + /// at \p RBIdx. + /// + /// The returned mapping works for instructions with the same kind of + /// operands for up to 3 operands. + /// + /// \pre \p RBIdx != PartialMappingIdx::None + static const RegisterBankInfo::ValueMapping * + getValueMapping(PartialMappingIdx RBIdx); + + /// Get the pointer to the ValueMapping of the operands of a copy + /// instruction from the \p SrcBankID register bank to the \p DstBankID + /// register bank with a size of \p Size. + static const RegisterBankInfo::ValueMapping * + getCopyMapping(unsigned DstBankID, unsigned SrcBankID, unsigned Size); + #define GET_TARGET_REGBANK_CLASS #include "PPCGenRegisterBank.inc" }; @@ -33,6 +59,14 @@ class PPCRegisterBankInfo final : public PPCGenRegisterBankInfo { public: PPCRegisterBankInfo(const TargetRegisterInfo &TRI); + + const RegisterBank &getRegBankFromRegClass(const TargetRegisterClass &RC, + LLT Ty) const override; + const InstructionMapping & + getInstrMapping(const MachineInstr &MI) const override; + + InstructionMappings + getInstrAlternativeMappings(const MachineInstr &MI) const override; }; } // namespace llvm Index: llvm/lib/Target/PowerPC/GISel/PPCRegisterBankInfo.cpp =================================================================== --- llvm/lib/Target/PowerPC/GISel/PPCRegisterBankInfo.cpp +++ llvm/lib/Target/PowerPC/GISel/PPCRegisterBankInfo.cpp @@ -21,6 +21,89 @@ #define GET_TARGET_REGBANK_IMPL #include "PPCGenRegisterBank.inc" +// This file will be TableGen'ed at some point. +#include "PPCGenRegisterBankInfo.def" + using namespace llvm; PPCRegisterBankInfo::PPCRegisterBankInfo(const TargetRegisterInfo &TRI) {} + +const RegisterBank & +PPCRegisterBankInfo::getRegBankFromRegClass(const TargetRegisterClass &RC, + LLT Ty) const { + switch (RC.getID()) { + case PPC::G8RCRegClassID: + case PPC::G8RC_and_G8RC_NOX0RegClassID: + return getRegBank(PPC::GPRRegBankID); + default: + llvm_unreachable("Unexpected register class"); + } +} + +const RegisterBankInfo::InstructionMapping & +PPCRegisterBankInfo::getInstrMapping(const MachineInstr &MI) const { + const unsigned Opc = MI.getOpcode(); + + // Try the default logic for non-generic instructions that are either copies + // or already have some operands assigned to banks. + if ((Opc != TargetOpcode::COPY && !isPreISelGenericOpcode(Opc)) || + Opc == TargetOpcode::G_PHI) { + const RegisterBankInfo::InstructionMapping &Mapping = + getInstrMappingImpl(MI); + if (Mapping.isValid()) + return Mapping; + } + + const MachineFunction &MF = *MI.getParent()->getParent(); + const MachineRegisterInfo &MRI = MF.getRegInfo(); + const TargetSubtargetInfo &STI = MF.getSubtarget(); + const TargetRegisterInfo &TRI = *STI.getRegisterInfo(); + + switch (Opc) { + // Bitwise ops. + case TargetOpcode::G_AND: + case TargetOpcode::G_OR: + case TargetOpcode::G_XOR: { + unsigned NumOperands = MI.getNumOperands(); + assert(NumOperands <= 3 && + "This code is for instructions with 3 or less operands"); + return getInstructionMapping(DefaultMappingID, 1, + getValueMapping(PMI_GPR64), NumOperands); + } + case TargetOpcode::COPY: { + Register DstReg = MI.getOperand(0).getReg(); + Register SrcReg = MI.getOperand(1).getReg(); + // Check if one of the register is not a generic register. + if ((Register::isPhysicalRegister(DstReg) || + !MRI.getType(DstReg).isValid()) || + (Register::isPhysicalRegister(SrcReg) || + !MRI.getType(SrcReg).isValid())) { + const RegisterBank *DstRB = getRegBank(DstReg, MRI, TRI); + const RegisterBank *SrcRB = getRegBank(SrcReg, MRI, TRI); + if (!DstRB) + DstRB = SrcRB; + else if (!SrcRB) + SrcRB = DstRB; + // If both RB are null that means both registers are generic. + // We shouldn't be here. + assert(DstRB && SrcRB && "Both RegBank were nullptr"); + unsigned Size = getSizeInBits(DstReg, MRI, TRI); + return getInstructionMapping( + DefaultMappingID, copyCost(*DstRB, *SrcRB, Size), + getCopyMapping(DstRB->getID(), SrcRB->getID(), Size), + // We only care about the mapping of the destination. + /*NumOperands*/ 1); + } + // Both registers are generic, use G_BITCAST. + // LLVM_FALLTHROUGH; + llvm_unreachable("Instruction mapping for G_BITCAST not implemented"); + } + } + llvm_unreachable("The target must implement this"); +} + +RegisterBankInfo::InstructionMappings +PPCRegisterBankInfo::getInstrAlternativeMappings(const MachineInstr &MI) const { + // TODO Implement. + return RegisterBankInfo::getInstrAlternativeMappings(MI); +} Index: llvm/test/CodeGen/PowerPC/GlobalISel/ppc-isel-logical.ll =================================================================== --- /dev/null +++ llvm/test/CodeGen/PowerPC/GlobalISel/ppc-isel-logical.ll @@ -0,0 +1,33 @@ +; RUN: llc -mtriple ppc64le-linux -global-isel -o - < %s | FileCheck %s -check-prefixes=CHECK,LINUX + +; CHECK-LABEL: test_and: +; LINUX: and 3, 3, 4 +; LINUX: blr +define i64 @test_and(i64 %a, i64 %b) { + %res = and i64 %a, %b + ret i64 %res +} + +define i64 @test_nand(i64 %a, i64 %b) { +; LINUX: nand 3, 3, 4 +; LINUX: blr + %and = and i64 %a, %b + %neg = xor i64 %and, -1 + ret i64 %neg +} + +; CHECK-LABEL: test_or: +; LINUX: or 3, 3, 4 +; LINUX: blr +define i64 @test_or(i64 %a, i64 %b) { + %res = or i64 %a, %b + ret i64 %res +} + +; CHECK-LABEL: test_xor: +; LINUX: xor 3, 3, 4 +; LINUX: blr +define i64 @test_xor(i64 %a, i64 %b) { + %res = xor i64 %a, %b + ret i64 %res +}