forked from OSchip/llvm-project
470 lines
16 KiB
C++
470 lines
16 KiB
C++
//===----- SVEIntrinsicOpts - SVE ACLE Intrinsics Opts --------------------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Performs general IR level optimizations on SVE intrinsics.
|
|
//
|
|
// This pass performs the following optimizations:
|
|
//
|
|
// - removes unnecessary ptrue intrinsics (llvm.aarch64.sve.ptrue), e.g:
|
|
// %1 = @llvm.aarch64.sve.ptrue.nxv4i1(i32 31)
|
|
// %2 = @llvm.aarch64.sve.ptrue.nxv8i1(i32 31)
|
|
// ; (%1 can be replaced with a reinterpret of %2)
|
|
//
|
|
// - optimizes ptest intrinsics where the operands are being needlessly
|
|
// converted to and from svbool_t.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "AArch64.h"
|
|
#include "Utils/AArch64BaseInfo.h"
|
|
#include "llvm/ADT/PostOrderIterator.h"
|
|
#include "llvm/ADT/SetVector.h"
|
|
#include "llvm/IR/Constants.h"
|
|
#include "llvm/IR/Dominators.h"
|
|
#include "llvm/IR/IRBuilder.h"
|
|
#include "llvm/IR/Instructions.h"
|
|
#include "llvm/IR/IntrinsicInst.h"
|
|
#include "llvm/IR/IntrinsicsAArch64.h"
|
|
#include "llvm/IR/LLVMContext.h"
|
|
#include "llvm/IR/PatternMatch.h"
|
|
#include "llvm/InitializePasses.h"
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
using namespace llvm;
|
|
using namespace llvm::PatternMatch;
|
|
|
|
#define DEBUG_TYPE "aarch64-sve-intrinsic-opts"
|
|
|
|
namespace {
|
|
struct SVEIntrinsicOpts : public ModulePass {
|
|
static char ID; // Pass identification, replacement for typeid
|
|
SVEIntrinsicOpts() : ModulePass(ID) {
|
|
initializeSVEIntrinsicOptsPass(*PassRegistry::getPassRegistry());
|
|
}
|
|
|
|
bool runOnModule(Module &M) override;
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override;
|
|
|
|
private:
|
|
bool coalescePTrueIntrinsicCalls(BasicBlock &BB,
|
|
SmallSetVector<IntrinsicInst *, 4> &PTrues);
|
|
bool optimizePTrueIntrinsicCalls(SmallSetVector<Function *, 4> &Functions);
|
|
bool optimizePredicateStore(Instruction *I);
|
|
bool optimizePredicateLoad(Instruction *I);
|
|
|
|
bool optimizeInstructions(SmallSetVector<Function *, 4> &Functions);
|
|
|
|
/// Operates at the function-scope. I.e., optimizations are applied local to
|
|
/// the functions themselves.
|
|
bool optimizeFunctions(SmallSetVector<Function *, 4> &Functions);
|
|
};
|
|
} // end anonymous namespace
|
|
|
|
void SVEIntrinsicOpts::getAnalysisUsage(AnalysisUsage &AU) const {
|
|
AU.addRequired<DominatorTreeWrapperPass>();
|
|
AU.setPreservesCFG();
|
|
}
|
|
|
|
char SVEIntrinsicOpts::ID = 0;
|
|
static const char *name = "SVE intrinsics optimizations";
|
|
INITIALIZE_PASS_BEGIN(SVEIntrinsicOpts, DEBUG_TYPE, name, false, false)
|
|
INITIALIZE_PASS_DEPENDENCY(DominatorTreeWrapperPass);
|
|
INITIALIZE_PASS_END(SVEIntrinsicOpts, DEBUG_TYPE, name, false, false)
|
|
|
|
ModulePass *llvm::createSVEIntrinsicOptsPass() {
|
|
return new SVEIntrinsicOpts();
|
|
}
|
|
|
|
/// Checks if a ptrue intrinsic call is promoted. The act of promoting a
|
|
/// ptrue will introduce zeroing. For example:
|
|
///
|
|
/// %1 = <vscale x 4 x i1> call @llvm.aarch64.sve.ptrue.nxv4i1(i32 31)
|
|
/// %2 = <vscale x 16 x i1> call @llvm.aarch64.sve.convert.to.svbool.nxv4i1(<vscale x 4 x i1> %1)
|
|
/// %3 = <vscale x 8 x i1> call @llvm.aarch64.sve.convert.from.svbool.nxv8i1(<vscale x 16 x i1> %2)
|
|
///
|
|
/// %1 is promoted, because it is converted:
|
|
///
|
|
/// <vscale x 4 x i1> => <vscale x 16 x i1> => <vscale x 8 x i1>
|
|
///
|
|
/// via a sequence of the SVE reinterpret intrinsics convert.{to,from}.svbool.
|
|
static bool isPTruePromoted(IntrinsicInst *PTrue) {
|
|
// Find all users of this intrinsic that are calls to convert-to-svbool
|
|
// reinterpret intrinsics.
|
|
SmallVector<IntrinsicInst *, 4> ConvertToUses;
|
|
for (User *User : PTrue->users()) {
|
|
if (match(User, m_Intrinsic<Intrinsic::aarch64_sve_convert_to_svbool>())) {
|
|
ConvertToUses.push_back(cast<IntrinsicInst>(User));
|
|
}
|
|
}
|
|
|
|
// If no such calls were found, this is ptrue is not promoted.
|
|
if (ConvertToUses.empty())
|
|
return false;
|
|
|
|
// Otherwise, try to find users of the convert-to-svbool intrinsics that are
|
|
// calls to the convert-from-svbool intrinsic, and would result in some lanes
|
|
// being zeroed.
|
|
const auto *PTrueVTy = cast<ScalableVectorType>(PTrue->getType());
|
|
for (IntrinsicInst *ConvertToUse : ConvertToUses) {
|
|
for (User *User : ConvertToUse->users()) {
|
|
auto *IntrUser = dyn_cast<IntrinsicInst>(User);
|
|
if (IntrUser && IntrUser->getIntrinsicID() ==
|
|
Intrinsic::aarch64_sve_convert_from_svbool) {
|
|
const auto *IntrUserVTy = cast<ScalableVectorType>(IntrUser->getType());
|
|
|
|
// Would some lanes become zeroed by the conversion?
|
|
if (IntrUserVTy->getElementCount().getKnownMinValue() >
|
|
PTrueVTy->getElementCount().getKnownMinValue())
|
|
// This is a promoted ptrue.
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// If no matching calls were found, this is not a promoted ptrue.
|
|
return false;
|
|
}
|
|
|
|
/// Attempts to coalesce ptrues in a basic block.
|
|
bool SVEIntrinsicOpts::coalescePTrueIntrinsicCalls(
|
|
BasicBlock &BB, SmallSetVector<IntrinsicInst *, 4> &PTrues) {
|
|
if (PTrues.size() <= 1)
|
|
return false;
|
|
|
|
// Find the ptrue with the most lanes.
|
|
auto *MostEncompassingPTrue = *std::max_element(
|
|
PTrues.begin(), PTrues.end(), [](auto *PTrue1, auto *PTrue2) {
|
|
auto *PTrue1VTy = cast<ScalableVectorType>(PTrue1->getType());
|
|
auto *PTrue2VTy = cast<ScalableVectorType>(PTrue2->getType());
|
|
return PTrue1VTy->getElementCount().getKnownMinValue() <
|
|
PTrue2VTy->getElementCount().getKnownMinValue();
|
|
});
|
|
|
|
// Remove the most encompassing ptrue, as well as any promoted ptrues, leaving
|
|
// behind only the ptrues to be coalesced.
|
|
PTrues.remove(MostEncompassingPTrue);
|
|
PTrues.remove_if(isPTruePromoted);
|
|
|
|
// Hoist MostEncompassingPTrue to the start of the basic block. It is always
|
|
// safe to do this, since ptrue intrinsic calls are guaranteed to have no
|
|
// predecessors.
|
|
MostEncompassingPTrue->moveBefore(BB, BB.getFirstInsertionPt());
|
|
|
|
LLVMContext &Ctx = BB.getContext();
|
|
IRBuilder<> Builder(Ctx);
|
|
Builder.SetInsertPoint(&BB, ++MostEncompassingPTrue->getIterator());
|
|
|
|
auto *MostEncompassingPTrueVTy =
|
|
cast<VectorType>(MostEncompassingPTrue->getType());
|
|
auto *ConvertToSVBool = Builder.CreateIntrinsic(
|
|
Intrinsic::aarch64_sve_convert_to_svbool, {MostEncompassingPTrueVTy},
|
|
{MostEncompassingPTrue});
|
|
|
|
bool ConvertFromCreated = false;
|
|
for (auto *PTrue : PTrues) {
|
|
auto *PTrueVTy = cast<VectorType>(PTrue->getType());
|
|
|
|
// Only create the converts if the types are not already the same, otherwise
|
|
// just use the most encompassing ptrue.
|
|
if (MostEncompassingPTrueVTy != PTrueVTy) {
|
|
ConvertFromCreated = true;
|
|
|
|
Builder.SetInsertPoint(&BB, ++ConvertToSVBool->getIterator());
|
|
auto *ConvertFromSVBool =
|
|
Builder.CreateIntrinsic(Intrinsic::aarch64_sve_convert_from_svbool,
|
|
{PTrueVTy}, {ConvertToSVBool});
|
|
PTrue->replaceAllUsesWith(ConvertFromSVBool);
|
|
} else
|
|
PTrue->replaceAllUsesWith(MostEncompassingPTrue);
|
|
|
|
PTrue->eraseFromParent();
|
|
}
|
|
|
|
// We never used the ConvertTo so remove it
|
|
if (!ConvertFromCreated)
|
|
ConvertToSVBool->eraseFromParent();
|
|
|
|
return true;
|
|
}
|
|
|
|
/// The goal of this function is to remove redundant calls to the SVE ptrue
|
|
/// intrinsic in each basic block within the given functions.
|
|
///
|
|
/// SVE ptrues have two representations in LLVM IR:
|
|
/// - a logical representation -- an arbitrary-width scalable vector of i1s,
|
|
/// i.e. <vscale x N x i1>.
|
|
/// - a physical representation (svbool, <vscale x 16 x i1>) -- a 16-element
|
|
/// scalable vector of i1s, i.e. <vscale x 16 x i1>.
|
|
///
|
|
/// The SVE ptrue intrinsic is used to create a logical representation of an SVE
|
|
/// predicate. Suppose that we have two SVE ptrue intrinsic calls: P1 and P2. If
|
|
/// P1 creates a logical SVE predicate that is at least as wide as the logical
|
|
/// SVE predicate created by P2, then all of the bits that are true in the
|
|
/// physical representation of P2 are necessarily also true in the physical
|
|
/// representation of P1. P1 'encompasses' P2, therefore, the intrinsic call to
|
|
/// P2 is redundant and can be replaced by an SVE reinterpret of P1 via
|
|
/// convert.{to,from}.svbool.
|
|
///
|
|
/// Currently, this pass only coalesces calls to SVE ptrue intrinsics
|
|
/// if they match the following conditions:
|
|
///
|
|
/// - the call to the intrinsic uses either the SV_ALL or SV_POW2 patterns.
|
|
/// SV_ALL indicates that all bits of the predicate vector are to be set to
|
|
/// true. SV_POW2 indicates that all bits of the predicate vector up to the
|
|
/// largest power-of-two are to be set to true.
|
|
/// - the result of the call to the intrinsic is not promoted to a wider
|
|
/// predicate. In this case, keeping the extra ptrue leads to better codegen
|
|
/// -- coalescing here would create an irreducible chain of SVE reinterprets
|
|
/// via convert.{to,from}.svbool.
|
|
///
|
|
/// EXAMPLE:
|
|
///
|
|
/// %1 = <vscale x 8 x i1> ptrue(i32 SV_ALL)
|
|
/// ; Logical: <1, 1, 1, 1, 1, 1, 1, 1>
|
|
/// ; Physical: <1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0>
|
|
/// ...
|
|
///
|
|
/// %2 = <vscale x 4 x i1> ptrue(i32 SV_ALL)
|
|
/// ; Logical: <1, 1, 1, 1>
|
|
/// ; Physical: <1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0>
|
|
/// ...
|
|
///
|
|
/// Here, %2 can be replaced by an SVE reinterpret of %1, giving, for instance:
|
|
///
|
|
/// %1 = <vscale x 8 x i1> ptrue(i32 i31)
|
|
/// %2 = <vscale x 16 x i1> convert.to.svbool(<vscale x 8 x i1> %1)
|
|
/// %3 = <vscale x 4 x i1> convert.from.svbool(<vscale x 16 x i1> %2)
|
|
///
|
|
bool SVEIntrinsicOpts::optimizePTrueIntrinsicCalls(
|
|
SmallSetVector<Function *, 4> &Functions) {
|
|
bool Changed = false;
|
|
|
|
for (auto *F : Functions) {
|
|
for (auto &BB : *F) {
|
|
SmallSetVector<IntrinsicInst *, 4> SVAllPTrues;
|
|
SmallSetVector<IntrinsicInst *, 4> SVPow2PTrues;
|
|
|
|
// For each basic block, collect the used ptrues and try to coalesce them.
|
|
for (Instruction &I : BB) {
|
|
if (I.use_empty())
|
|
continue;
|
|
|
|
auto *IntrI = dyn_cast<IntrinsicInst>(&I);
|
|
if (!IntrI || IntrI->getIntrinsicID() != Intrinsic::aarch64_sve_ptrue)
|
|
continue;
|
|
|
|
const auto PTruePattern =
|
|
cast<ConstantInt>(IntrI->getOperand(0))->getZExtValue();
|
|
|
|
if (PTruePattern == AArch64SVEPredPattern::all)
|
|
SVAllPTrues.insert(IntrI);
|
|
if (PTruePattern == AArch64SVEPredPattern::pow2)
|
|
SVPow2PTrues.insert(IntrI);
|
|
}
|
|
|
|
Changed |= coalescePTrueIntrinsicCalls(BB, SVAllPTrues);
|
|
Changed |= coalescePTrueIntrinsicCalls(BB, SVPow2PTrues);
|
|
}
|
|
}
|
|
|
|
return Changed;
|
|
}
|
|
|
|
// This is done in SVEIntrinsicOpts rather than InstCombine so that we introduce
|
|
// scalable stores as late as possible
|
|
bool SVEIntrinsicOpts::optimizePredicateStore(Instruction *I) {
|
|
auto *F = I->getFunction();
|
|
auto Attr = F->getFnAttribute(Attribute::VScaleRange);
|
|
if (!Attr.isValid())
|
|
return false;
|
|
|
|
unsigned MinVScale = Attr.getVScaleRangeMin();
|
|
Optional<unsigned> MaxVScale = Attr.getVScaleRangeMax();
|
|
// The transform needs to know the exact runtime length of scalable vectors
|
|
if (!MaxVScale || MinVScale != MaxVScale)
|
|
return false;
|
|
|
|
auto *PredType =
|
|
ScalableVectorType::get(Type::getInt1Ty(I->getContext()), 16);
|
|
auto *FixedPredType =
|
|
FixedVectorType::get(Type::getInt8Ty(I->getContext()), MinVScale * 2);
|
|
|
|
// If we have a store..
|
|
auto *Store = dyn_cast<StoreInst>(I);
|
|
if (!Store || !Store->isSimple())
|
|
return false;
|
|
|
|
// ..that is storing a predicate vector sized worth of bits..
|
|
if (Store->getOperand(0)->getType() != FixedPredType)
|
|
return false;
|
|
|
|
// ..where the value stored comes from a vector extract..
|
|
auto *IntrI = dyn_cast<IntrinsicInst>(Store->getOperand(0));
|
|
if (!IntrI ||
|
|
IntrI->getIntrinsicID() != Intrinsic::experimental_vector_extract)
|
|
return false;
|
|
|
|
// ..that is extracting from index 0..
|
|
if (!cast<ConstantInt>(IntrI->getOperand(1))->isZero())
|
|
return false;
|
|
|
|
// ..where the value being extract from comes from a bitcast
|
|
auto *BitCast = dyn_cast<BitCastInst>(IntrI->getOperand(0));
|
|
if (!BitCast)
|
|
return false;
|
|
|
|
// ..and the bitcast is casting from predicate type
|
|
if (BitCast->getOperand(0)->getType() != PredType)
|
|
return false;
|
|
|
|
IRBuilder<> Builder(I->getContext());
|
|
Builder.SetInsertPoint(I);
|
|
|
|
auto *PtrBitCast = Builder.CreateBitCast(
|
|
Store->getPointerOperand(),
|
|
PredType->getPointerTo(Store->getPointerAddressSpace()));
|
|
Builder.CreateStore(BitCast->getOperand(0), PtrBitCast);
|
|
|
|
Store->eraseFromParent();
|
|
if (IntrI->getNumUses() == 0)
|
|
IntrI->eraseFromParent();
|
|
if (BitCast->getNumUses() == 0)
|
|
BitCast->eraseFromParent();
|
|
|
|
return true;
|
|
}
|
|
|
|
// This is done in SVEIntrinsicOpts rather than InstCombine so that we introduce
|
|
// scalable loads as late as possible
|
|
bool SVEIntrinsicOpts::optimizePredicateLoad(Instruction *I) {
|
|
auto *F = I->getFunction();
|
|
auto Attr = F->getFnAttribute(Attribute::VScaleRange);
|
|
if (!Attr.isValid())
|
|
return false;
|
|
|
|
unsigned MinVScale = Attr.getVScaleRangeMin();
|
|
Optional<unsigned> MaxVScale = Attr.getVScaleRangeMax();
|
|
// The transform needs to know the exact runtime length of scalable vectors
|
|
if (!MaxVScale || MinVScale != MaxVScale)
|
|
return false;
|
|
|
|
auto *PredType =
|
|
ScalableVectorType::get(Type::getInt1Ty(I->getContext()), 16);
|
|
auto *FixedPredType =
|
|
FixedVectorType::get(Type::getInt8Ty(I->getContext()), MinVScale * 2);
|
|
|
|
// If we have a bitcast..
|
|
auto *BitCast = dyn_cast<BitCastInst>(I);
|
|
if (!BitCast || BitCast->getType() != PredType)
|
|
return false;
|
|
|
|
// ..whose operand is a vector_insert..
|
|
auto *IntrI = dyn_cast<IntrinsicInst>(BitCast->getOperand(0));
|
|
if (!IntrI ||
|
|
IntrI->getIntrinsicID() != Intrinsic::experimental_vector_insert)
|
|
return false;
|
|
|
|
// ..that is inserting into index zero of an undef vector..
|
|
if (!isa<UndefValue>(IntrI->getOperand(0)) ||
|
|
!cast<ConstantInt>(IntrI->getOperand(2))->isZero())
|
|
return false;
|
|
|
|
// ..where the value inserted comes from a load..
|
|
auto *Load = dyn_cast<LoadInst>(IntrI->getOperand(1));
|
|
if (!Load || !Load->isSimple())
|
|
return false;
|
|
|
|
// ..that is loading a predicate vector sized worth of bits..
|
|
if (Load->getType() != FixedPredType)
|
|
return false;
|
|
|
|
IRBuilder<> Builder(I->getContext());
|
|
Builder.SetInsertPoint(Load);
|
|
|
|
auto *PtrBitCast = Builder.CreateBitCast(
|
|
Load->getPointerOperand(),
|
|
PredType->getPointerTo(Load->getPointerAddressSpace()));
|
|
auto *LoadPred = Builder.CreateLoad(PredType, PtrBitCast);
|
|
|
|
BitCast->replaceAllUsesWith(LoadPred);
|
|
BitCast->eraseFromParent();
|
|
if (IntrI->getNumUses() == 0)
|
|
IntrI->eraseFromParent();
|
|
if (Load->getNumUses() == 0)
|
|
Load->eraseFromParent();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SVEIntrinsicOpts::optimizeInstructions(
|
|
SmallSetVector<Function *, 4> &Functions) {
|
|
bool Changed = false;
|
|
|
|
for (auto *F : Functions) {
|
|
DominatorTree *DT = &getAnalysis<DominatorTreeWrapperPass>(*F).getDomTree();
|
|
|
|
// Traverse the DT with an rpo walk so we see defs before uses, allowing
|
|
// simplification to be done incrementally.
|
|
BasicBlock *Root = DT->getRoot();
|
|
ReversePostOrderTraversal<BasicBlock *> RPOT(Root);
|
|
for (auto *BB : RPOT) {
|
|
for (Instruction &I : make_early_inc_range(*BB)) {
|
|
switch (I.getOpcode()) {
|
|
case Instruction::Store:
|
|
Changed |= optimizePredicateStore(&I);
|
|
break;
|
|
case Instruction::BitCast:
|
|
Changed |= optimizePredicateLoad(&I);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return Changed;
|
|
}
|
|
|
|
bool SVEIntrinsicOpts::optimizeFunctions(
|
|
SmallSetVector<Function *, 4> &Functions) {
|
|
bool Changed = false;
|
|
|
|
Changed |= optimizePTrueIntrinsicCalls(Functions);
|
|
Changed |= optimizeInstructions(Functions);
|
|
|
|
return Changed;
|
|
}
|
|
|
|
bool SVEIntrinsicOpts::runOnModule(Module &M) {
|
|
bool Changed = false;
|
|
SmallSetVector<Function *, 4> Functions;
|
|
|
|
// Check for SVE intrinsic declarations first so that we only iterate over
|
|
// relevant functions. Where an appropriate declaration is found, store the
|
|
// function(s) where it is used so we can target these only.
|
|
for (auto &F : M.getFunctionList()) {
|
|
if (!F.isDeclaration())
|
|
continue;
|
|
|
|
switch (F.getIntrinsicID()) {
|
|
case Intrinsic::experimental_vector_extract:
|
|
case Intrinsic::experimental_vector_insert:
|
|
case Intrinsic::aarch64_sve_ptrue:
|
|
for (User *U : F.users())
|
|
Functions.insert(cast<Instruction>(U)->getFunction());
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!Functions.empty())
|
|
Changed |= optimizeFunctions(Functions);
|
|
|
|
return Changed;
|
|
}
|