1 //===-- LVLGen.cpp - LVL instruction generator ----------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
10 #include "VESubtarget.h"
11 #include "llvm/CodeGen/MachineFunctionPass.h"
12 #include "llvm/CodeGen/MachineInstrBuilder.h"
13 #include "llvm/CodeGen/MachineRegisterInfo.h"
14 #include "llvm/CodeGen/TargetInstrInfo.h"
15 #include "llvm/Target/TargetMachine.h"
19 #define DEBUG_TYPE "lvl-gen"
22 struct LVLGen
: public MachineFunctionPass
{
23 const TargetInstrInfo
*TII
;
24 const TargetRegisterInfo
*TRI
;
27 LVLGen() : MachineFunctionPass(ID
) {}
28 bool runOnMachineBasicBlock(MachineBasicBlock
&MBB
);
29 bool runOnMachineFunction(MachineFunction
&F
) override
;
31 unsigned getVL(const MachineInstr
&MI
);
32 int getVLIndex(unsigned Opcode
);
36 } // end of anonymous namespace
38 FunctionPass
*llvm::createLVLGenPass() { return new LVLGen
; }
40 int LVLGen::getVLIndex(unsigned Opcode
) {
41 const MCInstrDesc
&MCID
= TII
->get(Opcode
);
43 // If an instruction has VLIndex information, return it.
44 if (HAS_VLINDEX(MCID
.TSFlags
))
45 return GET_VLINDEX(MCID
.TSFlags
);
50 // returns a register holding a vector length. NoRegister is returned when
51 // this MI does not have a vector length.
52 unsigned LVLGen::getVL(const MachineInstr
&MI
) {
53 int Index
= getVLIndex(MI
.getOpcode());
55 return MI
.getOperand(Index
).getReg();
57 return VE::NoRegister
;
60 bool LVLGen::runOnMachineBasicBlock(MachineBasicBlock
&MBB
) {
62 (MBB.getParent()->getSubtarget<VESubtarget>().getRegisterInfo()->getName(no))
65 bool HasRegForVL
= false;
68 for (MachineBasicBlock::iterator I
= MBB
.begin(); I
!= MBB
.end();) {
69 MachineBasicBlock::iterator MI
= I
;
71 // Check whether MI uses a vector length operand. If so, we prepare for VL
72 // register. We would like to reuse VL register as much as possible. We
73 // also would like to keep the number of LEA instructions as fewer as
74 // possible. Therefore, we use a regular scalar register to hold immediate
75 // values to load VL register. And try to reuse identical scalar registers
76 // to avoid new LVLr instructions as much as possible.
77 unsigned Reg
= getVL(*MI
);
78 if (Reg
!= VE::NoRegister
) {
79 LLVM_DEBUG(dbgs() << "Vector instruction found: ");
80 LLVM_DEBUG(MI
->dump());
81 LLVM_DEBUG(dbgs() << "Vector length is " << RegName(Reg
) << ". ");
82 LLVM_DEBUG(dbgs() << "Current VL is "
83 << (HasRegForVL
? RegName(RegForVL
) : "unknown")
86 if (!HasRegForVL
|| RegForVL
!= Reg
) {
87 // Use VL, but a different value in a different scalar register.
88 // So, generate new LVL instruction just before the current instruction.
89 LLVM_DEBUG(dbgs() << "Generate a LVL instruction to load "
90 << RegName(Reg
) << ".\n");
91 BuildMI(MBB
, I
, MI
->getDebugLoc(), TII
->get(VE::LVLr
)).addReg(Reg
);
96 LLVM_DEBUG(dbgs() << "Reuse current VL.\n");
99 // Check the update of a given scalar register holding an immediate value
100 // for VL register. Also, a call doesn't preserve VL register.
102 if (MI
->definesRegister(RegForVL
, TRI
) ||
103 MI
->modifiesRegister(RegForVL
, TRI
) ||
104 MI
->killsRegister(RegForVL
, TRI
) || MI
->isCall()) {
105 // The latest VL is needed to be updated, so disable HasRegForVL.
106 LLVM_DEBUG(dbgs() << RegName(RegForVL
) << " is needed to be updated: ");
107 LLVM_DEBUG(MI
->dump());
117 bool LVLGen::runOnMachineFunction(MachineFunction
&F
) {
118 LLVM_DEBUG(dbgs() << "********** Begin LVLGen **********\n");
119 LLVM_DEBUG(dbgs() << "********** Function: " << F
.getName() << '\n');
120 LLVM_DEBUG(F
.dump());
122 bool Changed
= false;
124 const VESubtarget
&Subtarget
= F
.getSubtarget
<VESubtarget
>();
125 TII
= Subtarget
.getInstrInfo();
126 TRI
= Subtarget
.getRegisterInfo();
128 for (MachineBasicBlock
&MBB
: F
)
129 Changed
|= runOnMachineBasicBlock(MBB
);
132 LLVM_DEBUG(dbgs() << "\n");
133 LLVM_DEBUG(F
.dump());
135 LLVM_DEBUG(dbgs() << "********** End LVLGen **********\n");