//===-- HexagonExpandPredSpillCode.cpp - Expand Predicate Spill Code ------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // The Hexagon processor has no instructions that load or store predicate // registers directly. So, when these registers must be spilled a general // purpose register must be found and the value copied to/from it from/to // the predicate register. This code currently does not use the register // scavenger mechanism available in the allocator. There are two registers // reserved to allow spilling/restoring predicate registers. One is used to // hold the predicate value. The other is used when stack frame offsets are // too large. // //===----------------------------------------------------------------------===// #include "Hexagon.h" #include "HexagonMachineFunctionInfo.h" #include "HexagonSubtarget.h" #include "llvm/ADT/Statistic.h" #include "llvm/CodeGen/LatencyPriorityQueue.h" #include "llvm/CodeGen/MachineDominators.h" #include "llvm/CodeGen/MachineFunctionPass.h" #include "llvm/CodeGen/MachineInstrBuilder.h" #include "llvm/CodeGen/MachineLoopInfo.h" #include "llvm/CodeGen/MachineRegisterInfo.h" #include "llvm/CodeGen/Passes.h" #include "llvm/CodeGen/ScheduleHazardRecognizer.h" #include "llvm/CodeGen/SchedulerRegistry.h" #include "llvm/Support/Compiler.h" #include "llvm/Support/Debug.h" #include "llvm/Support/MathExtras.h" #include "llvm/Target/TargetInstrInfo.h" #include "llvm/Target/TargetMachine.h" #include "llvm/Target/TargetRegisterInfo.h" using namespace llvm; namespace llvm { void initializeHexagonExpandPredSpillCodePass(PassRegistry&); } namespace { class HexagonExpandPredSpillCode : public MachineFunctionPass { public: static char ID; HexagonExpandPredSpillCode() : MachineFunctionPass(ID) { PassRegistry &Registry = *PassRegistry::getPassRegistry(); initializeHexagonExpandPredSpillCodePass(Registry); } const char *getPassName() const override { return "Hexagon Expand Predicate Spill Code"; } bool runOnMachineFunction(MachineFunction &Fn) override; }; char HexagonExpandPredSpillCode::ID = 0; bool HexagonExpandPredSpillCode::runOnMachineFunction(MachineFunction &Fn) { const HexagonSubtarget &QST = Fn.getSubtarget(); const HexagonInstrInfo *TII = QST.getInstrInfo(); // Loop over all of the basic blocks. for (MachineFunction::iterator MBBb = Fn.begin(), MBBe = Fn.end(); MBBb != MBBe; ++MBBb) { MachineBasicBlock* MBB = MBBb; // Traverse the basic block. for (MachineBasicBlock::iterator MII = MBB->begin(); MII != MBB->end(); ++MII) { MachineInstr *MI = MII; int Opc = MI->getOpcode(); if (Opc == Hexagon::S2_storerb_pci_pseudo || Opc == Hexagon::S2_storerh_pci_pseudo || Opc == Hexagon::S2_storeri_pci_pseudo || Opc == Hexagon::S2_storerd_pci_pseudo || Opc == Hexagon::S2_storerf_pci_pseudo) { unsigned Opcode; if (Opc == Hexagon::S2_storerd_pci_pseudo) Opcode = Hexagon::S2_storerd_pci; else if (Opc == Hexagon::S2_storeri_pci_pseudo) Opcode = Hexagon::S2_storeri_pci; else if (Opc == Hexagon::S2_storerh_pci_pseudo) Opcode = Hexagon::S2_storerh_pci; else if (Opc == Hexagon::S2_storerf_pci_pseudo) Opcode = Hexagon::S2_storerf_pci; else if (Opc == Hexagon::S2_storerb_pci_pseudo) Opcode = Hexagon::S2_storerb_pci; else llvm_unreachable("wrong Opc"); MachineOperand &Op0 = MI->getOperand(0); MachineOperand &Op1 = MI->getOperand(1); MachineOperand &Op2 = MI->getOperand(2); MachineOperand &Op3 = MI->getOperand(3); // Modifier value. MachineOperand &Op4 = MI->getOperand(4); // Emit a "C6 = Rn, C6 is the control register for M0". BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::A2_tfrrcr), Hexagon::C6)->addOperand(Op3); // Replace the pseude circ_ldd by the real circ_ldd. MachineInstr *NewMI = BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Opcode)); NewMI->addOperand(Op0); NewMI->addOperand(Op1); NewMI->addOperand(Op4); NewMI->addOperand(MachineOperand::CreateReg(Hexagon::M0, false, /*isDef*/ false, /*isImpl*/ true /*isKill*/)); NewMI->addOperand(Op2); MII = MBB->erase(MI); --MII; } else if (Opc == Hexagon::L2_loadrd_pci_pseudo || Opc == Hexagon::L2_loadri_pci_pseudo || Opc == Hexagon::L2_loadrh_pci_pseudo || Opc == Hexagon::L2_loadruh_pci_pseudo|| Opc == Hexagon::L2_loadrb_pci_pseudo || Opc == Hexagon::L2_loadrub_pci_pseudo) { unsigned Opcode; if (Opc == Hexagon::L2_loadrd_pci_pseudo) Opcode = Hexagon::L2_loadrd_pci; else if (Opc == Hexagon::L2_loadri_pci_pseudo) Opcode = Hexagon::L2_loadri_pci; else if (Opc == Hexagon::L2_loadrh_pci_pseudo) Opcode = Hexagon::L2_loadrh_pci; else if (Opc == Hexagon::L2_loadruh_pci_pseudo) Opcode = Hexagon::L2_loadruh_pci; else if (Opc == Hexagon::L2_loadrb_pci_pseudo) Opcode = Hexagon::L2_loadrb_pci; else if (Opc == Hexagon::L2_loadrub_pci_pseudo) Opcode = Hexagon::L2_loadrub_pci; else llvm_unreachable("wrong Opc"); MachineOperand &Op0 = MI->getOperand(0); MachineOperand &Op1 = MI->getOperand(1); MachineOperand &Op2 = MI->getOperand(2); MachineOperand &Op4 = MI->getOperand(4); // Modifier value. MachineOperand &Op5 = MI->getOperand(5); // Emit a "C6 = Rn, C6 is the control register for M0". BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::A2_tfrrcr), Hexagon::C6)->addOperand(Op4); // Replace the pseude circ_ldd by the real circ_ldd. MachineInstr *NewMI = BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Opcode)); NewMI->addOperand(Op1); NewMI->addOperand(Op0); NewMI->addOperand(Op2); NewMI->addOperand(Op5); NewMI->addOperand(MachineOperand::CreateReg(Hexagon::M0, false, /*isDef*/ false, /*isImpl*/ true /*isKill*/)); MII = MBB->erase(MI); --MII; } else if (Opc == Hexagon::L2_loadrd_pbr_pseudo || Opc == Hexagon::L2_loadri_pbr_pseudo || Opc == Hexagon::L2_loadrh_pbr_pseudo || Opc == Hexagon::L2_loadruh_pbr_pseudo|| Opc == Hexagon::L2_loadrb_pbr_pseudo || Opc == Hexagon::L2_loadrub_pbr_pseudo) { unsigned Opcode; if (Opc == Hexagon::L2_loadrd_pbr_pseudo) Opcode = Hexagon::L2_loadrd_pbr; else if (Opc == Hexagon::L2_loadri_pbr_pseudo) Opcode = Hexagon::L2_loadri_pbr; else if (Opc == Hexagon::L2_loadrh_pbr_pseudo) Opcode = Hexagon::L2_loadrh_pbr; else if (Opc == Hexagon::L2_loadruh_pbr_pseudo) Opcode = Hexagon::L2_loadruh_pbr; else if (Opc == Hexagon::L2_loadrb_pbr_pseudo) Opcode = Hexagon::L2_loadrb_pbr; else if (Opc == Hexagon::L2_loadrub_pbr_pseudo) Opcode = Hexagon::L2_loadrub_pbr; else llvm_unreachable("wrong Opc"); MachineOperand &Op0 = MI->getOperand(0); MachineOperand &Op1 = MI->getOperand(1); MachineOperand &Op2 = MI->getOperand(2); MachineOperand &Op4 = MI->getOperand(4); // Modifier value. // Emit a "C6 = Rn, C6 is the control register for M0". BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::A2_tfrrcr), Hexagon::C6)->addOperand(Op4); // Replace the pseudo brev_ldd by the real brev_ldd. MachineInstr *NewMI = BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Opcode)); NewMI->addOperand(Op1); NewMI->addOperand(Op0); NewMI->addOperand(Op2); NewMI->addOperand(MachineOperand::CreateReg(Hexagon::M0, false, /*isDef*/ false, /*isImpl*/ true /*isKill*/)); MII = MBB->erase(MI); --MII; } else if (Opc == Hexagon::S2_storerd_pbr_pseudo || Opc == Hexagon::S2_storeri_pbr_pseudo || Opc == Hexagon::S2_storerh_pbr_pseudo || Opc == Hexagon::S2_storerb_pbr_pseudo || Opc == Hexagon::S2_storerf_pbr_pseudo) { unsigned Opcode; if (Opc == Hexagon::S2_storerd_pbr_pseudo) Opcode = Hexagon::S2_storerd_pbr; else if (Opc == Hexagon::S2_storeri_pbr_pseudo) Opcode = Hexagon::S2_storeri_pbr; else if (Opc == Hexagon::S2_storerh_pbr_pseudo) Opcode = Hexagon::S2_storerh_pbr; else if (Opc == Hexagon::S2_storerf_pbr_pseudo) Opcode = Hexagon::S2_storerf_pbr; else if (Opc == Hexagon::S2_storerb_pbr_pseudo) Opcode = Hexagon::S2_storerb_pbr; else llvm_unreachable("wrong Opc"); MachineOperand &Op0 = MI->getOperand(0); MachineOperand &Op1 = MI->getOperand(1); MachineOperand &Op2 = MI->getOperand(2); MachineOperand &Op3 = MI->getOperand(3); // Modifier value. // Emit a "C6 = Rn, C6 is the control register for M0". BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::A2_tfrrcr), Hexagon::C6)->addOperand(Op3); // Replace the pseudo brev_ldd by the real brev_ldd. MachineInstr *NewMI = BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Opcode)); NewMI->addOperand(Op0); NewMI->addOperand(Op1); NewMI->addOperand(MachineOperand::CreateReg(Hexagon::M0, false, /*isDef*/ false, /*isImpl*/ true /*isKill*/)); NewMI->addOperand(Op2); MII = MBB->erase(MI); --MII; } else if (Opc == Hexagon::STriw_pred) { // STriw_pred [R30], ofst, SrcReg; unsigned FP = MI->getOperand(0).getReg(); assert(FP == QST.getRegisterInfo()->getFrameRegister() && "Not a Frame Pointer, Nor a Spill Slot"); assert(MI->getOperand(1).isImm() && "Not an offset"); int Offset = MI->getOperand(1).getImm(); int SrcReg = MI->getOperand(2).getReg(); assert(Hexagon::PredRegsRegClass.contains(SrcReg) && "Not a predicate register"); if (!TII->isValidOffset(Hexagon::S2_storeri_io, Offset)) { if (!TII->isValidOffset(Hexagon::A2_addi, Offset)) { BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::CONST32_Int_Real), HEXAGON_RESERVED_REG_1).addImm(Offset); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::A2_add), HEXAGON_RESERVED_REG_1) .addReg(FP).addReg(HEXAGON_RESERVED_REG_1); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::C2_tfrpr), HEXAGON_RESERVED_REG_2).addReg(SrcReg); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::S2_storeri_io)) .addReg(HEXAGON_RESERVED_REG_1) .addImm(0).addReg(HEXAGON_RESERVED_REG_2); } else { BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::A2_addi), HEXAGON_RESERVED_REG_1).addReg(FP).addImm(Offset); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::C2_tfrpr), HEXAGON_RESERVED_REG_2).addReg(SrcReg); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::S2_storeri_io)) .addReg(HEXAGON_RESERVED_REG_1) .addImm(0) .addReg(HEXAGON_RESERVED_REG_2); } } else { BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::C2_tfrpr), HEXAGON_RESERVED_REG_2).addReg(SrcReg); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::S2_storeri_io)). addReg(FP).addImm(Offset).addReg(HEXAGON_RESERVED_REG_2); } MII = MBB->erase(MI); --MII; } else if (Opc == Hexagon::LDriw_pred) { // DstReg = LDriw_pred [R30], ofst. int DstReg = MI->getOperand(0).getReg(); assert(Hexagon::PredRegsRegClass.contains(DstReg) && "Not a predicate register"); unsigned FP = MI->getOperand(1).getReg(); assert(FP == QST.getRegisterInfo()->getFrameRegister() && "Not a Frame Pointer, Nor a Spill Slot"); assert(MI->getOperand(2).isImm() && "Not an offset"); int Offset = MI->getOperand(2).getImm(); if (!TII->isValidOffset(Hexagon::L2_loadri_io, Offset)) { if (!TII->isValidOffset(Hexagon::A2_addi, Offset)) { BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::CONST32_Int_Real), HEXAGON_RESERVED_REG_1).addImm(Offset); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::A2_add), HEXAGON_RESERVED_REG_1) .addReg(FP) .addReg(HEXAGON_RESERVED_REG_1); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::L2_loadri_io), HEXAGON_RESERVED_REG_2) .addReg(HEXAGON_RESERVED_REG_1) .addImm(0); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::C2_tfrrp), DstReg).addReg(HEXAGON_RESERVED_REG_2); } else { BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::A2_addi), HEXAGON_RESERVED_REG_1).addReg(FP).addImm(Offset); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::L2_loadri_io), HEXAGON_RESERVED_REG_2) .addReg(HEXAGON_RESERVED_REG_1) .addImm(0); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::C2_tfrrp), DstReg).addReg(HEXAGON_RESERVED_REG_2); } } else { BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::L2_loadri_io), HEXAGON_RESERVED_REG_2).addReg(FP).addImm(Offset); BuildMI(*MBB, MII, MI->getDebugLoc(), TII->get(Hexagon::C2_tfrrp), DstReg).addReg(HEXAGON_RESERVED_REG_2); } MII = MBB->erase(MI); --MII; } } } return true; } } //===----------------------------------------------------------------------===// // Public Constructor Functions //===----------------------------------------------------------------------===// static void initializePassOnce(PassRegistry &Registry) { const char *Name = "Hexagon Expand Predicate Spill Code"; PassInfo *PI = new PassInfo(Name, "hexagon-spill-pred", &HexagonExpandPredSpillCode::ID, nullptr, false, false); Registry.registerPass(*PI, true); } void llvm::initializeHexagonExpandPredSpillCodePass(PassRegistry &Registry) { CALL_ONCE_INITIALIZATION(initializePassOnce) } FunctionPass* llvm::createHexagonExpandPredSpillCode() { return new HexagonExpandPredSpillCode(); }