2014-07-22 00:55:33 +08:00
|
|
|
//===-- SIShrinkInstructions.cpp - Shrink Instructions --------------------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2014-07-22 00:55:33 +08:00
|
|
|
//
|
|
|
|
/// The pass tries to use the 32-bit encoding for instructions when possible.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "AMDGPU.h"
|
2014-08-05 05:25:23 +08:00
|
|
|
#include "AMDGPUSubtarget.h"
|
2014-07-22 00:55:33 +08:00
|
|
|
#include "SIInstrInfo.h"
|
AMDGPU: Remove #include "MCTargetDesc/AMDGPUMCTargetDesc.h" from common headers
Summary:
MCTargetDesc/AMDGPUMCTargetDesc.h contains enums for all the instuction
and register defintions, which are huge so we only want to include
them where needed.
This will also make it easier if we want to split the R600 and GCN
definitions into separate tablegenerated files.
I was unable to remove AMDGPUMCTargetDesc.h from SIMachineFunctionInfo.h
because it uses some enums from the header to initialize default values
for the SIMachineFunction class, so I ended up having to remove includes of
SIMachineFunctionInfo.h from headers too.
Reviewers: arsenm, nhaehnle
Reviewed By: nhaehnle
Subscribers: MatzeB, kzhuravl, wdng, yaxunl, dstuttard, tpr, t-tye, javed.absar, llvm-commits
Differential Revision: https://reviews.llvm.org/D46272
llvm-svn: 332930
2018-05-22 10:03:23 +08:00
|
|
|
#include "MCTargetDesc/AMDGPUMCTargetDesc.h"
|
2014-07-22 00:55:33 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
|
|
|
#include "llvm/CodeGen/MachineInstrBuilder.h"
|
|
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
2014-08-01 08:32:33 +08:00
|
|
|
#include "llvm/IR/Constants.h"
|
2014-07-22 00:55:33 +08:00
|
|
|
#include "llvm/IR/Function.h"
|
2015-03-24 02:07:13 +08:00
|
|
|
#include "llvm/IR/LLVMContext.h"
|
2014-07-22 00:55:33 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
2015-03-24 02:07:13 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2014-07-22 00:55:33 +08:00
|
|
|
#include "llvm/Target/TargetMachine.h"
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "si-shrink-instructions"
|
|
|
|
|
|
|
|
STATISTIC(NumInstructionsShrunk,
|
|
|
|
"Number of 64-bit instruction reduced to 32-bit.");
|
2014-08-01 08:32:33 +08:00
|
|
|
STATISTIC(NumLiteralConstantsFolded,
|
|
|
|
"Number of literal constants folded into 32-bit instructions.");
|
2014-07-22 00:55:33 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class SIShrinkInstructions : public MachineFunctionPass {
|
|
|
|
public:
|
|
|
|
static char ID;
|
|
|
|
|
|
|
|
public:
|
|
|
|
SIShrinkInstructions() : MachineFunctionPass(ID) {
|
|
|
|
}
|
|
|
|
|
2014-08-31 00:48:34 +08:00
|
|
|
bool runOnMachineFunction(MachineFunction &MF) override;
|
2014-07-22 00:55:33 +08:00
|
|
|
|
2016-10-01 10:56:57 +08:00
|
|
|
StringRef getPassName() const override { return "SI Shrink Instructions"; }
|
2014-07-22 00:55:33 +08:00
|
|
|
|
2014-08-31 00:48:34 +08:00
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
2014-07-22 00:55:33 +08:00
|
|
|
AU.setPreservesCFG();
|
|
|
|
MachineFunctionPass::getAnalysisUsage(AU);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // End anonymous namespace.
|
|
|
|
|
2016-06-10 07:18:47 +08:00
|
|
|
INITIALIZE_PASS(SIShrinkInstructions, DEBUG_TYPE,
|
|
|
|
"SI Shrink Instructions", false, false)
|
2014-07-22 00:55:33 +08:00
|
|
|
|
|
|
|
char SIShrinkInstructions::ID = 0;
|
|
|
|
|
|
|
|
FunctionPass *llvm::createSIShrinkInstructionsPass() {
|
|
|
|
return new SIShrinkInstructions();
|
|
|
|
}
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// This function checks \p MI for operands defined by a move immediate
|
2014-08-01 08:32:33 +08:00
|
|
|
/// instruction and then folds the literal constant into the instruction if it
|
2017-07-11 03:53:57 +08:00
|
|
|
/// can. This function assumes that \p MI is a VOP1, VOP2, or VOPC instructions.
|
|
|
|
static bool foldImmediates(MachineInstr &MI, const SIInstrInfo *TII,
|
2014-08-01 08:32:33 +08:00
|
|
|
MachineRegisterInfo &MRI, bool TryToCommute = true) {
|
2015-10-20 12:35:43 +08:00
|
|
|
assert(TII->isVOP1(MI) || TII->isVOP2(MI) || TII->isVOPC(MI));
|
2014-08-01 08:32:33 +08:00
|
|
|
|
2015-02-14 03:05:03 +08:00
|
|
|
int Src0Idx = AMDGPU::getNamedOperandIdx(MI.getOpcode(), AMDGPU::OpName::src0);
|
2014-08-01 08:32:33 +08:00
|
|
|
|
|
|
|
// Try to fold Src0
|
2016-12-10 08:39:12 +08:00
|
|
|
MachineOperand &Src0 = MI.getOperand(Src0Idx);
|
2017-07-11 03:53:57 +08:00
|
|
|
if (Src0.isReg()) {
|
2015-02-14 03:05:03 +08:00
|
|
|
unsigned Reg = Src0.getReg();
|
2017-07-11 03:53:57 +08:00
|
|
|
if (TargetRegisterInfo::isVirtualRegister(Reg) && MRI.hasOneUse(Reg)) {
|
|
|
|
MachineInstr *Def = MRI.getUniqueVRegDef(Reg);
|
|
|
|
if (Def && Def->isMoveImmediate()) {
|
|
|
|
MachineOperand &MovSrc = Def->getOperand(1);
|
|
|
|
bool ConstantFolded = false;
|
|
|
|
|
|
|
|
if (MovSrc.isImm() && (isInt<32>(MovSrc.getImm()) ||
|
|
|
|
isUInt<32>(MovSrc.getImm()))) {
|
|
|
|
// It's possible to have only one component of a super-reg defined by
|
|
|
|
// a single mov, so we need to clear any subregister flag.
|
|
|
|
Src0.setSubReg(0);
|
|
|
|
Src0.ChangeToImmediate(MovSrc.getImm());
|
|
|
|
ConstantFolded = true;
|
2017-07-11 04:04:35 +08:00
|
|
|
} else if (MovSrc.isFI()) {
|
|
|
|
Src0.setSubReg(0);
|
|
|
|
Src0.ChangeToFrameIndex(MovSrc.getIndex());
|
|
|
|
ConstantFolded = true;
|
2017-07-11 03:53:57 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ConstantFolded) {
|
|
|
|
assert(MRI.use_empty(Reg));
|
2014-08-01 08:32:33 +08:00
|
|
|
Def->eraseFromParent();
|
2017-07-11 03:53:57 +08:00
|
|
|
++NumLiteralConstantsFolded;
|
|
|
|
return true;
|
|
|
|
}
|
2014-08-01 08:32:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have failed to fold src0, so commute the instruction and try again.
|
2017-07-11 03:53:57 +08:00
|
|
|
if (TryToCommute && MI.isCommutable()) {
|
|
|
|
if (TII->commuteInstruction(MI)) {
|
|
|
|
if (foldImmediates(MI, TII, MRI, false))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Commute back.
|
|
|
|
TII->commuteInstruction(MI);
|
|
|
|
}
|
|
|
|
}
|
2014-08-01 08:32:33 +08:00
|
|
|
|
2017-07-11 03:53:57 +08:00
|
|
|
return false;
|
2014-08-01 08:32:33 +08:00
|
|
|
}
|
|
|
|
|
2016-04-16 09:46:49 +08:00
|
|
|
static bool isKImmOperand(const SIInstrInfo *TII, const MachineOperand &Src) {
|
2016-12-10 08:39:12 +08:00
|
|
|
return isInt<16>(Src.getImm()) &&
|
|
|
|
!TII->isInlineConstant(*Src.getParent(),
|
|
|
|
Src.getParent()->getOperandNo(&Src));
|
2016-04-16 09:46:49 +08:00
|
|
|
}
|
|
|
|
|
2016-09-17 05:41:16 +08:00
|
|
|
static bool isKUImmOperand(const SIInstrInfo *TII, const MachineOperand &Src) {
|
2016-12-10 08:39:12 +08:00
|
|
|
return isUInt<16>(Src.getImm()) &&
|
|
|
|
!TII->isInlineConstant(*Src.getParent(),
|
|
|
|
Src.getParent()->getOperandNo(&Src));
|
2016-09-17 05:41:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool isKImmOrKUImmOperand(const SIInstrInfo *TII,
|
|
|
|
const MachineOperand &Src,
|
|
|
|
bool &IsUnsigned) {
|
|
|
|
if (isInt<16>(Src.getImm())) {
|
|
|
|
IsUnsigned = false;
|
2016-12-10 08:39:12 +08:00
|
|
|
return !TII->isInlineConstant(Src);
|
2016-09-17 05:41:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (isUInt<16>(Src.getImm())) {
|
|
|
|
IsUnsigned = true;
|
2016-12-10 08:39:12 +08:00
|
|
|
return !TII->isInlineConstant(Src);
|
2016-09-17 05:41:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-11-02 07:14:20 +08:00
|
|
|
/// \returns true if the constant in \p Src should be replaced with a bitreverse
|
|
|
|
/// of an inline immediate.
|
|
|
|
static bool isReverseInlineImm(const SIInstrInfo *TII,
|
|
|
|
const MachineOperand &Src,
|
|
|
|
int32_t &ReverseImm) {
|
2016-12-10 08:39:12 +08:00
|
|
|
if (!isInt<32>(Src.getImm()) || TII->isInlineConstant(Src))
|
2016-11-02 07:14:20 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
ReverseImm = reverseBits<int32_t>(static_cast<int32_t>(Src.getImm()));
|
|
|
|
return ReverseImm >= -16 && ReverseImm <= 64;
|
|
|
|
}
|
|
|
|
|
2016-09-04 01:25:39 +08:00
|
|
|
/// Copy implicit register operands from specified instruction to this
|
|
|
|
/// instruction that are not part of the instruction definition.
|
|
|
|
static void copyExtraImplicitOps(MachineInstr &NewMI, MachineFunction &MF,
|
|
|
|
const MachineInstr &MI) {
|
|
|
|
for (unsigned i = MI.getDesc().getNumOperands() +
|
|
|
|
MI.getDesc().getNumImplicitUses() +
|
|
|
|
MI.getDesc().getNumImplicitDefs(), e = MI.getNumOperands();
|
|
|
|
i != e; ++i) {
|
|
|
|
const MachineOperand &MO = MI.getOperand(i);
|
|
|
|
if ((MO.isReg() && MO.isImplicit()) || MO.isRegMask())
|
|
|
|
NewMI.addOperand(MF, MO);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-17 05:41:16 +08:00
|
|
|
static void shrinkScalarCompare(const SIInstrInfo *TII, MachineInstr &MI) {
|
|
|
|
// cmpk instructions do scc = dst <cc op> imm16, so commute the instruction to
|
|
|
|
// get constants on the RHS.
|
|
|
|
if (!MI.getOperand(0).isReg())
|
|
|
|
TII->commuteInstruction(MI, false, 0, 1);
|
|
|
|
|
|
|
|
const MachineOperand &Src1 = MI.getOperand(1);
|
|
|
|
if (!Src1.isImm())
|
|
|
|
return;
|
|
|
|
|
|
|
|
int SOPKOpc = AMDGPU::getSOPKOp(MI.getOpcode());
|
|
|
|
if (SOPKOpc == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// eq/ne is special because the imm16 can be treated as signed or unsigned,
|
2016-09-30 09:50:20 +08:00
|
|
|
// and initially selectd to the unsigned versions.
|
|
|
|
if (SOPKOpc == AMDGPU::S_CMPK_EQ_U32 || SOPKOpc == AMDGPU::S_CMPK_LG_U32) {
|
2016-09-17 05:41:16 +08:00
|
|
|
bool HasUImm;
|
|
|
|
if (isKImmOrKUImmOperand(TII, Src1, HasUImm)) {
|
2016-09-30 09:50:20 +08:00
|
|
|
if (!HasUImm) {
|
|
|
|
SOPKOpc = (SOPKOpc == AMDGPU::S_CMPK_EQ_U32) ?
|
|
|
|
AMDGPU::S_CMPK_EQ_I32 : AMDGPU::S_CMPK_LG_I32;
|
2016-09-17 05:41:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
MI.setDesc(TII->get(SOPKOpc));
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const MCInstrDesc &NewDesc = TII->get(SOPKOpc);
|
|
|
|
|
|
|
|
if ((TII->sopkIsZext(SOPKOpc) && isKUImmOperand(TII, Src1)) ||
|
|
|
|
(!TII->sopkIsZext(SOPKOpc) && isKImmOperand(TII, Src1))) {
|
|
|
|
MI.setDesc(NewDesc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[AMDGPU] Shrink scalar AND, OR, XOR instructions
This change attempts to shrink scalar AND, OR and XOR instructions which take an immediate that isn't inlineable.
It performs:
AND s0, s0, ~(1 << n) -> BITSET0 s0, n
OR s0, s0, (1 << n) -> BITSET1 s0, n
AND s0, s1, x -> ANDN2 s0, s1, ~x
OR s0, s1, x -> ORN2 s0, s1, ~x
XOR s0, s1, x -> XNOR s0, s1, ~x
In particular, this catches setting and clearing the sign bit for fabs (and x, 0x7ffffffff -> bitset0 x, 31 and or x, 0x80000000 -> bitset1 x, 31).
llvm-svn: 348601
2018-12-07 23:33:21 +08:00
|
|
|
/// Attempt to shink AND/OR/XOR operations requiring non-inlineable literals.
|
|
|
|
/// For AND or OR, try using S_BITSET{0,1} to clear or set bits.
|
|
|
|
/// If the inverse of the immediate is legal, use ANDN2, ORN2 or
|
|
|
|
/// XNOR (as a ^ b == ~(a ^ ~b)).
|
|
|
|
/// \returns true if the caller should continue the machine function iterator
|
|
|
|
static bool shrinkScalarLogicOp(const GCNSubtarget &ST,
|
|
|
|
MachineRegisterInfo &MRI,
|
|
|
|
const SIInstrInfo *TII,
|
|
|
|
MachineInstr &MI) {
|
|
|
|
unsigned Opc = MI.getOpcode();
|
|
|
|
const MachineOperand *Dest = &MI.getOperand(0);
|
|
|
|
MachineOperand *Src0 = &MI.getOperand(1);
|
|
|
|
MachineOperand *Src1 = &MI.getOperand(2);
|
|
|
|
MachineOperand *SrcReg = Src0;
|
|
|
|
MachineOperand *SrcImm = Src1;
|
|
|
|
|
|
|
|
if (SrcImm->isImm() &&
|
|
|
|
!AMDGPU::isInlinableLiteral32(SrcImm->getImm(), ST.hasInv2PiInlineImm())) {
|
|
|
|
uint32_t Imm = static_cast<uint32_t>(SrcImm->getImm());
|
|
|
|
uint32_t NewImm = 0;
|
|
|
|
|
|
|
|
if (Opc == AMDGPU::S_AND_B32) {
|
|
|
|
if (isPowerOf2_32(~Imm)) {
|
|
|
|
NewImm = countTrailingOnes(Imm);
|
|
|
|
Opc = AMDGPU::S_BITSET0_B32;
|
|
|
|
} else if (AMDGPU::isInlinableLiteral32(~Imm, ST.hasInv2PiInlineImm())) {
|
|
|
|
NewImm = ~Imm;
|
|
|
|
Opc = AMDGPU::S_ANDN2_B32;
|
|
|
|
}
|
|
|
|
} else if (Opc == AMDGPU::S_OR_B32) {
|
|
|
|
if (isPowerOf2_32(Imm)) {
|
|
|
|
NewImm = countTrailingZeros(Imm);
|
|
|
|
Opc = AMDGPU::S_BITSET1_B32;
|
|
|
|
} else if (AMDGPU::isInlinableLiteral32(~Imm, ST.hasInv2PiInlineImm())) {
|
|
|
|
NewImm = ~Imm;
|
|
|
|
Opc = AMDGPU::S_ORN2_B32;
|
|
|
|
}
|
|
|
|
} else if (Opc == AMDGPU::S_XOR_B32) {
|
|
|
|
if (AMDGPU::isInlinableLiteral32(~Imm, ST.hasInv2PiInlineImm())) {
|
|
|
|
NewImm = ~Imm;
|
|
|
|
Opc = AMDGPU::S_XNOR_B32;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
llvm_unreachable("unexpected opcode");
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((Opc == AMDGPU::S_ANDN2_B32 || Opc == AMDGPU::S_ORN2_B32) &&
|
|
|
|
SrcImm == Src0) {
|
|
|
|
if (!TII->commuteInstruction(MI, false, 1, 2))
|
|
|
|
NewImm = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (NewImm != 0) {
|
|
|
|
if (TargetRegisterInfo::isVirtualRegister(Dest->getReg()) &&
|
|
|
|
SrcReg->isReg()) {
|
|
|
|
MRI.setRegAllocationHint(Dest->getReg(), 0, SrcReg->getReg());
|
|
|
|
MRI.setRegAllocationHint(SrcReg->getReg(), 0, Dest->getReg());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SrcReg->isReg() && SrcReg->getReg() == Dest->getReg()) {
|
|
|
|
MI.setDesc(TII->get(Opc));
|
|
|
|
if (Opc == AMDGPU::S_BITSET0_B32 ||
|
|
|
|
Opc == AMDGPU::S_BITSET1_B32) {
|
|
|
|
Src0->ChangeToImmediate(NewImm);
|
|
|
|
MI.RemoveOperand(2);
|
|
|
|
} else {
|
|
|
|
SrcImm->setImm(NewImm);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-10-30 01:26:01 +08:00
|
|
|
// This is the same as MachineInstr::readsRegister/modifiesRegister except
|
|
|
|
// it takes subregs into account.
|
|
|
|
static bool instAccessReg(iterator_range<MachineInstr::const_mop_iterator> &&R,
|
|
|
|
unsigned Reg, unsigned SubReg,
|
|
|
|
const SIRegisterInfo &TRI) {
|
|
|
|
for (const MachineOperand &MO : R) {
|
|
|
|
if (!MO.isReg())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(Reg) &&
|
|
|
|
TargetRegisterInfo::isPhysicalRegister(MO.getReg())) {
|
|
|
|
if (TRI.regsOverlap(Reg, MO.getReg()))
|
|
|
|
return true;
|
|
|
|
} else if (MO.getReg() == Reg &&
|
|
|
|
TargetRegisterInfo::isVirtualRegister(Reg)) {
|
|
|
|
LaneBitmask Overlap = TRI.getSubRegIndexLaneMask(SubReg) &
|
|
|
|
TRI.getSubRegIndexLaneMask(MO.getSubReg());
|
|
|
|
if (Overlap.any())
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool instReadsReg(const MachineInstr *MI,
|
|
|
|
unsigned Reg, unsigned SubReg,
|
|
|
|
const SIRegisterInfo &TRI) {
|
|
|
|
return instAccessReg(MI->uses(), Reg, SubReg, TRI);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool instModifiesReg(const MachineInstr *MI,
|
|
|
|
unsigned Reg, unsigned SubReg,
|
|
|
|
const SIRegisterInfo &TRI) {
|
|
|
|
return instAccessReg(MI->defs(), Reg, SubReg, TRI);
|
|
|
|
}
|
|
|
|
|
|
|
|
static TargetInstrInfo::RegSubRegPair
|
|
|
|
getSubRegForIndex(unsigned Reg, unsigned Sub, unsigned I,
|
|
|
|
const SIRegisterInfo &TRI, const MachineRegisterInfo &MRI) {
|
|
|
|
if (TRI.getRegSizeInBits(Reg, MRI) != 32) {
|
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(Reg)) {
|
|
|
|
Reg = TRI.getSubReg(Reg, TRI.getSubRegFromChannel(I));
|
|
|
|
} else {
|
|
|
|
LaneBitmask LM = TRI.getSubRegIndexLaneMask(Sub);
|
|
|
|
Sub = TRI.getSubRegFromChannel(I + countTrailingZeros(LM.getAsInteger()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return TargetInstrInfo::RegSubRegPair(Reg, Sub);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Match:
|
|
|
|
// mov t, x
|
|
|
|
// mov x, y
|
|
|
|
// mov y, t
|
|
|
|
//
|
|
|
|
// =>
|
|
|
|
//
|
|
|
|
// mov t, x (t is potentially dead and move eliminated)
|
|
|
|
// v_swap_b32 x, y
|
|
|
|
//
|
|
|
|
// Returns next valid instruction pointer if was able to create v_swap_b32.
|
|
|
|
//
|
|
|
|
// This shall not be done too early not to prevent possible folding which may
|
|
|
|
// remove matched moves, and this should prefereably be done before RA to
|
|
|
|
// release saved registers and also possibly after RA which can insert copies
|
|
|
|
// too.
|
|
|
|
//
|
|
|
|
// This is really just a generic peephole that is not a canocical shrinking,
|
|
|
|
// although requirements match the pass placement and it reduces code size too.
|
|
|
|
static MachineInstr* matchSwap(MachineInstr &MovT, MachineRegisterInfo &MRI,
|
|
|
|
const SIInstrInfo *TII) {
|
|
|
|
assert(MovT.getOpcode() == AMDGPU::V_MOV_B32_e32 ||
|
|
|
|
MovT.getOpcode() == AMDGPU::COPY);
|
|
|
|
|
|
|
|
unsigned T = MovT.getOperand(0).getReg();
|
|
|
|
unsigned Tsub = MovT.getOperand(0).getSubReg();
|
|
|
|
MachineOperand &Xop = MovT.getOperand(1);
|
|
|
|
|
|
|
|
if (!Xop.isReg())
|
|
|
|
return nullptr;
|
|
|
|
unsigned X = Xop.getReg();
|
|
|
|
unsigned Xsub = Xop.getSubReg();
|
|
|
|
|
|
|
|
unsigned Size = TII->getOpSize(MovT, 0) / 4;
|
|
|
|
|
|
|
|
const SIRegisterInfo &TRI = TII->getRegisterInfo();
|
|
|
|
if (!TRI.isVGPR(MRI, X))
|
2018-10-30 01:53:23 +08:00
|
|
|
return nullptr;
|
2018-10-30 01:26:01 +08:00
|
|
|
|
|
|
|
for (MachineOperand &YTop : MRI.use_nodbg_operands(T)) {
|
|
|
|
if (YTop.getSubReg() != Tsub)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
MachineInstr &MovY = *YTop.getParent();
|
|
|
|
if ((MovY.getOpcode() != AMDGPU::V_MOV_B32_e32 &&
|
|
|
|
MovY.getOpcode() != AMDGPU::COPY) ||
|
|
|
|
MovY.getOperand(1).getSubReg() != Tsub)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
unsigned Y = MovY.getOperand(0).getReg();
|
|
|
|
unsigned Ysub = MovY.getOperand(0).getSubReg();
|
|
|
|
|
|
|
|
if (!TRI.isVGPR(MRI, Y) || MovT.getParent() != MovY.getParent())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
MachineInstr *MovX = nullptr;
|
|
|
|
auto I = std::next(MovT.getIterator()), E = MovT.getParent()->instr_end();
|
|
|
|
for (auto IY = MovY.getIterator(); I != E && I != IY; ++I) {
|
|
|
|
if (instReadsReg(&*I, X, Xsub, TRI) ||
|
|
|
|
instModifiesReg(&*I, Y, Ysub, TRI) ||
|
|
|
|
instModifiesReg(&*I, T, Tsub, TRI) ||
|
|
|
|
(MovX && instModifiesReg(&*I, X, Xsub, TRI))) {
|
|
|
|
MovX = nullptr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!instReadsReg(&*I, Y, Ysub, TRI)) {
|
|
|
|
if (!MovX && instModifiesReg(&*I, X, Xsub, TRI)) {
|
|
|
|
MovX = nullptr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (MovX ||
|
|
|
|
(I->getOpcode() != AMDGPU::V_MOV_B32_e32 &&
|
|
|
|
I->getOpcode() != AMDGPU::COPY) ||
|
|
|
|
I->getOperand(0).getReg() != X ||
|
|
|
|
I->getOperand(0).getSubReg() != Xsub) {
|
|
|
|
MovX = nullptr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
MovX = &*I;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!MovX || I == E)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
LLVM_DEBUG(dbgs() << "Matched v_swap_b32:\n" << MovT << *MovX << MovY);
|
|
|
|
|
|
|
|
for (unsigned I = 0; I < Size; ++I) {
|
|
|
|
TargetInstrInfo::RegSubRegPair X1, Y1;
|
|
|
|
X1 = getSubRegForIndex(X, Xsub, I, TRI, MRI);
|
|
|
|
Y1 = getSubRegForIndex(Y, Ysub, I, TRI, MRI);
|
|
|
|
BuildMI(*MovT.getParent(), MovX->getIterator(), MovT.getDebugLoc(),
|
|
|
|
TII->get(AMDGPU::V_SWAP_B32))
|
|
|
|
.addDef(X1.Reg, 0, X1.SubReg)
|
|
|
|
.addDef(Y1.Reg, 0, Y1.SubReg)
|
|
|
|
.addReg(Y1.Reg, 0, Y1.SubReg)
|
|
|
|
.addReg(X1.Reg, 0, X1.SubReg).getInstr();
|
|
|
|
}
|
|
|
|
MovX->eraseFromParent();
|
|
|
|
MovY.eraseFromParent();
|
|
|
|
MachineInstr *Next = &*std::next(MovT.getIterator());
|
|
|
|
if (MRI.use_nodbg_empty(T))
|
|
|
|
MovT.eraseFromParent();
|
|
|
|
else
|
|
|
|
Xop.setIsKill(false);
|
|
|
|
|
|
|
|
return Next;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2014-07-22 00:55:33 +08:00
|
|
|
bool SIShrinkInstructions::runOnMachineFunction(MachineFunction &MF) {
|
2017-12-16 06:22:58 +08:00
|
|
|
if (skipFunction(MF.getFunction()))
|
2016-04-26 06:23:44 +08:00
|
|
|
return false;
|
|
|
|
|
2014-07-22 00:55:33 +08:00
|
|
|
MachineRegisterInfo &MRI = MF.getRegInfo();
|
2018-07-12 04:59:01 +08:00
|
|
|
const GCNSubtarget &ST = MF.getSubtarget<GCNSubtarget>();
|
2016-06-24 14:30:11 +08:00
|
|
|
const SIInstrInfo *TII = ST.getInstrInfo();
|
|
|
|
|
2014-07-22 00:55:33 +08:00
|
|
|
std::vector<unsigned> I1Defs;
|
|
|
|
|
|
|
|
for (MachineFunction::iterator BI = MF.begin(), BE = MF.end();
|
|
|
|
BI != BE; ++BI) {
|
|
|
|
|
|
|
|
MachineBasicBlock &MBB = *BI;
|
|
|
|
MachineBasicBlock::iterator I, Next;
|
|
|
|
for (I = MBB.begin(); I != MBB.end(); I = Next) {
|
|
|
|
Next = std::next(I);
|
|
|
|
MachineInstr &MI = *I;
|
|
|
|
|
2016-03-11 15:42:49 +08:00
|
|
|
if (MI.getOpcode() == AMDGPU::V_MOV_B32_e32) {
|
|
|
|
// If this has a literal constant source that is the same as the
|
|
|
|
// reversed bits of an inline immediate, replace with a bitreverse of
|
|
|
|
// that constant. This saves 4 bytes in the common case of materializing
|
|
|
|
// sign bits.
|
|
|
|
|
|
|
|
// Test if we are after regalloc. We only want to do this after any
|
|
|
|
// optimizations happen because this will confuse them.
|
|
|
|
// XXX - not exactly a check for post-regalloc run.
|
|
|
|
MachineOperand &Src = MI.getOperand(1);
|
|
|
|
if (Src.isImm() &&
|
|
|
|
TargetRegisterInfo::isPhysicalRegister(MI.getOperand(0).getReg())) {
|
2016-11-02 07:14:20 +08:00
|
|
|
int32_t ReverseImm;
|
|
|
|
if (isReverseInlineImm(TII, Src, ReverseImm)) {
|
|
|
|
MI.setDesc(TII->get(AMDGPU::V_BFREV_B32_e32));
|
|
|
|
Src.setImm(ReverseImm);
|
|
|
|
continue;
|
2016-03-11 15:42:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-30 01:26:01 +08:00
|
|
|
if (ST.hasSwap() && (MI.getOpcode() == AMDGPU::V_MOV_B32_e32 ||
|
|
|
|
MI.getOpcode() == AMDGPU::COPY)) {
|
|
|
|
if (auto *NextMI = matchSwap(MI, MRI, TII)) {
|
|
|
|
Next = NextMI->getIterator();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-26 03:53:22 +08:00
|
|
|
// Combine adjacent s_nops to use the immediate operand encoding how long
|
|
|
|
// to wait.
|
|
|
|
//
|
|
|
|
// s_nop N
|
|
|
|
// s_nop M
|
|
|
|
// =>
|
|
|
|
// s_nop (N + M)
|
|
|
|
if (MI.getOpcode() == AMDGPU::S_NOP &&
|
|
|
|
Next != MBB.end() &&
|
|
|
|
(*Next).getOpcode() == AMDGPU::S_NOP) {
|
|
|
|
|
|
|
|
MachineInstr &NextMI = *Next;
|
|
|
|
// The instruction encodes the amount to wait with an offset of 1,
|
|
|
|
// i.e. 0 is wait 1 cycle. Convert both to cycles and then convert back
|
|
|
|
// after adding.
|
|
|
|
uint8_t Nop0 = MI.getOperand(0).getImm() + 1;
|
|
|
|
uint8_t Nop1 = NextMI.getOperand(0).getImm() + 1;
|
|
|
|
|
|
|
|
// Make sure we don't overflow the bounds.
|
|
|
|
if (Nop0 + Nop1 <= 8) {
|
|
|
|
NextMI.getOperand(0).setImm(Nop0 + Nop1 - 1);
|
|
|
|
MI.eraseFromParent();
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-04-16 09:46:49 +08:00
|
|
|
// FIXME: We also need to consider movs of constant operands since
|
|
|
|
// immediate operands are not folded if they have more than one use, and
|
|
|
|
// the operand folding pass is unaware if the immediate will be free since
|
|
|
|
// it won't know if the src == dest constraint will end up being
|
|
|
|
// satisfied.
|
|
|
|
if (MI.getOpcode() == AMDGPU::S_ADD_I32 ||
|
|
|
|
MI.getOpcode() == AMDGPU::S_MUL_I32) {
|
2016-09-09 01:35:41 +08:00
|
|
|
const MachineOperand *Dest = &MI.getOperand(0);
|
|
|
|
MachineOperand *Src0 = &MI.getOperand(1);
|
|
|
|
MachineOperand *Src1 = &MI.getOperand(2);
|
|
|
|
|
|
|
|
if (!Src0->isReg() && Src1->isReg()) {
|
|
|
|
if (TII->commuteInstruction(MI, false, 1, 2))
|
|
|
|
std::swap(Src0, Src1);
|
|
|
|
}
|
2016-04-16 09:46:49 +08:00
|
|
|
|
|
|
|
// FIXME: This could work better if hints worked with subregisters. If
|
|
|
|
// we have a vector add of a constant, we usually don't get the correct
|
|
|
|
// allocation due to the subregister usage.
|
2016-09-09 01:35:41 +08:00
|
|
|
if (TargetRegisterInfo::isVirtualRegister(Dest->getReg()) &&
|
|
|
|
Src0->isReg()) {
|
|
|
|
MRI.setRegAllocationHint(Dest->getReg(), 0, Src0->getReg());
|
|
|
|
MRI.setRegAllocationHint(Src0->getReg(), 0, Dest->getReg());
|
2016-04-16 09:46:49 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-09-09 01:35:41 +08:00
|
|
|
if (Src0->isReg() && Src0->getReg() == Dest->getReg()) {
|
|
|
|
if (Src1->isImm() && isKImmOperand(TII, *Src1)) {
|
2016-04-16 09:46:49 +08:00
|
|
|
unsigned Opc = (MI.getOpcode() == AMDGPU::S_ADD_I32) ?
|
|
|
|
AMDGPU::S_ADDK_I32 : AMDGPU::S_MULK_I32;
|
|
|
|
|
|
|
|
MI.setDesc(TII->get(Opc));
|
|
|
|
MI.tieOperands(0, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-17 05:41:16 +08:00
|
|
|
// Try to use s_cmpk_*
|
|
|
|
if (MI.isCompare() && TII->isSOPC(MI)) {
|
|
|
|
shrinkScalarCompare(TII, MI);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-04-16 09:46:49 +08:00
|
|
|
// Try to use S_MOVK_I32, which will save 4 bytes for small immediates.
|
|
|
|
if (MI.getOpcode() == AMDGPU::S_MOV_B32) {
|
2016-11-02 07:14:20 +08:00
|
|
|
const MachineOperand &Dst = MI.getOperand(0);
|
|
|
|
MachineOperand &Src = MI.getOperand(1);
|
2016-04-16 09:46:49 +08:00
|
|
|
|
2016-11-02 07:14:20 +08:00
|
|
|
if (Src.isImm() &&
|
|
|
|
TargetRegisterInfo::isPhysicalRegister(Dst.getReg())) {
|
|
|
|
int32_t ReverseImm;
|
|
|
|
if (isKImmOperand(TII, Src))
|
|
|
|
MI.setDesc(TII->get(AMDGPU::S_MOVK_I32));
|
|
|
|
else if (isReverseInlineImm(TII, Src, ReverseImm)) {
|
|
|
|
MI.setDesc(TII->get(AMDGPU::S_BREV_B32));
|
|
|
|
Src.setImm(ReverseImm);
|
|
|
|
}
|
|
|
|
}
|
2016-04-16 09:46:49 +08:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
[AMDGPU] Shrink scalar AND, OR, XOR instructions
This change attempts to shrink scalar AND, OR and XOR instructions which take an immediate that isn't inlineable.
It performs:
AND s0, s0, ~(1 << n) -> BITSET0 s0, n
OR s0, s0, (1 << n) -> BITSET1 s0, n
AND s0, s1, x -> ANDN2 s0, s1, ~x
OR s0, s1, x -> ORN2 s0, s1, ~x
XOR s0, s1, x -> XNOR s0, s1, ~x
In particular, this catches setting and clearing the sign bit for fabs (and x, 0x7ffffffff -> bitset0 x, 31 and or x, 0x80000000 -> bitset1 x, 31).
llvm-svn: 348601
2018-12-07 23:33:21 +08:00
|
|
|
// Shrink scalar logic operations.
|
|
|
|
if (MI.getOpcode() == AMDGPU::S_AND_B32 ||
|
|
|
|
MI.getOpcode() == AMDGPU::S_OR_B32 ||
|
|
|
|
MI.getOpcode() == AMDGPU::S_XOR_B32) {
|
|
|
|
if (shrinkScalarLogicOp(ST, MRI, TII, MI))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-08-01 08:32:28 +08:00
|
|
|
if (!TII->hasVALU32BitEncoding(MI.getOpcode()))
|
2014-07-22 00:55:33 +08:00
|
|
|
continue;
|
|
|
|
|
2018-08-29 02:22:34 +08:00
|
|
|
if (!TII->canShrink(MI, MRI)) {
|
2014-09-17 02:00:23 +08:00
|
|
|
// Try commuting the instruction and see if that enables us to shrink
|
2014-07-22 00:55:33 +08:00
|
|
|
// it.
|
2016-06-30 08:01:54 +08:00
|
|
|
if (!MI.isCommutable() || !TII->commuteInstruction(MI) ||
|
2018-08-29 02:22:34 +08:00
|
|
|
!TII->canShrink(MI, MRI))
|
2014-07-22 00:55:33 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-01-16 02:42:51 +08:00
|
|
|
// getVOPe32 could be -1 here if we started with an instruction that had
|
2014-08-01 08:32:28 +08:00
|
|
|
// a 32-bit encoding and then commuted it to an instruction that did not.
|
2015-01-16 02:42:51 +08:00
|
|
|
if (!TII->hasVALU32BitEncoding(MI.getOpcode()))
|
2014-08-01 08:32:28 +08:00
|
|
|
continue;
|
|
|
|
|
2015-01-16 02:42:51 +08:00
|
|
|
int Op32 = AMDGPU::getVOPe32(MI.getOpcode());
|
|
|
|
|
2014-07-22 00:55:33 +08:00
|
|
|
if (TII->isVOPC(Op32)) {
|
|
|
|
unsigned DstReg = MI.getOperand(0).getReg();
|
|
|
|
if (TargetRegisterInfo::isVirtualRegister(DstReg)) {
|
2015-08-08 08:41:45 +08:00
|
|
|
// VOPC instructions can only write to the VCC register. We can't
|
|
|
|
// force them to use VCC here, because this is only one register and
|
|
|
|
// cannot deal with sequences which would require multiple copies of
|
|
|
|
// VCC, e.g. S_AND_B64 (vcc = V_CMP_...), (vcc = V_CMP_...)
|
2014-07-22 00:55:33 +08:00
|
|
|
//
|
2014-09-22 01:27:32 +08:00
|
|
|
// So, instead of forcing the instruction to write to VCC, we provide
|
2018-04-13 19:37:06 +08:00
|
|
|
// a hint to the register allocator to use VCC and then we will run
|
2014-09-22 01:27:32 +08:00
|
|
|
// this pass again after RA and shrink it if it outputs to VCC.
|
2014-07-22 00:55:33 +08:00
|
|
|
MRI.setRegAllocationHint(MI.getOperand(0).getReg(), 0, AMDGPU::VCC);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (DstReg != AMDGPU::VCC)
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-07-14 22:15:03 +08:00
|
|
|
if (Op32 == AMDGPU::V_CNDMASK_B32_e32) {
|
|
|
|
// We shrink V_CNDMASK_B32_e64 using regalloc hints like we do for VOPC
|
|
|
|
// instructions.
|
|
|
|
const MachineOperand *Src2 =
|
|
|
|
TII->getNamedOperand(MI, AMDGPU::OpName::src2);
|
|
|
|
if (!Src2->isReg())
|
|
|
|
continue;
|
|
|
|
unsigned SReg = Src2->getReg();
|
|
|
|
if (TargetRegisterInfo::isVirtualRegister(SReg)) {
|
|
|
|
MRI.setRegAllocationHint(SReg, 0, AMDGPU::VCC);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (SReg != AMDGPU::VCC)
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-01-12 06:35:17 +08:00
|
|
|
// Check for the bool flag output for instructions like V_ADD_I32_e64.
|
|
|
|
const MachineOperand *SDst = TII->getNamedOperand(MI,
|
|
|
|
AMDGPU::OpName::sdst);
|
|
|
|
|
2017-01-12 06:58:12 +08:00
|
|
|
// Check the carry-in operand for v_addc_u32_e64.
|
|
|
|
const MachineOperand *Src2 = TII->getNamedOperand(MI,
|
|
|
|
AMDGPU::OpName::src2);
|
|
|
|
|
|
|
|
if (SDst) {
|
|
|
|
if (SDst->getReg() != AMDGPU::VCC) {
|
|
|
|
if (TargetRegisterInfo::isVirtualRegister(SDst->getReg()))
|
|
|
|
MRI.setRegAllocationHint(SDst->getReg(), 0, AMDGPU::VCC);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// All of the instructions with carry outs also have an SGPR input in
|
|
|
|
// src2.
|
|
|
|
if (Src2 && Src2->getReg() != AMDGPU::VCC) {
|
|
|
|
if (TargetRegisterInfo::isVirtualRegister(Src2->getReg()))
|
|
|
|
MRI.setRegAllocationHint(Src2->getReg(), 0, AMDGPU::VCC);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
2017-01-12 06:35:17 +08:00
|
|
|
}
|
|
|
|
|
2014-07-22 00:55:33 +08:00
|
|
|
// We can shrink this instruction
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Shrinking " << MI);
|
2014-07-22 00:55:33 +08:00
|
|
|
|
2018-08-29 02:34:24 +08:00
|
|
|
MachineInstr *Inst32 = TII->buildShrunkInst(MI, Op32);
|
2014-07-22 00:55:33 +08:00
|
|
|
++NumInstructionsShrunk;
|
2014-08-01 08:32:33 +08:00
|
|
|
|
2016-07-19 08:35:03 +08:00
|
|
|
// Copy extra operands not present in the instruction definition.
|
2016-09-04 01:25:39 +08:00
|
|
|
copyExtraImplicitOps(*Inst32, MF, MI);
|
2016-07-19 08:35:03 +08:00
|
|
|
|
|
|
|
MI.eraseFromParent();
|
2014-08-01 08:32:33 +08:00
|
|
|
foldImmediates(*Inst32, TII, MRI);
|
2016-07-19 08:35:03 +08:00
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "e32 MI = " << *Inst32 << '\n');
|
2014-07-22 00:55:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|