2012-10-17 18:25:06 +00:00
|
|
|
//===- LoopVectorize.cpp - A Loop Vectorizer ------------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
2013-01-07 10:44:06 +00:00
|
|
|
//
|
|
|
|
// This is the LLVM loop vectorizer. This pass modifies 'vectorizable' loops
|
2013-05-06 03:06:36 +00:00
|
|
|
// and generates target-independent LLVM-IR.
|
|
|
|
// The vectorizer uses the TargetTransformInfo analysis to estimate the costs
|
|
|
|
// of instructions in order to estimate the profitability of vectorization.
|
2013-01-07 10:44:06 +00:00
|
|
|
//
|
2013-02-08 17:43:32 +00:00
|
|
|
// The loop vectorizer combines consecutive loop iterations into a single
|
2013-01-07 10:44:06 +00:00
|
|
|
// 'wide' iteration. After this transformation the index is incremented
|
|
|
|
// by the SIMD vector width, and not by one.
|
|
|
|
//
|
|
|
|
// This pass has three parts:
|
|
|
|
// 1. The main loop pass that drives the different parts.
|
|
|
|
// 2. LoopVectorizationLegality - A unit that checks for the legality
|
|
|
|
// of the vectorization.
|
|
|
|
// 3. InnerLoopVectorizer - A unit that performs the actual
|
|
|
|
// widening of instructions.
|
|
|
|
// 4. LoopVectorizationCostModel - A unit that checks for the profitability
|
|
|
|
// of vectorization. It decides on the optimal vector width, which
|
|
|
|
// can be one, if vectorization is not profitable.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// The reduction-variable vectorization is based on the paper:
|
|
|
|
// D. Nuzman and R. Henderson. Multi-platform Auto-vectorization.
|
|
|
|
//
|
|
|
|
// Variable uniformity checks are inspired by:
|
2013-02-08 12:58:29 +00:00
|
|
|
// Karrenberg, R. and Hack, S. Whole Function Vectorization.
|
2013-01-07 10:44:06 +00:00
|
|
|
//
|
|
|
|
// Other ideas/concepts are from:
|
|
|
|
// A. Zaks and D. Nuzman. Autovectorization in GCC-two years later.
|
|
|
|
//
|
|
|
|
// S. Maleki, Y. Gao, M. Garzaran, T. Wong and D. Padua. An Evaluation of
|
|
|
|
// Vectorizing Compilers.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#define LV_NAME "loop-vectorize"
|
|
|
|
#define DEBUG_TYPE LV_NAME
|
|
|
|
|
|
|
|
#include "llvm/Transforms/Vectorize.h"
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2013-06-24 03:55:44 +00:00
|
|
|
#include "llvm/ADT/EquivalenceClasses.h"
|
2013-01-07 10:44:06 +00:00
|
|
|
#include "llvm/ADT/MapVector.h"
|
2013-06-24 12:09:12 +00:00
|
|
|
#include "llvm/ADT/SetVector.h"
|
2013-01-07 10:44:06 +00:00
|
|
|
#include "llvm/ADT/SmallPtrSet.h"
|
2013-01-04 17:48:25 +00:00
|
|
|
#include "llvm/ADT/SmallSet.h"
|
2013-01-07 10:44:06 +00:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2012-10-17 18:25:06 +00:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
|
|
|
#include "llvm/Analysis/AliasAnalysis.h"
|
2012-10-29 21:52:38 +00:00
|
|
|
#include "llvm/Analysis/Dominators.h"
|
2012-10-17 18:25:06 +00:00
|
|
|
#include "llvm/Analysis/LoopInfo.h"
|
2012-12-04 06:15:11 +00:00
|
|
|
#include "llvm/Analysis/LoopIterator.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/Analysis/LoopPass.h"
|
2013-01-07 10:44:06 +00:00
|
|
|
#include "llvm/Analysis/ScalarEvolution.h"
|
2012-12-10 21:39:02 +00:00
|
|
|
#include "llvm/Analysis/ScalarEvolutionExpander.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/Analysis/ScalarEvolutionExpressions.h"
|
2013-01-07 03:08:10 +00:00
|
|
|
#include "llvm/Analysis/TargetTransformInfo.h"
|
2012-10-24 20:36:32 +00:00
|
|
|
#include "llvm/Analysis/ValueTracking.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/Analysis/Verifier.h"
|
2013-01-02 11:36:10 +00:00
|
|
|
#include "llvm/IR/Constants.h"
|
|
|
|
#include "llvm/IR/DataLayout.h"
|
|
|
|
#include "llvm/IR/DerivedTypes.h"
|
|
|
|
#include "llvm/IR/Function.h"
|
2013-01-07 10:44:06 +00:00
|
|
|
#include "llvm/IR/IRBuilder.h"
|
2013-01-02 11:36:10 +00:00
|
|
|
#include "llvm/IR/Instructions.h"
|
|
|
|
#include "llvm/IR/IntrinsicInst.h"
|
|
|
|
#include "llvm/IR/LLVMContext.h"
|
|
|
|
#include "llvm/IR/Module.h"
|
|
|
|
#include "llvm/IR/Type.h"
|
|
|
|
#include "llvm/IR/Value.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/Pass.h"
|
2012-10-17 18:25:06 +00:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Support/Debug.h"
|
2013-04-19 21:03:36 +00:00
|
|
|
#include "llvm/Support/PatternMatch.h"
|
2012-10-17 18:25:06 +00:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2013-05-22 16:54:56 +00:00
|
|
|
#include "llvm/Support/ValueHandle.h"
|
2013-02-27 15:24:19 +00:00
|
|
|
#include "llvm/Target/TargetLibraryInfo.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/Transforms/Scalar.h"
|
|
|
|
#include "llvm/Transforms/Utils/BasicBlockUtils.h"
|
2012-10-17 18:25:06 +00:00
|
|
|
#include "llvm/Transforms/Utils/Local.h"
|
2013-01-07 10:44:06 +00:00
|
|
|
#include <algorithm>
|
|
|
|
#include <map>
|
|
|
|
|
|
|
|
using namespace llvm;
|
2013-04-19 21:03:36 +00:00
|
|
|
using namespace llvm::PatternMatch;
|
2012-10-17 18:25:06 +00:00
|
|
|
|
|
|
|
static cl::opt<unsigned>
|
2012-10-24 20:36:32 +00:00
|
|
|
VectorizationFactor("force-vector-width", cl::init(0), cl::Hidden,
|
2012-12-10 21:39:02 +00:00
|
|
|
cl::desc("Sets the SIMD width. Zero is autoselect."));
|
2012-10-24 20:36:32 +00:00
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
static cl::opt<unsigned>
|
2013-01-04 17:48:25 +00:00
|
|
|
VectorizationUnroll("force-vector-unroll", cl::init(0), cl::Hidden,
|
2013-01-03 00:52:27 +00:00
|
|
|
cl::desc("Sets the vectorization unroll count. "
|
|
|
|
"Zero is autoselect."));
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
static cl::opt<bool>
|
2012-12-21 04:47:54 +00:00
|
|
|
EnableIfConversion("enable-if-conversion", cl::init(true), cl::Hidden,
|
2012-12-03 21:06:35 +00:00
|
|
|
cl::desc("Enable if-conversion during vectorization."));
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// We don't vectorize loops with a known constant trip count below this number.
|
2013-01-29 21:42:08 +00:00
|
|
|
static cl::opt<unsigned>
|
2013-02-07 22:34:07 +00:00
|
|
|
TinyTripCountVectorThreshold("vectorizer-min-trip-count", cl::init(16),
|
|
|
|
cl::Hidden,
|
|
|
|
cl::desc("Don't vectorize loops with a constant "
|
|
|
|
"trip count that is smaller than this "
|
|
|
|
"value."));
|
2013-01-07 21:54:51 +00:00
|
|
|
|
|
|
|
/// We don't unroll loops with a known constant trip count below this number.
|
|
|
|
static const unsigned TinyTripCountUnrollThreshold = 128;
|
|
|
|
|
2013-04-26 05:08:59 +00:00
|
|
|
/// When performing memory disambiguation checks at runtime do not make more
|
|
|
|
/// than this number of comparisons.
|
|
|
|
static const unsigned RuntimeMemoryCheckThreshold = 8;
|
2013-01-07 10:44:06 +00:00
|
|
|
|
2013-05-28 20:00:34 +00:00
|
|
|
/// Maximum simd width.
|
|
|
|
static const unsigned MaxVectorWidth = 64;
|
|
|
|
|
|
|
|
/// Maximum vectorization unroll count.
|
|
|
|
static const unsigned MaxUnrollFactor = 16;
|
2013-03-02 01:33:49 +00:00
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
/// The cost of a loop that is considered 'small' by the unroller.
|
|
|
|
static const unsigned SmallLoopCost = 20;
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
namespace {
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
// Forward declarations.
|
|
|
|
class LoopVectorizationLegality;
|
|
|
|
class LoopVectorizationCostModel;
|
|
|
|
|
|
|
|
/// InnerLoopVectorizer vectorizes loops which contain only one basic
|
|
|
|
/// block to a specified vectorization factor (VF).
|
|
|
|
/// This class performs the widening of scalars into vectors, or multiple
|
|
|
|
/// scalars. This class also implements the following features:
|
|
|
|
/// * It inserts an epilogue loop for handling loops that don't have iteration
|
|
|
|
/// counts that are known to be a multiple of the vectorization factor.
|
|
|
|
/// * It handles the code generation for reduction variables.
|
|
|
|
/// * Scalarization (implementation using scalars) of un-vectorizable
|
|
|
|
/// instructions.
|
|
|
|
/// InnerLoopVectorizer does not perform any vectorization-legality
|
|
|
|
/// checks, and relies on the caller to check for the different legality
|
|
|
|
/// aspects. The InnerLoopVectorizer relies on the
|
|
|
|
/// LoopVectorizationLegality class to provide information about the induction
|
|
|
|
/// and reduction variables that were found to a given vectorization factor.
|
|
|
|
class InnerLoopVectorizer {
|
|
|
|
public:
|
|
|
|
InnerLoopVectorizer(Loop *OrigLoop, ScalarEvolution *SE, LoopInfo *LI,
|
2013-02-27 15:24:19 +00:00
|
|
|
DominatorTree *DT, DataLayout *DL,
|
|
|
|
const TargetLibraryInfo *TLI, unsigned VecWidth,
|
2013-01-07 10:44:06 +00:00
|
|
|
unsigned UnrollFactor)
|
2013-02-27 15:24:19 +00:00
|
|
|
: OrigLoop(OrigLoop), SE(SE), LI(LI), DT(DT), DL(DL), TLI(TLI),
|
|
|
|
VF(VecWidth), UF(UnrollFactor), Builder(SE->getContext()), Induction(0),
|
2013-01-07 10:44:06 +00:00
|
|
|
OldInduction(0), WidenMap(UnrollFactor) {}
|
|
|
|
|
|
|
|
// Perform the actual loop widening (vectorization).
|
|
|
|
void vectorize(LoopVectorizationLegality *Legal) {
|
|
|
|
// Create a new empty loop. Unlink the old loop and connect the new one.
|
|
|
|
createEmptyLoop(Legal);
|
|
|
|
// Widen each instruction in the old loop to a new one in the new loop.
|
|
|
|
// Use the Legality module to find the induction and reduction variables.
|
|
|
|
vectorizeLoop(Legal);
|
|
|
|
// Register the new loop and update the analysis passes.
|
|
|
|
updateAnalysis();
|
|
|
|
}
|
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
virtual ~InnerLoopVectorizer() {}
|
|
|
|
|
|
|
|
protected:
|
2013-01-07 10:44:06 +00:00
|
|
|
/// A small list of PHINodes.
|
|
|
|
typedef SmallVector<PHINode*, 4> PhiVector;
|
|
|
|
/// When we unroll loops we have multiple vector values for each scalar.
|
|
|
|
/// This data structure holds the unrolled and vectorized values that
|
|
|
|
/// originated from one scalar instruction.
|
|
|
|
typedef SmallVector<Value*, 2> VectorParts;
|
|
|
|
|
2013-06-27 20:31:06 +00:00
|
|
|
// When we if-convert we need create edge masks. We have to cache values so
|
|
|
|
// that we don't end up with exponential recursion/IR.
|
|
|
|
typedef DenseMap<std::pair<BasicBlock*, BasicBlock*>,
|
|
|
|
VectorParts> EdgeMaskCache;
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Add code that checks at runtime if the accessed arrays overlap.
|
|
|
|
/// Returns the comparator value or NULL if no check is needed.
|
2013-01-19 13:57:58 +00:00
|
|
|
Instruction *addRuntimeCheck(LoopVectorizationLegality *Legal,
|
|
|
|
Instruction *Loc);
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Create an empty loop, based on the loop ranges of the old loop.
|
|
|
|
void createEmptyLoop(LoopVectorizationLegality *Legal);
|
|
|
|
/// Copy and widen the instructions from the old loop.
|
2013-08-26 22:33:26 +00:00
|
|
|
virtual void vectorizeLoop(LoopVectorizationLegality *Legal);
|
|
|
|
|
|
|
|
/// \brief The Loop exit block may have single value PHI nodes where the
|
|
|
|
/// incoming value is 'Undef'. While vectorizing we only handled real values
|
|
|
|
/// that were defined inside the loop. Here we fix the 'undef case'.
|
|
|
|
/// See PR14725.
|
|
|
|
void fixLCSSAPHIs();
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// A helper function that computes the predicate of the block BB, assuming
|
|
|
|
/// that the header block of the loop is set to True. It returns the *entry*
|
|
|
|
/// mask for the block BB.
|
|
|
|
VectorParts createBlockInMask(BasicBlock *BB);
|
|
|
|
/// A helper function that computes the predicate of the edge between SRC
|
|
|
|
/// and DST.
|
|
|
|
VectorParts createEdgeMask(BasicBlock *Src, BasicBlock *Dst);
|
|
|
|
|
|
|
|
/// A helper function to vectorize a single BB within the innermost loop.
|
|
|
|
void vectorizeBlockInLoop(LoopVectorizationLegality *Legal, BasicBlock *BB,
|
|
|
|
PhiVector *PV);
|
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
/// Vectorize a single PHINode in a block. This method handles the induction
|
|
|
|
/// variable canonicalization. It supports both VF = 1 for unrolled loops and
|
|
|
|
/// arbitrary length vectors.
|
|
|
|
void widenPHIInstruction(Instruction *PN, VectorParts &Entry,
|
|
|
|
LoopVectorizationLegality *Legal,
|
|
|
|
unsigned UF, unsigned VF, PhiVector *PV);
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Insert the new loop to the loop hierarchy and pass manager
|
|
|
|
/// and update the analysis passes.
|
|
|
|
void updateAnalysis();
|
|
|
|
|
|
|
|
/// This instruction is un-vectorizable. Implement it as a sequence
|
|
|
|
/// of scalars.
|
2013-08-26 22:33:26 +00:00
|
|
|
virtual void scalarizeInstruction(Instruction *Instr);
|
2013-01-07 10:44:06 +00:00
|
|
|
|
2013-01-25 21:47:42 +00:00
|
|
|
/// Vectorize Load and Store instructions,
|
2013-08-26 22:33:26 +00:00
|
|
|
virtual void vectorizeMemoryInstruction(Instruction *Instr,
|
2013-01-25 21:47:42 +00:00
|
|
|
LoopVectorizationLegality *Legal);
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Create a broadcast instruction. This method generates a broadcast
|
|
|
|
/// instruction (shuffle) for loop invariant values and for the induction
|
|
|
|
/// value. If this is the induction variable then we extend it to N, N+1, ...
|
|
|
|
/// this is needed because each iteration in the loop corresponds to a SIMD
|
|
|
|
/// element.
|
2013-08-26 22:33:26 +00:00
|
|
|
virtual Value *getBroadcastInstrs(Value *V);
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// This function adds 0, 1, 2 ... to each vector element, starting at zero.
|
|
|
|
/// If Negate is set then negative numbers are added e.g. (0, -1, -2, ...).
|
|
|
|
/// The sequence starts at StartIndex.
|
2013-08-26 22:33:26 +00:00
|
|
|
virtual Value *getConsecutiveVector(Value* Val, int StartIdx, bool Negate);
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// When we go over instructions in the basic block we rely on previous
|
|
|
|
/// values within the current basic block or on loop invariant values.
|
|
|
|
/// When we widen (vectorize) values we place them in the map. If the values
|
|
|
|
/// are not within the map, they have to be loop invariant, so we simply
|
|
|
|
/// broadcast them into a vector.
|
|
|
|
VectorParts &getVectorValue(Value *V);
|
|
|
|
|
|
|
|
/// Generate a shuffle sequence that will reverse the vector Vec.
|
2013-08-26 22:33:26 +00:00
|
|
|
virtual Value *reverseVector(Value *Vec);
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// This is a helper class that holds the vectorizer state. It maps scalar
|
|
|
|
/// instructions to vector instructions. When the code is 'unrolled' then
|
|
|
|
/// then a single scalar value is mapped to multiple vector parts. The parts
|
|
|
|
/// are stored in the VectorPart type.
|
|
|
|
struct ValueMap {
|
|
|
|
/// C'tor. UnrollFactor controls the number of vectors ('parts') that
|
|
|
|
/// are mapped.
|
|
|
|
ValueMap(unsigned UnrollFactor) : UF(UnrollFactor) {}
|
|
|
|
|
|
|
|
/// \return True if 'Key' is saved in the Value Map.
|
2013-01-29 17:31:33 +00:00
|
|
|
bool has(Value *Key) const { return MapStorage.count(Key); }
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// Initializes a new entry in the map. Sets all of the vector parts to the
|
|
|
|
/// save value in 'Val'.
|
|
|
|
/// \return A reference to a vector with splat values.
|
|
|
|
VectorParts &splat(Value *Key, Value *Val) {
|
2013-01-29 17:31:33 +00:00
|
|
|
VectorParts &Entry = MapStorage[Key];
|
|
|
|
Entry.assign(UF, Val);
|
|
|
|
return Entry;
|
2013-01-07 10:44:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
///\return A reference to the value that is stored at 'Key'.
|
|
|
|
VectorParts &get(Value *Key) {
|
2013-01-29 17:31:33 +00:00
|
|
|
VectorParts &Entry = MapStorage[Key];
|
|
|
|
if (Entry.empty())
|
|
|
|
Entry.resize(UF);
|
|
|
|
assert(Entry.size() == UF);
|
|
|
|
return Entry;
|
2013-01-07 10:44:06 +00:00
|
|
|
}
|
|
|
|
|
2013-01-29 17:31:33 +00:00
|
|
|
private:
|
2013-01-07 10:44:06 +00:00
|
|
|
/// The unroll factor. Each entry in the map stores this number of vector
|
|
|
|
/// elements.
|
|
|
|
unsigned UF;
|
|
|
|
|
|
|
|
/// Map storage. We use std::map and not DenseMap because insertions to a
|
|
|
|
/// dense map invalidates its iterators.
|
2013-01-29 17:31:33 +00:00
|
|
|
std::map<Value *, VectorParts> MapStorage;
|
2013-01-07 10:44:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// The original loop.
|
|
|
|
Loop *OrigLoop;
|
|
|
|
/// Scev analysis to use.
|
|
|
|
ScalarEvolution *SE;
|
|
|
|
/// Loop Info.
|
|
|
|
LoopInfo *LI;
|
|
|
|
/// Dominator Tree.
|
|
|
|
DominatorTree *DT;
|
|
|
|
/// Data Layout.
|
|
|
|
DataLayout *DL;
|
2013-02-27 15:24:19 +00:00
|
|
|
/// Target Library Info.
|
|
|
|
const TargetLibraryInfo *TLI;
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// The vectorization SIMD factor to use. Each vector will have this many
|
|
|
|
/// vector elements.
|
|
|
|
unsigned VF;
|
2013-08-26 22:33:26 +00:00
|
|
|
|
|
|
|
protected:
|
2013-01-07 10:44:06 +00:00
|
|
|
/// The vectorization unroll factor to use. Each scalar is vectorized to this
|
|
|
|
/// many different vector instructions.
|
|
|
|
unsigned UF;
|
|
|
|
|
|
|
|
/// The builder that we use
|
|
|
|
IRBuilder<> Builder;
|
|
|
|
|
|
|
|
// --- Vectorization state ---
|
|
|
|
|
|
|
|
/// The vector-loop preheader.
|
|
|
|
BasicBlock *LoopVectorPreHeader;
|
|
|
|
/// The scalar-loop preheader.
|
|
|
|
BasicBlock *LoopScalarPreHeader;
|
|
|
|
/// Middle Block between the vector and the scalar.
|
|
|
|
BasicBlock *LoopMiddleBlock;
|
|
|
|
///The ExitBlock of the scalar loop.
|
|
|
|
BasicBlock *LoopExitBlock;
|
|
|
|
///The vector loop body.
|
|
|
|
BasicBlock *LoopVectorBody;
|
|
|
|
///The scalar loop body.
|
|
|
|
BasicBlock *LoopScalarBody;
|
2013-01-19 13:57:58 +00:00
|
|
|
/// A list of all bypass blocks. The first block is the entry of the loop.
|
|
|
|
SmallVector<BasicBlock *, 4> LoopBypassBlocks;
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// The new Induction variable which was added to the new block.
|
|
|
|
PHINode *Induction;
|
|
|
|
/// The induction variable of the old basic block.
|
|
|
|
PHINode *OldInduction;
|
2013-05-11 23:04:28 +00:00
|
|
|
/// Holds the extended (to the widest induction type) start index.
|
|
|
|
Value *ExtendedIdx;
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Maps scalars to widened vectors.
|
|
|
|
ValueMap WidenMap;
|
2013-06-27 20:31:06 +00:00
|
|
|
EdgeMaskCache MaskCache;
|
2013-01-07 10:44:06 +00:00
|
|
|
};
|
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
class InnerLoopUnroller : public InnerLoopVectorizer {
|
|
|
|
public:
|
|
|
|
InnerLoopUnroller(Loop *OrigLoop, ScalarEvolution *SE, LoopInfo *LI,
|
|
|
|
DominatorTree *DT, DataLayout *DL,
|
|
|
|
const TargetLibraryInfo *TLI, unsigned UnrollFactor) :
|
|
|
|
InnerLoopVectorizer(OrigLoop, SE, LI, DT, DL, TLI, 1, UnrollFactor) { }
|
|
|
|
|
|
|
|
private:
|
|
|
|
virtual void scalarizeInstruction(Instruction *Instr);
|
|
|
|
virtual void vectorizeMemoryInstruction(Instruction *Instr,
|
|
|
|
LoopVectorizationLegality *Legal);
|
|
|
|
virtual Value *getBroadcastInstrs(Value *V);
|
|
|
|
virtual Value *getConsecutiveVector(Value* Val, int StartIdx, bool Negate);
|
|
|
|
virtual Value *reverseVector(Value *Vec);
|
|
|
|
};
|
|
|
|
|
2013-06-28 00:38:54 +00:00
|
|
|
/// \brief Look for a meaningful debug location on the instruction or it's
|
|
|
|
/// operands.
|
|
|
|
static Instruction *getDebugLocFromInstOrOperands(Instruction *I) {
|
|
|
|
if (!I)
|
|
|
|
return I;
|
|
|
|
|
|
|
|
DebugLoc Empty;
|
|
|
|
if (I->getDebugLoc() != Empty)
|
|
|
|
return I;
|
|
|
|
|
|
|
|
for (User::op_iterator OI = I->op_begin(), OE = I->op_end(); OI != OE; ++OI) {
|
|
|
|
if (Instruction *OpInst = dyn_cast<Instruction>(*OI))
|
|
|
|
if (OpInst->getDebugLoc() != Empty)
|
|
|
|
return OpInst;
|
|
|
|
}
|
|
|
|
|
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
2013-06-28 16:26:54 +00:00
|
|
|
/// \brief Set the debug location in the builder using the debug location in the
|
|
|
|
/// instruction.
|
2013-06-28 17:14:48 +00:00
|
|
|
static void setDebugLocFromInst(IRBuilder<> &B, const Value *Ptr) {
|
|
|
|
if (const Instruction *Inst = dyn_cast_or_null<Instruction>(Ptr))
|
2013-06-28 16:26:54 +00:00
|
|
|
B.SetCurrentDebugLocation(Inst->getDebugLoc());
|
|
|
|
else
|
|
|
|
B.SetCurrentDebugLocation(DebugLoc());
|
|
|
|
}
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// LoopVectorizationLegality checks if it is legal to vectorize a loop, and
|
|
|
|
/// to what vectorization factor.
|
|
|
|
/// This class does not look at the profitability of vectorization, only the
|
|
|
|
/// legality. This class has two main kinds of checks:
|
|
|
|
/// * Memory checks - The code in canVectorizeMemory checks if vectorization
|
|
|
|
/// will change the order of memory accesses in a way that will change the
|
|
|
|
/// correctness of the program.
|
|
|
|
/// * Scalars checks - The code in canVectorizeInstrs and canVectorizeMemory
|
|
|
|
/// checks for a number of different conditions, such as the availability of a
|
|
|
|
/// single induction variable, that all types are supported and vectorize-able,
|
|
|
|
/// etc. This code reflects the capabilities of InnerLoopVectorizer.
|
|
|
|
/// This class is also used by InnerLoopVectorizer for identifying
|
|
|
|
/// induction variable and the different reduction variables.
|
|
|
|
class LoopVectorizationLegality {
|
|
|
|
public:
|
|
|
|
LoopVectorizationLegality(Loop *L, ScalarEvolution *SE, DataLayout *DL,
|
2013-06-24 12:09:15 +00:00
|
|
|
DominatorTree *DT, TargetLibraryInfo *TLI)
|
|
|
|
: TheLoop(L), SE(SE), DL(DL), DT(DT), TLI(TLI),
|
2013-05-15 01:44:30 +00:00
|
|
|
Induction(0), WidestIndTy(0), HasFunNoNaNAttr(false),
|
2013-06-28 20:46:27 +00:00
|
|
|
MaxSafeDepDistBytes(-1U) {}
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// This enum represents the kinds of reductions that we support.
|
|
|
|
enum ReductionKind {
|
2013-01-07 23:13:00 +00:00
|
|
|
RK_NoReduction, ///< Not a reduction.
|
|
|
|
RK_IntegerAdd, ///< Sum of integers.
|
|
|
|
RK_IntegerMult, ///< Product of integers.
|
|
|
|
RK_IntegerOr, ///< Bitwise or logical OR of numbers.
|
|
|
|
RK_IntegerAnd, ///< Bitwise or logical AND of numbers.
|
|
|
|
RK_IntegerXor, ///< Bitwise or logical XOR of numbers.
|
2013-04-18 20:13:04 +00:00
|
|
|
RK_IntegerMinMax, ///< Min/max implemented in terms of select(cmp()).
|
2013-01-07 23:13:00 +00:00
|
|
|
RK_FloatAdd, ///< Sum of floats.
|
2013-05-05 01:54:48 +00:00
|
|
|
RK_FloatMult, ///< Product of floats.
|
|
|
|
RK_FloatMinMax ///< Min/max implemented in terms of select(cmp()).
|
2013-01-07 10:44:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// This enum represents the kinds of inductions that we support.
|
|
|
|
enum InductionKind {
|
2013-01-08 17:23:17 +00:00
|
|
|
IK_NoInduction, ///< Not an induction variable.
|
|
|
|
IK_IntInduction, ///< Integer induction variable. Step = 1.
|
|
|
|
IK_ReverseIntInduction, ///< Reverse int induction variable. Step = -1.
|
2013-01-23 01:35:00 +00:00
|
|
|
IK_PtrInduction, ///< Pointer induction var. Step = sizeof(elem).
|
|
|
|
IK_ReversePtrInduction ///< Reverse ptr indvar. Step = - sizeof(elem).
|
2013-01-07 10:44:06 +00:00
|
|
|
};
|
|
|
|
|
2013-04-19 21:03:36 +00:00
|
|
|
// This enum represents the kind of minmax reduction.
|
|
|
|
enum MinMaxReductionKind {
|
|
|
|
MRK_Invalid,
|
|
|
|
MRK_UIntMin,
|
|
|
|
MRK_UIntMax,
|
|
|
|
MRK_SIntMin,
|
2013-05-05 01:54:48 +00:00
|
|
|
MRK_SIntMax,
|
|
|
|
MRK_FloatMin,
|
|
|
|
MRK_FloatMax
|
2013-04-19 21:03:36 +00:00
|
|
|
};
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// This POD struct holds information about reduction variables.
|
|
|
|
struct ReductionDescriptor {
|
2013-01-07 23:13:00 +00:00
|
|
|
ReductionDescriptor() : StartValue(0), LoopExitInstr(0),
|
2013-04-19 21:03:36 +00:00
|
|
|
Kind(RK_NoReduction), MinMaxKind(MRK_Invalid) {}
|
2013-01-07 10:44:06 +00:00
|
|
|
|
2013-04-18 17:22:34 +00:00
|
|
|
ReductionDescriptor(Value *Start, Instruction *Exit, ReductionKind K,
|
2013-04-19 21:03:36 +00:00
|
|
|
MinMaxReductionKind MK)
|
|
|
|
: StartValue(Start), LoopExitInstr(Exit), Kind(K), MinMaxKind(MK) {}
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
// The starting value of the reduction.
|
|
|
|
// It does not have to be zero!
|
2013-05-22 16:54:56 +00:00
|
|
|
TrackingVH<Value> StartValue;
|
2013-01-07 10:44:06 +00:00
|
|
|
// The instruction who's value is used outside the loop.
|
|
|
|
Instruction *LoopExitInstr;
|
|
|
|
// The kind of the reduction.
|
|
|
|
ReductionKind Kind;
|
2013-04-18 17:22:34 +00:00
|
|
|
// If this a min/max reduction the kind of reduction.
|
2013-04-19 21:03:36 +00:00
|
|
|
MinMaxReductionKind MinMaxKind;
|
2013-04-18 17:22:34 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// This POD struct holds information about a potential reduction operation.
|
|
|
|
struct ReductionInstDesc {
|
|
|
|
ReductionInstDesc(bool IsRedux, Instruction *I) :
|
2013-04-19 21:03:36 +00:00
|
|
|
IsReduction(IsRedux), PatternLastInst(I), MinMaxKind(MRK_Invalid) {}
|
2013-04-18 17:22:34 +00:00
|
|
|
|
2013-04-19 21:03:36 +00:00
|
|
|
ReductionInstDesc(Instruction *I, MinMaxReductionKind K) :
|
|
|
|
IsReduction(true), PatternLastInst(I), MinMaxKind(K) {}
|
2013-04-18 17:22:34 +00:00
|
|
|
|
|
|
|
// Is this instruction a reduction candidate.
|
|
|
|
bool IsReduction;
|
|
|
|
// The last instruction in a min/max pattern (select of the select(icmp())
|
|
|
|
// pattern), or the current reduction instruction otherwise.
|
|
|
|
Instruction *PatternLastInst;
|
|
|
|
// If this is a min/max pattern the comparison predicate.
|
2013-04-19 21:03:36 +00:00
|
|
|
MinMaxReductionKind MinMaxKind;
|
2013-01-07 10:44:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// This POD struct holds information about the memory runtime legality
|
|
|
|
// check that a group of pointers do not overlap.
|
|
|
|
struct RuntimePointerCheck {
|
|
|
|
RuntimePointerCheck() : Need(false) {}
|
|
|
|
|
|
|
|
/// Reset the state of the pointer runtime information.
|
|
|
|
void reset() {
|
|
|
|
Need = false;
|
|
|
|
Pointers.clear();
|
|
|
|
Starts.clear();
|
|
|
|
Ends.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Insert a pointer and calculate the start and end SCEVs.
|
2013-06-24 12:09:15 +00:00
|
|
|
void insert(ScalarEvolution *SE, Loop *Lp, Value *Ptr, bool WritePtr,
|
|
|
|
unsigned DepSetId);
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// This flag indicates if we need to add the runtime check.
|
|
|
|
bool Need;
|
|
|
|
/// Holds the pointers that we need to check.
|
2013-05-22 16:54:56 +00:00
|
|
|
SmallVector<TrackingVH<Value>, 2> Pointers;
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Holds the pointer value at the beginning of the loop.
|
|
|
|
SmallVector<const SCEV*, 2> Starts;
|
|
|
|
/// Holds the pointer value at the end of the loop.
|
|
|
|
SmallVector<const SCEV*, 2> Ends;
|
2013-04-25 19:55:03 +00:00
|
|
|
/// Holds the information if this pointer is used for writing to memory.
|
|
|
|
SmallVector<bool, 2> IsWritePtr;
|
2013-06-24 12:09:15 +00:00
|
|
|
/// Holds the id of the set of pointers that could be dependent because of a
|
|
|
|
/// shared underlying object.
|
|
|
|
SmallVector<unsigned, 2> DependencySetId;
|
2013-01-07 10:44:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// A POD for saving information about induction variables.
|
|
|
|
struct InductionInfo {
|
|
|
|
InductionInfo(Value *Start, InductionKind K) : StartValue(Start), IK(K) {}
|
2013-01-08 17:23:17 +00:00
|
|
|
InductionInfo() : StartValue(0), IK(IK_NoInduction) {}
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Start value.
|
2013-05-22 16:54:56 +00:00
|
|
|
TrackingVH<Value> StartValue;
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Induction kind.
|
|
|
|
InductionKind IK;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// ReductionList contains the reduction descriptors for all
|
|
|
|
/// of the reductions that were found in the loop.
|
|
|
|
typedef DenseMap<PHINode*, ReductionDescriptor> ReductionList;
|
|
|
|
|
|
|
|
/// InductionList saves induction variables and maps them to the
|
|
|
|
/// induction descriptor.
|
|
|
|
typedef MapVector<PHINode*, InductionInfo> InductionList;
|
|
|
|
|
|
|
|
/// Returns true if it is legal to vectorize this loop.
|
|
|
|
/// This does not mean that it is profitable to vectorize this
|
|
|
|
/// loop, only that it is legal to do so.
|
|
|
|
bool canVectorize();
|
|
|
|
|
|
|
|
/// Returns the Induction variable.
|
|
|
|
PHINode *getInduction() { return Induction; }
|
|
|
|
|
|
|
|
/// Returns the reduction variables found in the loop.
|
|
|
|
ReductionList *getReductionVars() { return &Reductions; }
|
|
|
|
|
|
|
|
/// Returns the induction variables found in the loop.
|
|
|
|
InductionList *getInductionVars() { return &Inductions; }
|
|
|
|
|
2013-05-11 23:04:28 +00:00
|
|
|
/// Returns the widest induction type.
|
|
|
|
Type *getWidestInductionType() { return WidestIndTy; }
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Returns True if V is an induction variable in this loop.
|
|
|
|
bool isInductionVariable(const Value *V);
|
|
|
|
|
|
|
|
/// Return true if the block BB needs to be predicated in order for the loop
|
|
|
|
/// to be vectorized.
|
|
|
|
bool blockNeedsPredication(BasicBlock *BB);
|
|
|
|
|
|
|
|
/// Check if this pointer is consecutive when vectorizing. This happens
|
|
|
|
/// when the last index of the GEP is the induction variable, or that the
|
|
|
|
/// pointer itself is an induction variable.
|
|
|
|
/// This check allows us to vectorize A[idx] into a wide load/store.
|
|
|
|
/// Returns:
|
|
|
|
/// 0 - Stride is unknown or non consecutive.
|
|
|
|
/// 1 - Address is consecutive.
|
|
|
|
/// -1 - Address is consecutive, and decreasing.
|
|
|
|
int isConsecutivePtr(Value *Ptr);
|
|
|
|
|
|
|
|
/// Returns true if the value V is uniform within the loop.
|
|
|
|
bool isUniform(Value *V);
|
|
|
|
|
|
|
|
/// Returns true if this instruction will remain scalar after vectorization.
|
|
|
|
bool isUniformAfterVectorization(Instruction* I) { return Uniforms.count(I); }
|
|
|
|
|
|
|
|
/// Returns the information that we collected about runtime memory check.
|
|
|
|
RuntimePointerCheck *getRuntimePointerCheck() { return &PtrRtCheck; }
|
2013-04-19 21:03:36 +00:00
|
|
|
|
|
|
|
/// This function returns the identity element (or neutral element) for
|
|
|
|
/// the operation K.
|
2013-05-05 01:54:42 +00:00
|
|
|
static Constant *getReductionIdentity(ReductionKind K, Type *Tp);
|
2013-06-24 12:09:15 +00:00
|
|
|
|
|
|
|
unsigned getMaxSafeDepDistBytes() { return MaxSafeDepDistBytes; }
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
private:
|
|
|
|
/// Check if a single basic block loop is vectorizable.
|
|
|
|
/// At this point we know that this is a loop with a constant trip count
|
|
|
|
/// and we only need to check individual instructions.
|
|
|
|
bool canVectorizeInstrs();
|
|
|
|
|
|
|
|
/// When we vectorize loops we may change the order in which
|
|
|
|
/// we read and write from memory. This method checks if it is
|
|
|
|
/// legal to vectorize the code, considering only memory constrains.
|
|
|
|
/// Returns true if the loop is vectorizable
|
|
|
|
bool canVectorizeMemory();
|
|
|
|
|
|
|
|
/// Return true if we can vectorize this loop using the IF-conversion
|
|
|
|
/// transformation.
|
|
|
|
bool canVectorizeWithIfConvert();
|
|
|
|
|
|
|
|
/// Collect the variables that need to stay uniform after vectorization.
|
|
|
|
void collectLoopUniforms();
|
|
|
|
|
|
|
|
/// Return true if all of the instructions in the block can be speculatively
|
2013-06-28 20:46:27 +00:00
|
|
|
/// executed. \p SafePtrs is a list of addresses that are known to be legal
|
|
|
|
/// and we know that we can read from them without segfault.
|
|
|
|
bool blockCanBePredicated(BasicBlock *BB, SmallPtrSet<Value *, 8>& SafePtrs);
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// Returns True, if 'Phi' is the kind of reduction variable for type
|
|
|
|
/// 'Kind'. If this is a reduction variable, it adds it to ReductionList.
|
|
|
|
bool AddReductionVar(PHINode *Phi, ReductionKind Kind);
|
2013-04-18 17:22:34 +00:00
|
|
|
/// Returns a struct describing if the instruction 'I' can be a reduction
|
|
|
|
/// variable of type 'Kind'. If the reduction is a min/max pattern of
|
|
|
|
/// select(icmp()) this function advances the instruction pointer 'I' from the
|
|
|
|
/// compare instruction to the select instruction and stores this pointer in
|
|
|
|
/// 'PatternLastInst' member of the returned struct.
|
|
|
|
ReductionInstDesc isReductionInstr(Instruction *I, ReductionKind Kind,
|
2013-04-19 21:03:36 +00:00
|
|
|
ReductionInstDesc &Desc);
|
|
|
|
/// Returns true if the instruction is a Select(ICmp(X, Y), X, Y) instruction
|
|
|
|
/// pattern corresponding to a min(X, Y) or max(X, Y).
|
|
|
|
static ReductionInstDesc isMinMaxSelectCmpPattern(Instruction *I,
|
|
|
|
ReductionInstDesc &Prev);
|
2013-01-07 10:44:06 +00:00
|
|
|
/// Returns the induction kind of Phi. This function may return NoInduction
|
|
|
|
/// if the PHI is not an induction variable.
|
|
|
|
InductionKind isInductionVariable(PHINode *Phi);
|
|
|
|
|
|
|
|
/// The loop that we evaluate.
|
|
|
|
Loop *TheLoop;
|
|
|
|
/// Scev analysis.
|
|
|
|
ScalarEvolution *SE;
|
|
|
|
/// DataLayout analysis.
|
|
|
|
DataLayout *DL;
|
2013-02-21 22:39:03 +00:00
|
|
|
/// Dominators.
|
2013-01-07 10:44:06 +00:00
|
|
|
DominatorTree *DT;
|
2013-02-27 15:24:19 +00:00
|
|
|
/// Target Library Info.
|
|
|
|
TargetLibraryInfo *TLI;
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
// --- vectorization state --- //
|
|
|
|
|
|
|
|
/// Holds the integer induction variable. This is the counter of the
|
|
|
|
/// loop.
|
|
|
|
PHINode *Induction;
|
|
|
|
/// Holds the reduction variables.
|
|
|
|
ReductionList Reductions;
|
|
|
|
/// Holds all of the induction variables that we found in the loop.
|
|
|
|
/// Notice that inductions don't need to start at zero and that induction
|
|
|
|
/// variables can be pointers.
|
|
|
|
InductionList Inductions;
|
2013-05-11 23:04:28 +00:00
|
|
|
/// Holds the widest induction type encountered.
|
|
|
|
Type *WidestIndTy;
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// Allowed outside users. This holds the reduction
|
|
|
|
/// vars which can be accessed from outside the loop.
|
|
|
|
SmallPtrSet<Value*, 4> AllowedExit;
|
|
|
|
/// This set holds the variables which are known to be uniform after
|
|
|
|
/// vectorization.
|
|
|
|
SmallPtrSet<Instruction*, 4> Uniforms;
|
|
|
|
/// We need to check that all of the pointers in this list are disjoint
|
|
|
|
/// at runtime.
|
|
|
|
RuntimePointerCheck PtrRtCheck;
|
2013-05-05 01:54:48 +00:00
|
|
|
/// Can we assume the absence of NaNs.
|
|
|
|
bool HasFunNoNaNAttr;
|
2013-05-15 01:44:30 +00:00
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
unsigned MaxSafeDepDistBytes;
|
2013-01-07 10:44:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// LoopVectorizationCostModel - estimates the expected speedups due to
|
|
|
|
/// vectorization.
|
|
|
|
/// In many cases vectorization is not profitable. This can happen because of
|
|
|
|
/// a number of reasons. In this class we mainly attempt to predict the
|
|
|
|
/// expected speedup/slowdowns due to the supported instruction set. We use the
|
|
|
|
/// TargetTransformInfo to query the different backends for the cost of
|
|
|
|
/// different operations.
|
|
|
|
class LoopVectorizationCostModel {
|
|
|
|
public:
|
|
|
|
LoopVectorizationCostModel(Loop *L, ScalarEvolution *SE, LoopInfo *LI,
|
|
|
|
LoopVectorizationLegality *Legal,
|
2013-02-05 15:08:02 +00:00
|
|
|
const TargetTransformInfo &TTI,
|
2013-02-27 15:24:19 +00:00
|
|
|
DataLayout *DL, const TargetLibraryInfo *TLI)
|
|
|
|
: TheLoop(L), SE(SE), LI(LI), Legal(Legal), TTI(TTI), DL(DL), TLI(TLI) {}
|
2013-01-07 10:44:06 +00:00
|
|
|
|
2013-01-28 16:02:45 +00:00
|
|
|
/// Information about vectorization costs
|
|
|
|
struct VectorizationFactor {
|
|
|
|
unsigned Width; // Vector width with best cost
|
|
|
|
unsigned Cost; // Cost of the loop with that width
|
|
|
|
};
|
2013-01-20 05:24:29 +00:00
|
|
|
/// \return The most profitable vectorization factor and the cost of that VF.
|
2013-01-07 10:44:06 +00:00
|
|
|
/// This method checks every power of two up to VF. If UserVF is not ZERO
|
|
|
|
/// then this vectorization factor will be selected if vectorization is
|
|
|
|
/// possible.
|
2013-02-07 22:34:07 +00:00
|
|
|
VectorizationFactor selectVectorizationFactor(bool OptForSize,
|
|
|
|
unsigned UserVF);
|
2013-01-07 10:44:06 +00:00
|
|
|
|
2013-01-28 16:02:45 +00:00
|
|
|
/// \return The size (in bits) of the widest type in the code that
|
2013-01-09 22:29:00 +00:00
|
|
|
/// needs to be vectorized. We ignore values that remain scalar such as
|
|
|
|
/// 64 bit loop indices.
|
|
|
|
unsigned getWidestType();
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// \return The most profitable unroll factor.
|
|
|
|
/// If UserUF is non-zero then this method finds the best unroll-factor
|
|
|
|
/// based on register pressure and other parameters.
|
2013-01-20 05:24:29 +00:00
|
|
|
/// VF and LoopCost are the selected vectorization factor and the cost of the
|
|
|
|
/// selected VF.
|
|
|
|
unsigned selectUnrollFactor(bool OptForSize, unsigned UserUF, unsigned VF,
|
|
|
|
unsigned LoopCost);
|
2013-01-07 10:44:06 +00:00
|
|
|
|
|
|
|
/// \brief A struct that represents some properties of the register usage
|
|
|
|
/// of a loop.
|
|
|
|
struct RegisterUsage {
|
|
|
|
/// Holds the number of loop invariant values that are used in the loop.
|
|
|
|
unsigned LoopInvariantRegs;
|
|
|
|
/// Holds the maximum number of concurrent live intervals in the loop.
|
|
|
|
unsigned MaxLocalUsers;
|
|
|
|
/// Holds the number of instructions in the loop.
|
|
|
|
unsigned NumInstructions;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// \return information about the register usage of the loop.
|
|
|
|
RegisterUsage calculateRegisterUsage();
|
|
|
|
|
|
|
|
private:
|
|
|
|
/// Returns the expected execution cost. The unit of the cost does
|
|
|
|
/// not matter because we use the 'cost' units to compare different
|
|
|
|
/// vector widths. The cost that is returned is *not* normalized by
|
|
|
|
/// the factor width.
|
|
|
|
unsigned expectedCost(unsigned VF);
|
|
|
|
|
|
|
|
/// Returns the execution time cost of an instruction for a given vector
|
|
|
|
/// width. Vector width of one means scalar.
|
|
|
|
unsigned getInstructionCost(Instruction *I, unsigned VF);
|
|
|
|
|
2013-02-07 19:05:21 +00:00
|
|
|
/// A helper function for converting Scalar types to vector types.
|
|
|
|
/// If the incoming type is void, we return void. If the VF is 1, we return
|
|
|
|
/// the scalar type.
|
|
|
|
static Type* ToVectorTy(Type *Scalar, unsigned VF);
|
|
|
|
|
2013-02-05 15:08:02 +00:00
|
|
|
/// Returns whether the instruction is a load or store and will be a emitted
|
|
|
|
/// as a vector operation.
|
|
|
|
bool isConsecutiveLoadOrStore(Instruction *I);
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
/// The loop that we evaluate.
|
|
|
|
Loop *TheLoop;
|
|
|
|
/// Scev analysis.
|
|
|
|
ScalarEvolution *SE;
|
|
|
|
/// Loop Info analysis.
|
|
|
|
LoopInfo *LI;
|
|
|
|
/// Vectorization legality.
|
|
|
|
LoopVectorizationLegality *Legal;
|
|
|
|
/// Vector target information.
|
2013-01-07 11:12:29 +00:00
|
|
|
const TargetTransformInfo &TTI;
|
2013-02-05 15:08:02 +00:00
|
|
|
/// Target data layout information.
|
|
|
|
DataLayout *DL;
|
2013-02-27 15:24:19 +00:00
|
|
|
/// Target Library Info.
|
|
|
|
const TargetLibraryInfo *TLI;
|
2013-01-07 10:44:06 +00:00
|
|
|
};
|
|
|
|
|
2013-05-28 20:00:34 +00:00
|
|
|
/// Utility class for getting and setting loop vectorizer hints in the form
|
|
|
|
/// of loop metadata.
|
|
|
|
struct LoopVectorizeHints {
|
|
|
|
/// Vectorization width.
|
|
|
|
unsigned Width;
|
|
|
|
/// Vectorization unroll factor.
|
|
|
|
unsigned Unroll;
|
|
|
|
|
2013-08-28 18:33:10 +00:00
|
|
|
LoopVectorizeHints(const Loop *L, bool DisableUnrolling)
|
2013-05-28 20:00:34 +00:00
|
|
|
: Width(VectorizationFactor)
|
2013-08-28 18:33:10 +00:00
|
|
|
, Unroll(DisableUnrolling ? 1 : VectorizationUnroll)
|
2013-05-28 20:00:34 +00:00
|
|
|
, LoopID(L->getLoopID()) {
|
|
|
|
getHints(L);
|
|
|
|
// The command line options override any loop metadata except for when
|
|
|
|
// width == 1 which is used to indicate the loop is already vectorized.
|
|
|
|
if (VectorizationFactor.getNumOccurrences() > 0 && Width != 1)
|
|
|
|
Width = VectorizationFactor;
|
|
|
|
if (VectorizationUnroll.getNumOccurrences() > 0)
|
|
|
|
Unroll = VectorizationUnroll;
|
2013-08-28 18:33:10 +00:00
|
|
|
|
|
|
|
DEBUG(if (DisableUnrolling && Unroll == 1)
|
|
|
|
dbgs() << "LV: Unrolling disabled by the pass manager\n");
|
2013-05-28 20:00:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the loop vectorizer metadata prefix.
|
|
|
|
static StringRef Prefix() { return "llvm.vectorizer."; }
|
|
|
|
|
|
|
|
MDNode *createHint(LLVMContext &Context, StringRef Name, unsigned V) {
|
|
|
|
SmallVector<Value*, 2> Vals;
|
|
|
|
Vals.push_back(MDString::get(Context, Name));
|
|
|
|
Vals.push_back(ConstantInt::get(Type::getInt32Ty(Context), V));
|
|
|
|
return MDNode::get(Context, Vals);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Mark the loop L as already vectorized by setting the width to 1.
|
|
|
|
void setAlreadyVectorized(Loop *L) {
|
|
|
|
LLVMContext &Context = L->getHeader()->getContext();
|
|
|
|
|
|
|
|
Width = 1;
|
|
|
|
|
|
|
|
// Create a new loop id with one more operand for the already_vectorized
|
|
|
|
// hint. If the loop already has a loop id then copy the existing operands.
|
|
|
|
SmallVector<Value*, 4> Vals(1);
|
|
|
|
if (LoopID)
|
|
|
|
for (unsigned i = 1, ie = LoopID->getNumOperands(); i < ie; ++i)
|
|
|
|
Vals.push_back(LoopID->getOperand(i));
|
|
|
|
|
2013-05-29 03:13:47 +00:00
|
|
|
Vals.push_back(createHint(Context, Twine(Prefix(), "width").str(), Width));
|
2013-05-28 20:00:34 +00:00
|
|
|
|
|
|
|
MDNode *NewLoopID = MDNode::get(Context, Vals);
|
|
|
|
// Set operand 0 to refer to the loop id itself.
|
|
|
|
NewLoopID->replaceOperandWith(0, NewLoopID);
|
|
|
|
|
|
|
|
L->setLoopID(NewLoopID);
|
|
|
|
if (LoopID)
|
|
|
|
LoopID->replaceAllUsesWith(NewLoopID);
|
|
|
|
|
|
|
|
LoopID = NewLoopID;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
MDNode *LoopID;
|
|
|
|
|
|
|
|
/// Find hints specified in the loop metadata.
|
|
|
|
void getHints(const Loop *L) {
|
|
|
|
if (!LoopID)
|
|
|
|
return;
|
2013-05-29 03:13:41 +00:00
|
|
|
|
2013-05-28 20:00:34 +00:00
|
|
|
// First operand should refer to the loop id itself.
|
|
|
|
assert(LoopID->getNumOperands() > 0 && "requires at least one operand");
|
|
|
|
assert(LoopID->getOperand(0) == LoopID && "invalid loop id");
|
2013-05-29 03:13:41 +00:00
|
|
|
|
2013-05-28 20:00:34 +00:00
|
|
|
for (unsigned i = 1, ie = LoopID->getNumOperands(); i < ie; ++i) {
|
|
|
|
const MDString *S = 0;
|
|
|
|
SmallVector<Value*, 4> Args;
|
|
|
|
|
|
|
|
// The expected hint is either a MDString or a MDNode with the first
|
|
|
|
// operand a MDString.
|
|
|
|
if (const MDNode *MD = dyn_cast<MDNode>(LoopID->getOperand(i))) {
|
|
|
|
if (!MD || MD->getNumOperands() == 0)
|
|
|
|
continue;
|
|
|
|
S = dyn_cast<MDString>(MD->getOperand(0));
|
|
|
|
for (unsigned i = 1, ie = MD->getNumOperands(); i < ie; ++i)
|
|
|
|
Args.push_back(MD->getOperand(i));
|
|
|
|
} else {
|
|
|
|
S = dyn_cast<MDString>(LoopID->getOperand(i));
|
|
|
|
assert(Args.size() == 0 && "too many arguments for MDString");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!S)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Check if the hint starts with the vectorizer prefix.
|
|
|
|
StringRef Hint = S->getString();
|
|
|
|
if (!Hint.startswith(Prefix()))
|
|
|
|
continue;
|
|
|
|
// Remove the prefix.
|
|
|
|
Hint = Hint.substr(Prefix().size(), StringRef::npos);
|
2013-05-29 03:13:41 +00:00
|
|
|
|
2013-05-28 20:00:34 +00:00
|
|
|
if (Args.size() == 1)
|
|
|
|
getHint(Hint, Args[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check string hint with one operand.
|
|
|
|
void getHint(StringRef Hint, Value *Arg) {
|
|
|
|
const ConstantInt *C = dyn_cast<ConstantInt>(Arg);
|
|
|
|
if (!C) return;
|
|
|
|
unsigned Val = C->getZExtValue();
|
|
|
|
|
|
|
|
if (Hint == "width") {
|
2013-09-10 23:45:25 +00:00
|
|
|
if (isPowerOf2_32(Val) && Val <= MaxVectorWidth)
|
|
|
|
Width = Val;
|
|
|
|
else
|
|
|
|
DEBUG(dbgs() << "LV: ignoring invalid width hint metadata");
|
2013-05-28 20:00:34 +00:00
|
|
|
} else if (Hint == "unroll") {
|
2013-09-10 23:45:25 +00:00
|
|
|
if (isPowerOf2_32(Val) && Val <= MaxUnrollFactor)
|
|
|
|
Unroll = Val;
|
|
|
|
else
|
|
|
|
DEBUG(dbgs() << "LV: ignoring invalid unroll hint metadata");
|
|
|
|
} else {
|
2013-05-28 20:00:34 +00:00
|
|
|
DEBUG(dbgs() << "LV: ignoring unknown hint " << Hint);
|
2013-09-10 23:45:25 +00:00
|
|
|
}
|
2013-05-28 20:00:34 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-12-10 21:39:02 +00:00
|
|
|
/// The LoopVectorize Pass.
|
2012-10-17 18:25:06 +00:00
|
|
|
struct LoopVectorize : public LoopPass {
|
2012-12-12 01:11:46 +00:00
|
|
|
/// Pass identification, replacement for typeid
|
|
|
|
static char ID;
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-08-28 18:33:10 +00:00
|
|
|
explicit LoopVectorize(bool NoUnrolling = false)
|
|
|
|
: LoopPass(ID), DisableUnrolling(NoUnrolling) {
|
2012-10-17 18:25:06 +00:00
|
|
|
initializeLoopVectorizePass(*PassRegistry::getPassRegistry());
|
|
|
|
}
|
|
|
|
|
|
|
|
ScalarEvolution *SE;
|
|
|
|
DataLayout *DL;
|
|
|
|
LoopInfo *LI;
|
2012-10-24 20:36:32 +00:00
|
|
|
TargetTransformInfo *TTI;
|
2012-10-29 21:52:38 +00:00
|
|
|
DominatorTree *DT;
|
2013-02-27 15:24:19 +00:00
|
|
|
TargetLibraryInfo *TLI;
|
2013-08-28 18:33:10 +00:00
|
|
|
bool DisableUnrolling;
|
2012-10-17 18:25:06 +00:00
|
|
|
|
|
|
|
virtual bool runOnLoop(Loop *L, LPPassManager &LPM) {
|
2012-10-22 03:52:53 +00:00
|
|
|
// We only vectorize innermost loops.
|
2012-10-17 18:25:06 +00:00
|
|
|
if (!L->empty())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
SE = &getAnalysis<ScalarEvolution>();
|
|
|
|
DL = getAnalysisIfAvailable<DataLayout>();
|
|
|
|
LI = &getAnalysis<LoopInfo>();
|
2013-01-07 11:12:29 +00:00
|
|
|
TTI = &getAnalysis<TargetTransformInfo>();
|
2012-10-29 21:52:38 +00:00
|
|
|
DT = &getAnalysis<DominatorTree>();
|
2013-02-27 15:24:19 +00:00
|
|
|
TLI = getAnalysisIfAvailable<TargetLibraryInfo>();
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-09-18 12:43:35 +00:00
|
|
|
// If the target claims to have no vector registers don't attempt
|
|
|
|
// vectorization.
|
|
|
|
if (!TTI->getNumberOfRegisters(true))
|
|
|
|
return false;
|
|
|
|
|
2013-04-24 16:15:58 +00:00
|
|
|
if (DL == NULL) {
|
|
|
|
DEBUG(dbgs() << "LV: Not vectorizing because of missing data layout");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
DEBUG(dbgs() << "LV: Checking a loop in \"" <<
|
2012-10-17 23:40:15 +00:00
|
|
|
L->getHeader()->getParent()->getName() << "\"\n");
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-08-28 18:33:10 +00:00
|
|
|
LoopVectorizeHints Hints(L, DisableUnrolling);
|
2013-05-28 20:00:34 +00:00
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
if (Hints.Width == 1 && Hints.Unroll == 1) {
|
2013-05-28 20:00:34 +00:00
|
|
|
DEBUG(dbgs() << "LV: Not vectorizing.\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
// Check if it is legal to vectorize the loop.
|
2013-06-24 12:09:15 +00:00
|
|
|
LoopVectorizationLegality LVL(L, SE, DL, DT, TLI);
|
2012-10-24 20:36:32 +00:00
|
|
|
if (!LVL.canVectorize()) {
|
|
|
|
DEBUG(dbgs() << "LV: Not vectorizing.\n");
|
2012-10-17 18:25:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-12-12 01:11:46 +00:00
|
|
|
// Use the cost model.
|
2013-02-27 15:24:19 +00:00
|
|
|
LoopVectorizationCostModel CM(L, SE, LI, &LVL, *TTI, DL, TLI);
|
2012-12-12 19:29:45 +00:00
|
|
|
|
2013-02-08 17:43:32 +00:00
|
|
|
// Check the function attributes to find out if this function should be
|
2012-12-12 19:29:45 +00:00
|
|
|
// optimized for size.
|
|
|
|
Function *F = L->getHeader()->getParent();
|
2013-01-02 23:54:43 +00:00
|
|
|
Attribute::AttrKind SzAttr = Attribute::OptimizeForSize;
|
|
|
|
Attribute::AttrKind FlAttr = Attribute::NoImplicitFloat;
|
|
|
|
unsigned FnIndex = AttributeSet::FunctionIndex;
|
|
|
|
bool OptForSize = F->getAttributes().hasAttribute(FnIndex, SzAttr);
|
|
|
|
bool NoFloat = F->getAttributes().hasAttribute(FnIndex, FlAttr);
|
|
|
|
|
|
|
|
if (NoFloat) {
|
|
|
|
DEBUG(dbgs() << "LV: Can't vectorize when the NoImplicitFloat"
|
|
|
|
"attribute is used.\n");
|
|
|
|
return false;
|
|
|
|
}
|
2012-12-12 19:29:45 +00:00
|
|
|
|
2013-01-20 05:24:29 +00:00
|
|
|
// Select the optimal vectorization factor.
|
2013-01-28 16:02:45 +00:00
|
|
|
LoopVectorizationCostModel::VectorizationFactor VF;
|
2013-05-28 20:00:34 +00:00
|
|
|
VF = CM.selectVectorizationFactor(OptForSize, Hints.Width);
|
2013-01-20 05:24:29 +00:00
|
|
|
// Select the unroll factor.
|
2013-05-28 20:00:34 +00:00
|
|
|
unsigned UF = CM.selectUnrollFactor(OptForSize, Hints.Unroll, VF.Width,
|
|
|
|
VF.Cost);
|
2012-12-12 01:11:46 +00:00
|
|
|
|
2013-01-28 16:02:45 +00:00
|
|
|
if (VF.Width == 1) {
|
2012-12-12 01:11:46 +00:00
|
|
|
DEBUG(dbgs() << "LV: Vectorization is possible but not beneficial.\n");
|
2012-10-24 20:36:32 +00:00
|
|
|
}
|
|
|
|
|
2013-01-28 16:02:45 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a vectorizable loop ("<< VF.Width << ") in "<<
|
2012-12-13 00:21:03 +00:00
|
|
|
F->getParent()->getModuleIdentifier()<<"\n");
|
2013-01-04 17:48:25 +00:00
|
|
|
DEBUG(dbgs() << "LV: Unroll Factor is " << UF << "\n");
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
if (VF.Width == 1) {
|
|
|
|
if (UF == 1)
|
|
|
|
return false;
|
|
|
|
// We decided not to vectorize, but we may want to unroll.
|
|
|
|
InnerLoopUnroller Unroller(L, SE, LI, DT, DL, TLI, UF);
|
|
|
|
Unroller.vectorize(&LVL);
|
|
|
|
} else {
|
|
|
|
// If we decided that it is *legal* to vectorize the loop then do it.
|
|
|
|
InnerLoopVectorizer LB(L, SE, LI, DT, DL, TLI, VF.Width, UF);
|
|
|
|
LB.vectorize(&LVL);
|
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-05-28 20:00:34 +00:00
|
|
|
// Mark the loop as already vectorized to avoid vectorizing again.
|
|
|
|
Hints.setAlreadyVectorized(L);
|
|
|
|
|
2012-10-18 05:29:12 +00:00
|
|
|
DEBUG(verifyFunction(*L->getHeader()->getParent()));
|
2012-10-17 18:25:06 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
|
|
|
LoopPass::getAnalysisUsage(AU);
|
|
|
|
AU.addRequiredID(LoopSimplifyID);
|
2012-10-19 23:05:40 +00:00
|
|
|
AU.addRequiredID(LCSSAID);
|
2013-01-07 11:12:29 +00:00
|
|
|
AU.addRequired<DominatorTree>();
|
2012-10-17 18:25:06 +00:00
|
|
|
AU.addRequired<LoopInfo>();
|
|
|
|
AU.addRequired<ScalarEvolution>();
|
2013-01-07 11:12:29 +00:00
|
|
|
AU.addRequired<TargetTransformInfo>();
|
2012-10-29 21:52:38 +00:00
|
|
|
AU.addPreserved<LoopInfo>();
|
|
|
|
AU.addPreserved<DominatorTree>();
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2013-01-07 10:44:06 +00:00
|
|
|
} // end anonymous namespace
|
2012-12-10 21:39:02 +00:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Implementation of LoopVectorizationLegality, InnerLoopVectorizer and
|
|
|
|
// LoopVectorizationCostModel.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
void
|
|
|
|
LoopVectorizationLegality::RuntimePointerCheck::insert(ScalarEvolution *SE,
|
2013-04-25 19:55:03 +00:00
|
|
|
Loop *Lp, Value *Ptr,
|
2013-06-24 12:09:15 +00:00
|
|
|
bool WritePtr,
|
|
|
|
unsigned DepSetId) {
|
2012-12-10 21:39:02 +00:00
|
|
|
const SCEV *Sc = SE->getSCEV(Ptr);
|
|
|
|
const SCEVAddRecExpr *AR = dyn_cast<SCEVAddRecExpr>(Sc);
|
|
|
|
assert(AR && "Invalid addrec expression");
|
2013-05-31 21:48:56 +00:00
|
|
|
const SCEV *Ex = SE->getBackedgeTakenCount(Lp);
|
2012-12-10 21:39:02 +00:00
|
|
|
const SCEV *ScEnd = AR->evaluateAtIteration(Ex, *SE);
|
|
|
|
Pointers.push_back(Ptr);
|
|
|
|
Starts.push_back(AR->getStart());
|
|
|
|
Ends.push_back(ScEnd);
|
2013-04-25 19:55:03 +00:00
|
|
|
IsWritePtr.push_back(WritePtr);
|
2013-06-24 12:09:15 +00:00
|
|
|
DependencySetId.push_back(DepSetId);
|
2012-12-10 21:39:02 +00:00
|
|
|
}
|
|
|
|
|
2012-12-03 21:33:08 +00:00
|
|
|
Value *InnerLoopVectorizer::getBroadcastInstrs(Value *V) {
|
2012-11-29 19:25:41 +00:00
|
|
|
// Save the current insertion location.
|
|
|
|
Instruction *Loc = Builder.GetInsertPoint();
|
|
|
|
|
|
|
|
// We need to place the broadcast of invariant variables outside the loop.
|
2012-12-10 19:25:06 +00:00
|
|
|
Instruction *Instr = dyn_cast<Instruction>(V);
|
|
|
|
bool NewInstr = (Instr && Instr->getParent() == LoopVectorBody);
|
|
|
|
bool Invariant = OrigLoop->isLoopInvariant(V) && !NewInstr;
|
2012-11-29 19:25:41 +00:00
|
|
|
|
|
|
|
// Place the code for broadcasting invariant variables in the new preheader.
|
|
|
|
if (Invariant)
|
|
|
|
Builder.SetInsertPoint(LoopVectorPreHeader->getTerminator());
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
// Broadcast the scalar into all locations in the vector.
|
2013-01-01 19:55:16 +00:00
|
|
|
Value *Shuf = Builder.CreateVectorSplat(VF, V, "broadcast");
|
2012-11-29 19:25:41 +00:00
|
|
|
|
|
|
|
// Restore the builder insertion point.
|
|
|
|
if (Invariant)
|
|
|
|
Builder.SetInsertPoint(Loc);
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
return Shuf;
|
|
|
|
}
|
|
|
|
|
2013-05-07 04:37:05 +00:00
|
|
|
Value *InnerLoopVectorizer::getConsecutiveVector(Value* Val, int StartIdx,
|
2013-01-03 00:52:27 +00:00
|
|
|
bool Negate) {
|
2012-10-17 18:25:06 +00:00
|
|
|
assert(Val->getType()->isVectorTy() && "Must be a vector");
|
|
|
|
assert(Val->getType()->getScalarType()->isIntegerTy() &&
|
|
|
|
"Elem must be an integer");
|
|
|
|
// Create the types.
|
|
|
|
Type *ITy = Val->getType()->getScalarType();
|
|
|
|
VectorType *Ty = cast<VectorType>(Val->getType());
|
2012-12-10 19:25:06 +00:00
|
|
|
int VLen = Ty->getNumElements();
|
2012-10-17 18:25:06 +00:00
|
|
|
SmallVector<Constant*, 8> Indices;
|
|
|
|
|
|
|
|
// Create a vector of consecutive numbers from zero to VF.
|
2013-01-03 00:52:27 +00:00
|
|
|
for (int i = 0; i < VLen; ++i) {
|
2013-05-07 04:37:05 +00:00
|
|
|
int64_t Idx = Negate ? (-i) : i;
|
|
|
|
Indices.push_back(ConstantInt::get(ITy, StartIdx + Idx, Negate));
|
2013-01-03 00:52:27 +00:00
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
|
|
|
|
// Add the consecutive indices to the vector value.
|
|
|
|
Constant *Cv = ConstantVector::get(Indices);
|
|
|
|
assert(Cv->getType() == Val->getType() && "Invalid consecutive vec");
|
2012-10-19 23:27:19 +00:00
|
|
|
return Builder.CreateAdd(Val, Cv, "induction");
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-26 19:08:17 +00:00
|
|
|
int LoopVectorizationLegality::isConsecutivePtr(Value *Ptr) {
|
2012-11-25 08:41:35 +00:00
|
|
|
assert(Ptr->getType()->isPointerTy() && "Unexpected non ptr");
|
2013-01-23 01:35:00 +00:00
|
|
|
// Make sure that the pointer does not point to structs.
|
|
|
|
if (cast<PointerType>(Ptr->getType())->getElementType()->isAggregateType())
|
|
|
|
return 0;
|
2012-11-25 08:41:35 +00:00
|
|
|
|
2012-12-10 19:25:06 +00:00
|
|
|
// If this value is a pointer induction variable we know it is consecutive.
|
2012-11-25 08:41:35 +00:00
|
|
|
PHINode *Phi = dyn_cast_or_null<PHINode>(Ptr);
|
2012-12-10 19:25:06 +00:00
|
|
|
if (Phi && Inductions.count(Phi)) {
|
|
|
|
InductionInfo II = Inductions[Phi];
|
2013-01-08 17:23:17 +00:00
|
|
|
if (IK_PtrInduction == II.IK)
|
2012-12-26 19:08:17 +00:00
|
|
|
return 1;
|
2013-01-23 01:35:00 +00:00
|
|
|
else if (IK_ReversePtrInduction == II.IK)
|
|
|
|
return -1;
|
2012-12-10 19:25:06 +00:00
|
|
|
}
|
2012-11-25 08:41:35 +00:00
|
|
|
|
2012-10-22 18:27:56 +00:00
|
|
|
GetElementPtrInst *Gep = dyn_cast_or_null<GetElementPtrInst>(Ptr);
|
2012-10-17 18:25:06 +00:00
|
|
|
if (!Gep)
|
2012-12-26 19:08:17 +00:00
|
|
|
return 0;
|
2012-10-17 18:25:06 +00:00
|
|
|
|
|
|
|
unsigned NumOperands = Gep->getNumOperands();
|
|
|
|
Value *LastIndex = Gep->getOperand(NumOperands - 1);
|
|
|
|
|
2013-01-23 01:35:00 +00:00
|
|
|
Value *GpPtr = Gep->getPointerOperand();
|
|
|
|
// If this GEP value is a consecutive pointer induction variable and all of
|
|
|
|
// the indices are constant then we know it is consecutive. We can
|
|
|
|
Phi = dyn_cast<PHINode>(GpPtr);
|
|
|
|
if (Phi && Inductions.count(Phi)) {
|
|
|
|
|
|
|
|
// Make sure that the pointer does not point to structs.
|
|
|
|
PointerType *GepPtrType = cast<PointerType>(GpPtr->getType());
|
|
|
|
if (GepPtrType->getElementType()->isAggregateType())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// Make sure that all of the index operands are loop invariant.
|
|
|
|
for (unsigned i = 1; i < NumOperands; ++i)
|
|
|
|
if (!SE->isLoopInvariant(SE->getSCEV(Gep->getOperand(i)), TheLoop))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
InductionInfo II = Inductions[Phi];
|
|
|
|
if (IK_PtrInduction == II.IK)
|
|
|
|
return 1;
|
|
|
|
else if (IK_ReversePtrInduction == II.IK)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
// Check that all of the gep indices are uniform except for the last.
|
|
|
|
for (unsigned i = 0; i < NumOperands - 1; ++i)
|
2012-10-20 08:26:33 +00:00
|
|
|
if (!SE->isLoopInvariant(SE->getSCEV(Gep->getOperand(i)), TheLoop))
|
2012-12-26 19:08:17 +00:00
|
|
|
return 0;
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2012-11-29 19:25:41 +00:00
|
|
|
// We can emit wide load/stores only if the last index is the induction
|
2012-10-18 17:31:49 +00:00
|
|
|
// variable.
|
2012-10-17 18:25:06 +00:00
|
|
|
const SCEV *Last = SE->getSCEV(LastIndex);
|
|
|
|
if (const SCEVAddRecExpr *AR = dyn_cast<SCEVAddRecExpr>(Last)) {
|
|
|
|
const SCEV *Step = AR->getStepRecurrence(*SE);
|
|
|
|
|
|
|
|
// The memory is consecutive because the last index is consecutive
|
|
|
|
// and all other indices are loop invariant.
|
|
|
|
if (Step->isOne())
|
2012-12-26 19:08:17 +00:00
|
|
|
return 1;
|
|
|
|
if (Step->isAllOnesValue())
|
|
|
|
return -1;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-26 19:08:17 +00:00
|
|
|
return 0;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2012-11-09 07:09:44 +00:00
|
|
|
bool LoopVectorizationLegality::isUniform(Value *V) {
|
|
|
|
return (SE->isLoopInvariant(SE->getSCEV(V), TheLoop));
|
|
|
|
}
|
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
InnerLoopVectorizer::VectorParts&
|
|
|
|
InnerLoopVectorizer::getVectorValue(Value *V) {
|
2012-11-29 19:25:41 +00:00
|
|
|
assert(V != Induction && "The new induction variable should not be used.");
|
2012-10-19 23:05:40 +00:00
|
|
|
assert(!V->getType()->isVectorTy() && "Can't widen a vector");
|
2012-10-18 17:31:49 +00:00
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
// If we have this scalar in the map, return it.
|
|
|
|
if (WidenMap.has(V))
|
|
|
|
return WidenMap.get(V);
|
|
|
|
|
|
|
|
// If this scalar is unknown, assume that it is a constant or that it is
|
|
|
|
// loop invariant. Broadcast V and save the value for future uses.
|
2012-10-18 17:31:49 +00:00
|
|
|
Value *B = getBroadcastInstrs(V);
|
2013-01-29 17:31:33 +00:00
|
|
|
return WidenMap.splat(V, B);
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-26 19:08:17 +00:00
|
|
|
Value *InnerLoopVectorizer::reverseVector(Value *Vec) {
|
|
|
|
assert(Vec->getType()->isVectorTy() && "Invalid type");
|
|
|
|
SmallVector<Constant*, 8> ShuffleMask;
|
|
|
|
for (unsigned i = 0; i < VF; ++i)
|
|
|
|
ShuffleMask.push_back(Builder.getInt32(VF - i - 1));
|
|
|
|
|
|
|
|
return Builder.CreateShuffleVector(Vec, UndefValue::get(Vec->getType()),
|
|
|
|
ConstantVector::get(ShuffleMask),
|
|
|
|
"reverse");
|
|
|
|
}
|
|
|
|
|
2013-01-25 21:47:42 +00:00
|
|
|
|
|
|
|
void InnerLoopVectorizer::vectorizeMemoryInstruction(Instruction *Instr,
|
|
|
|
LoopVectorizationLegality *Legal) {
|
|
|
|
// Attempt to issue a wide load.
|
|
|
|
LoadInst *LI = dyn_cast<LoadInst>(Instr);
|
|
|
|
StoreInst *SI = dyn_cast<StoreInst>(Instr);
|
|
|
|
|
|
|
|
assert((LI || SI) && "Invalid Load/Store instruction");
|
|
|
|
|
|
|
|
Type *ScalarDataTy = LI ? LI->getType() : SI->getValueOperand()->getType();
|
|
|
|
Type *DataTy = VectorType::get(ScalarDataTy, VF);
|
|
|
|
Value *Ptr = LI ? LI->getPointerOperand() : SI->getPointerOperand();
|
|
|
|
unsigned Alignment = LI ? LI->getAlignment() : SI->getAlignment();
|
2013-06-17 18:49:06 +00:00
|
|
|
unsigned AddressSpace = Ptr->getType()->getPointerAddressSpace();
|
2013-04-24 16:16:01 +00:00
|
|
|
unsigned ScalarAllocatedSize = DL->getTypeAllocSize(ScalarDataTy);
|
|
|
|
unsigned VectorElementSize = DL->getTypeStoreSize(DataTy)/VF;
|
|
|
|
|
|
|
|
if (ScalarAllocatedSize != VectorElementSize)
|
|
|
|
return scalarizeInstruction(Instr);
|
|
|
|
|
2013-01-25 21:47:42 +00:00
|
|
|
// If the pointer is loop invariant or if it is non consecutive,
|
|
|
|
// scalarize the load.
|
2013-04-24 16:16:03 +00:00
|
|
|
int ConsecutiveStride = Legal->isConsecutivePtr(Ptr);
|
|
|
|
bool Reverse = ConsecutiveStride < 0;
|
2013-01-25 21:47:42 +00:00
|
|
|
bool UniformLoad = LI && Legal->isUniform(Ptr);
|
2013-04-24 16:16:03 +00:00
|
|
|
if (!ConsecutiveStride || UniformLoad)
|
2013-01-25 21:47:42 +00:00
|
|
|
return scalarizeInstruction(Instr);
|
|
|
|
|
|
|
|
Constant *Zero = Builder.getInt32(0);
|
|
|
|
VectorParts &Entry = WidenMap.get(Instr);
|
|
|
|
|
|
|
|
// Handle consecutive loads/stores.
|
|
|
|
GetElementPtrInst *Gep = dyn_cast<GetElementPtrInst>(Ptr);
|
|
|
|
if (Gep && Legal->isInductionVariable(Gep->getPointerOperand())) {
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, Gep);
|
2013-01-25 21:47:42 +00:00
|
|
|
Value *PtrOperand = Gep->getPointerOperand();
|
|
|
|
Value *FirstBasePtr = getVectorValue(PtrOperand)[0];
|
|
|
|
FirstBasePtr = Builder.CreateExtractElement(FirstBasePtr, Zero);
|
|
|
|
|
|
|
|
// Create the new GEP with the new induction variable.
|
|
|
|
GetElementPtrInst *Gep2 = cast<GetElementPtrInst>(Gep->clone());
|
|
|
|
Gep2->setOperand(0, FirstBasePtr);
|
|
|
|
Gep2->setName("gep.indvar.base");
|
|
|
|
Ptr = Builder.Insert(Gep2);
|
|
|
|
} else if (Gep) {
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, Gep);
|
2013-01-25 21:47:42 +00:00
|
|
|
assert(SE->isLoopInvariant(SE->getSCEV(Gep->getPointerOperand()),
|
|
|
|
OrigLoop) && "Base ptr must be invariant");
|
|
|
|
|
|
|
|
// The last index does not have to be the induction. It can be
|
|
|
|
// consecutive and be a function of the index. For example A[I+1];
|
|
|
|
unsigned NumOperands = Gep->getNumOperands();
|
2013-06-27 15:11:55 +00:00
|
|
|
unsigned LastOperand = NumOperands - 1;
|
2013-01-25 21:47:42 +00:00
|
|
|
// Create the new GEP with the new induction variable.
|
|
|
|
GetElementPtrInst *Gep2 = cast<GetElementPtrInst>(Gep->clone());
|
2013-06-27 15:11:55 +00:00
|
|
|
|
|
|
|
for (unsigned i = 0; i < NumOperands; ++i) {
|
|
|
|
Value *GepOperand = Gep->getOperand(i);
|
|
|
|
Instruction *GepOperandInst = dyn_cast<Instruction>(GepOperand);
|
|
|
|
|
|
|
|
// Update last index or loop invariant instruction anchored in loop.
|
|
|
|
if (i == LastOperand ||
|
|
|
|
(GepOperandInst && OrigLoop->contains(GepOperandInst))) {
|
|
|
|
assert((i == LastOperand ||
|
|
|
|
SE->isLoopInvariant(SE->getSCEV(GepOperandInst), OrigLoop)) &&
|
|
|
|
"Must be last index or loop invariant");
|
|
|
|
|
|
|
|
VectorParts &GEPParts = getVectorValue(GepOperand);
|
|
|
|
Value *Index = GEPParts[0];
|
|
|
|
Index = Builder.CreateExtractElement(Index, Zero);
|
|
|
|
Gep2->setOperand(i, Index);
|
|
|
|
Gep2->setName("gep.indvar.idx");
|
|
|
|
}
|
|
|
|
}
|
2013-01-25 21:47:42 +00:00
|
|
|
Ptr = Builder.Insert(Gep2);
|
|
|
|
} else {
|
|
|
|
// Use the induction element ptr.
|
|
|
|
assert(isa<PHINode>(Ptr) && "Invalid induction ptr");
|
2013-06-28 17:14:48 +00:00
|
|
|
setDebugLocFromInst(Builder, Ptr);
|
2013-01-25 21:47:42 +00:00
|
|
|
VectorParts &PtrVal = getVectorValue(Ptr);
|
|
|
|
Ptr = Builder.CreateExtractElement(PtrVal[0], Zero);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle Stores:
|
|
|
|
if (SI) {
|
|
|
|
assert(!Legal->isUniform(SI->getPointerOperand()) &&
|
|
|
|
"We do not allow storing to uniform addresses");
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, SI);
|
2013-06-27 00:45:41 +00:00
|
|
|
// We don't want to update the value in the map as it might be used in
|
|
|
|
// another expression. So don't use a reference type for "StoredVal".
|
|
|
|
VectorParts StoredVal = getVectorValue(SI->getValueOperand());
|
2013-01-25 21:47:42 +00:00
|
|
|
|
|
|
|
for (unsigned Part = 0; Part < UF; ++Part) {
|
|
|
|
// Calculate the pointer for the specific unroll-part.
|
|
|
|
Value *PartPtr = Builder.CreateGEP(Ptr, Builder.getInt32(Part * VF));
|
|
|
|
|
|
|
|
if (Reverse) {
|
|
|
|
// If we store to reverse consecutive memory locations then we need
|
|
|
|
// to reverse the order of elements in the stored value.
|
|
|
|
StoredVal[Part] = reverseVector(StoredVal[Part]);
|
|
|
|
// If the address is consecutive but reversed, then the
|
|
|
|
// wide store needs to start at the last vector element.
|
|
|
|
PartPtr = Builder.CreateGEP(Ptr, Builder.getInt32(-Part * VF));
|
|
|
|
PartPtr = Builder.CreateGEP(PartPtr, Builder.getInt32(1 - VF));
|
|
|
|
}
|
|
|
|
|
2013-07-31 22:17:45 +00:00
|
|
|
Value *VecPtr = Builder.CreateBitCast(PartPtr,
|
|
|
|
DataTy->getPointerTo(AddressSpace));
|
2013-01-25 21:47:42 +00:00
|
|
|
Builder.CreateStore(StoredVal[Part], VecPtr)->setAlignment(Alignment);
|
|
|
|
}
|
2013-06-27 00:45:41 +00:00
|
|
|
return;
|
2013-01-25 21:47:42 +00:00
|
|
|
}
|
|
|
|
|
2013-06-28 00:38:54 +00:00
|
|
|
// Handle loads.
|
|
|
|
assert(LI && "Must have a load instruction");
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, LI);
|
2013-01-25 21:47:42 +00:00
|
|
|
for (unsigned Part = 0; Part < UF; ++Part) {
|
|
|
|
// Calculate the pointer for the specific unroll-part.
|
|
|
|
Value *PartPtr = Builder.CreateGEP(Ptr, Builder.getInt32(Part * VF));
|
|
|
|
|
|
|
|
if (Reverse) {
|
|
|
|
// If the address is consecutive but reversed, then the
|
|
|
|
// wide store needs to start at the last vector element.
|
|
|
|
PartPtr = Builder.CreateGEP(Ptr, Builder.getInt32(-Part * VF));
|
|
|
|
PartPtr = Builder.CreateGEP(PartPtr, Builder.getInt32(1 - VF));
|
|
|
|
}
|
|
|
|
|
2013-07-31 22:17:45 +00:00
|
|
|
Value *VecPtr = Builder.CreateBitCast(PartPtr,
|
|
|
|
DataTy->getPointerTo(AddressSpace));
|
2013-01-25 21:47:42 +00:00
|
|
|
Value *LI = Builder.CreateLoad(VecPtr, "wide.load");
|
|
|
|
cast<LoadInst>(LI)->setAlignment(Alignment);
|
|
|
|
Entry[Part] = Reverse ? reverseVector(LI) : LI;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:33:08 +00:00
|
|
|
void InnerLoopVectorizer::scalarizeInstruction(Instruction *Instr) {
|
2012-10-17 18:25:06 +00:00
|
|
|
assert(!Instr->getType()->isAggregateType() && "Can't handle vectors");
|
|
|
|
// Holds vector parameters or scalars, in case of uniform vals.
|
2013-01-03 00:52:27 +00:00
|
|
|
SmallVector<VectorParts, 4> Params;
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, Instr);
|
2013-06-28 00:38:54 +00:00
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
// Find all of the vectorized parameters.
|
|
|
|
for (unsigned op = 0, e = Instr->getNumOperands(); op != e; ++op) {
|
|
|
|
Value *SrcOp = Instr->getOperand(op);
|
|
|
|
|
|
|
|
// If we are accessing the old induction variable, use the new one.
|
|
|
|
if (SrcOp == OldInduction) {
|
2012-11-29 19:25:41 +00:00
|
|
|
Params.push_back(getVectorValue(SrcOp));
|
2012-10-17 18:25:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try using previously calculated values.
|
|
|
|
Instruction *SrcInst = dyn_cast<Instruction>(SrcOp);
|
|
|
|
|
|
|
|
// If the src is an instruction that appeared earlier in the basic block
|
2012-10-18 05:29:12 +00:00
|
|
|
// then it should already be vectorized.
|
2012-12-20 20:24:40 +00:00
|
|
|
if (SrcInst && OrigLoop->contains(SrcInst)) {
|
2013-01-03 00:52:27 +00:00
|
|
|
assert(WidenMap.has(SrcInst) && "Source operand is unavailable");
|
2012-10-17 18:25:06 +00:00
|
|
|
// The parameter is a vector value from earlier.
|
2013-01-03 00:52:27 +00:00
|
|
|
Params.push_back(WidenMap.get(SrcInst));
|
2012-10-17 18:25:06 +00:00
|
|
|
} else {
|
|
|
|
// The parameter is a scalar from outside the loop. Maybe even a constant.
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts Scalars;
|
|
|
|
Scalars.append(UF, SrcOp);
|
|
|
|
Params.push_back(Scalars);
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(Params.size() == Instr->getNumOperands() &&
|
|
|
|
"Invalid number of operands");
|
|
|
|
|
|
|
|
// Does this instruction return a value ?
|
|
|
|
bool IsVoidRetTy = Instr->getType()->isVoidTy();
|
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
Value *UndefVec = IsVoidRetTy ? 0 :
|
|
|
|
UndefValue::get(VectorType::get(Instr->getType(), VF));
|
|
|
|
// Create a new entry in the WidenMap and initialize it to Undef or Null.
|
|
|
|
VectorParts &VecResults = WidenMap.splat(Instr, UndefVec);
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-04-23 17:12:42 +00:00
|
|
|
// For each vector unroll 'part':
|
|
|
|
for (unsigned Part = 0; Part < UF; ++Part) {
|
|
|
|
// For each scalar that we create:
|
|
|
|
for (unsigned Width = 0; Width < VF; ++Width) {
|
2013-01-03 00:52:27 +00:00
|
|
|
Instruction *Cloned = Instr->clone();
|
|
|
|
if (!IsVoidRetTy)
|
|
|
|
Cloned->setName(Instr->getName() + ".cloned");
|
|
|
|
// Replace the operands of the cloned instrucions with extracted scalars.
|
|
|
|
for (unsigned op = 0, e = Instr->getNumOperands(); op != e; ++op) {
|
|
|
|
Value *Op = Params[op][Part];
|
|
|
|
// Param is a vector. Need to extract the right lane.
|
|
|
|
if (Op->getType()->isVectorTy())
|
|
|
|
Op = Builder.CreateExtractElement(Op, Builder.getInt32(Width));
|
|
|
|
Cloned->setOperand(op, Op);
|
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
// Place the cloned scalar in the new loop.
|
|
|
|
Builder.Insert(Cloned);
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
// If the original scalar returns a value we need to place it in a vector
|
|
|
|
// so that future users will be able to use it.
|
|
|
|
if (!IsVoidRetTy)
|
|
|
|
VecResults[Part] = Builder.CreateInsertElement(VecResults[Part], Cloned,
|
|
|
|
Builder.getInt32(Width));
|
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-19 13:57:58 +00:00
|
|
|
Instruction *
|
2012-12-03 21:33:08 +00:00
|
|
|
InnerLoopVectorizer::addRuntimeCheck(LoopVectorizationLegality *Legal,
|
2012-12-10 19:25:06 +00:00
|
|
|
Instruction *Loc) {
|
2012-11-25 16:27:16 +00:00
|
|
|
LoopVectorizationLegality::RuntimePointerCheck *PtrRtCheck =
|
2012-12-10 21:39:02 +00:00
|
|
|
Legal->getRuntimePointerCheck();
|
2012-11-25 16:27:16 +00:00
|
|
|
|
|
|
|
if (!PtrRtCheck->Need)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
unsigned NumPointers = PtrRtCheck->Pointers.size();
|
2013-06-24 12:09:15 +00:00
|
|
|
SmallVector<TrackingVH<Value> , 2> Starts;
|
|
|
|
SmallVector<TrackingVH<Value> , 2> Ends;
|
2012-11-25 16:27:16 +00:00
|
|
|
|
|
|
|
SCEVExpander Exp(*SE, "induction");
|
|
|
|
|
|
|
|
// Use this type for pointer arithmetic.
|
2012-12-10 19:25:06 +00:00
|
|
|
Type* PtrArithTy = Type::getInt8PtrTy(Loc->getContext(), 0);
|
2012-11-25 16:27:16 +00:00
|
|
|
|
2012-11-29 19:25:41 +00:00
|
|
|
for (unsigned i = 0; i < NumPointers; ++i) {
|
2012-11-25 16:27:16 +00:00
|
|
|
Value *Ptr = PtrRtCheck->Pointers[i];
|
|
|
|
const SCEV *Sc = SE->getSCEV(Ptr);
|
|
|
|
|
|
|
|
if (SE->isLoopInvariant(Sc, OrigLoop)) {
|
2012-11-30 22:37:11 +00:00
|
|
|
DEBUG(dbgs() << "LV: Adding RT check for a loop invariant ptr:" <<
|
2012-11-25 16:27:16 +00:00
|
|
|
*Ptr <<"\n");
|
|
|
|
Starts.push_back(Ptr);
|
|
|
|
Ends.push_back(Ptr);
|
|
|
|
} else {
|
|
|
|
DEBUG(dbgs() << "LV: Adding RT check for range:" << *Ptr <<"\n");
|
|
|
|
|
2012-12-10 19:25:06 +00:00
|
|
|
Value *Start = Exp.expandCodeFor(PtrRtCheck->Starts[i], PtrArithTy, Loc);
|
2012-11-25 16:27:16 +00:00
|
|
|
Value *End = Exp.expandCodeFor(PtrRtCheck->Ends[i], PtrArithTy, Loc);
|
|
|
|
Starts.push_back(Start);
|
|
|
|
Ends.push_back(End);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-25 19:43:15 +00:00
|
|
|
IRBuilder<> ChkBuilder(Loc);
|
2013-06-24 12:09:15 +00:00
|
|
|
// Our instructions might fold to a constant.
|
|
|
|
Value *MemoryRuntimeCheck = 0;
|
2012-11-25 16:27:16 +00:00
|
|
|
for (unsigned i = 0; i < NumPointers; ++i) {
|
|
|
|
for (unsigned j = i+1; j < NumPointers; ++j) {
|
2013-04-25 19:55:03 +00:00
|
|
|
// No need to check if two readonly pointers intersect.
|
|
|
|
if (!PtrRtCheck->IsWritePtr[i] && !PtrRtCheck->IsWritePtr[j])
|
|
|
|
continue;
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
// Only need to check pointers between two different dependency sets.
|
|
|
|
if (PtrRtCheck->DependencySetId[i] == PtrRtCheck->DependencySetId[j])
|
|
|
|
continue;
|
|
|
|
|
2013-01-25 19:43:15 +00:00
|
|
|
Value *Start0 = ChkBuilder.CreateBitCast(Starts[i], PtrArithTy, "bc");
|
|
|
|
Value *Start1 = ChkBuilder.CreateBitCast(Starts[j], PtrArithTy, "bc");
|
|
|
|
Value *End0 = ChkBuilder.CreateBitCast(Ends[i], PtrArithTy, "bc");
|
|
|
|
Value *End1 = ChkBuilder.CreateBitCast(Ends[j], PtrArithTy, "bc");
|
|
|
|
|
|
|
|
Value *Cmp0 = ChkBuilder.CreateICmpULE(Start0, End1, "bound0");
|
|
|
|
Value *Cmp1 = ChkBuilder.CreateICmpULE(Start1, End0, "bound1");
|
|
|
|
Value *IsConflict = ChkBuilder.CreateAnd(Cmp0, Cmp1, "found.conflict");
|
|
|
|
if (MemoryRuntimeCheck)
|
|
|
|
IsConflict = ChkBuilder.CreateOr(MemoryRuntimeCheck, IsConflict,
|
|
|
|
"conflict.rdx");
|
2013-06-24 12:09:15 +00:00
|
|
|
MemoryRuntimeCheck = IsConflict;
|
2012-11-25 16:27:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
// We have to do this trickery because the IRBuilder might fold the check to a
|
|
|
|
// constant expression in which case there is no Instruction anchored in a
|
|
|
|
// the block.
|
|
|
|
LLVMContext &Ctx = Loc->getContext();
|
|
|
|
Instruction * Check = BinaryOperator::CreateAnd(MemoryRuntimeCheck,
|
|
|
|
ConstantInt::getTrue(Ctx));
|
|
|
|
ChkBuilder.Insert(Check, "memcheck.conflict");
|
|
|
|
return Check;
|
2012-11-25 16:27:16 +00:00
|
|
|
}
|
|
|
|
|
2012-10-30 00:40:39 +00:00
|
|
|
void
|
2012-12-03 21:33:08 +00:00
|
|
|
InnerLoopVectorizer::createEmptyLoop(LoopVectorizationLegality *Legal) {
|
2012-10-18 05:29:12 +00:00
|
|
|
/*
|
|
|
|
In this function we generate a new loop. The new loop will contain
|
|
|
|
the vectorized instructions while the old loop will continue to run the
|
|
|
|
scalar remainder.
|
|
|
|
|
2013-01-19 13:57:58 +00:00
|
|
|
[ ] <-- vector loop bypass (may consist of multiple blocks).
|
2012-12-12 01:33:47 +00:00
|
|
|
/ |
|
|
|
|
/ v
|
2012-12-10 21:39:02 +00:00
|
|
|
| [ ] <-- vector pre header.
|
|
|
|
| |
|
|
|
|
| v
|
|
|
|
| [ ] \
|
|
|
|
| [ ]_| <-- vector loop.
|
|
|
|
| |
|
2012-12-12 01:33:47 +00:00
|
|
|
\ v
|
|
|
|
>[ ] <--- middle-block.
|
|
|
|
/ |
|
|
|
|
/ v
|
2012-12-10 21:39:02 +00:00
|
|
|
| [ ] <--- new preheader.
|
|
|
|
| |
|
|
|
|
| v
|
|
|
|
| [ ] \
|
|
|
|
| [ ]_| <-- old scalar loop to handle remainder.
|
2012-12-12 01:33:47 +00:00
|
|
|
\ |
|
|
|
|
\ v
|
|
|
|
>[ ] <-- exit block.
|
2012-10-18 05:29:12 +00:00
|
|
|
...
|
|
|
|
*/
|
|
|
|
|
2012-11-26 19:51:46 +00:00
|
|
|
BasicBlock *OldBasicBlock = OrigLoop->getHeader();
|
|
|
|
BasicBlock *BypassBlock = OrigLoop->getLoopPreheader();
|
|
|
|
BasicBlock *ExitBlock = OrigLoop->getExitBlock();
|
|
|
|
assert(ExitBlock && "Must have an exit block");
|
|
|
|
|
2012-11-25 08:41:35 +00:00
|
|
|
// Some loops have a single integer induction variable, while other loops
|
|
|
|
// don't. One example is c++ iterators that often have multiple pointer
|
|
|
|
// induction variables. In the code below we also support a case where we
|
|
|
|
// don't have a single induction variable.
|
2012-11-09 07:09:44 +00:00
|
|
|
OldInduction = Legal->getInduction();
|
2013-05-11 23:04:28 +00:00
|
|
|
Type *IdxTy = Legal->getWidestInductionType();
|
2012-11-09 07:09:44 +00:00
|
|
|
|
|
|
|
// Find the loop boundaries.
|
2013-05-31 21:48:56 +00:00
|
|
|
const SCEV *ExitCount = SE->getBackedgeTakenCount(OrigLoop);
|
2012-11-09 07:09:44 +00:00
|
|
|
assert(ExitCount != SE->getCouldNotCompute() && "Invalid loop count");
|
|
|
|
|
|
|
|
// Get the total trip count from the count by adding 1.
|
|
|
|
ExitCount = SE->getAddExpr(ExitCount,
|
|
|
|
SE->getConstant(ExitCount->getType(), 1));
|
|
|
|
|
2012-11-26 19:51:46 +00:00
|
|
|
// Expand the trip count and place the new instructions in the preheader.
|
|
|
|
// Notice that the pre-header does not change, only the loop body.
|
|
|
|
SCEVExpander Exp(*SE, "induction");
|
|
|
|
|
|
|
|
// Count holds the overall loop count (N).
|
|
|
|
Value *Count = Exp.expandCodeFor(ExitCount, ExitCount->getType(),
|
|
|
|
BypassBlock->getTerminator());
|
2012-10-18 05:29:12 +00:00
|
|
|
|
2012-11-25 08:41:35 +00:00
|
|
|
// The loop index does not have to start at Zero. Find the original start
|
|
|
|
// value from the induction PHI node. If we don't have an induction variable
|
|
|
|
// then we know that it starts at zero.
|
2013-05-11 23:04:28 +00:00
|
|
|
Builder.SetInsertPoint(BypassBlock->getTerminator());
|
|
|
|
Value *StartIdx = ExtendedIdx = OldInduction ?
|
|
|
|
Builder.CreateZExt(OldInduction->getIncomingValueForBlock(BypassBlock),
|
|
|
|
IdxTy):
|
|
|
|
ConstantInt::get(IdxTy, 0);
|
2012-10-31 00:45:26 +00:00
|
|
|
|
2012-10-18 06:09:21 +00:00
|
|
|
assert(BypassBlock && "Invalid loop structure");
|
2013-01-19 13:57:58 +00:00
|
|
|
LoopBypassBlocks.push_back(BypassBlock);
|
2012-11-26 19:51:46 +00:00
|
|
|
|
|
|
|
// Split the single block loop into the two loop structure described above.
|
2012-10-18 05:29:12 +00:00
|
|
|
BasicBlock *VectorPH =
|
2012-12-10 21:39:02 +00:00
|
|
|
BypassBlock->splitBasicBlock(BypassBlock->getTerminator(), "vector.ph");
|
2012-11-26 19:51:46 +00:00
|
|
|
BasicBlock *VecBody =
|
2012-12-10 21:39:02 +00:00
|
|
|
VectorPH->splitBasicBlock(VectorPH->getTerminator(), "vector.body");
|
2012-11-26 19:51:46 +00:00
|
|
|
BasicBlock *MiddleBlock =
|
2012-12-10 21:39:02 +00:00
|
|
|
VecBody->splitBasicBlock(VecBody->getTerminator(), "middle.block");
|
2012-10-18 05:29:12 +00:00
|
|
|
BasicBlock *ScalarPH =
|
2012-12-10 21:39:02 +00:00
|
|
|
MiddleBlock->splitBasicBlock(MiddleBlock->getTerminator(), "scalar.ph");
|
2012-11-26 19:51:46 +00:00
|
|
|
|
2013-07-13 06:20:06 +00:00
|
|
|
// Create and register the new vector loop.
|
|
|
|
Loop* Lp = new Loop();
|
|
|
|
Loop *ParentLoop = OrigLoop->getParentLoop();
|
|
|
|
|
|
|
|
// Insert the new loop into the loop nest and register the new basic blocks
|
|
|
|
// before calling any utilities such as SCEV that require valid LoopInfo.
|
|
|
|
if (ParentLoop) {
|
|
|
|
ParentLoop->addChildLoop(Lp);
|
|
|
|
ParentLoop->addBasicBlockToLoop(ScalarPH, LI->getBase());
|
|
|
|
ParentLoop->addBasicBlockToLoop(VectorPH, LI->getBase());
|
|
|
|
ParentLoop->addBasicBlockToLoop(MiddleBlock, LI->getBase());
|
|
|
|
} else {
|
|
|
|
LI->addTopLevelLoop(Lp);
|
|
|
|
}
|
|
|
|
Lp->addBasicBlockToLoop(VecBody, LI->getBase());
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
// Use this IR builder to create the loop instructions (Phi, Br, Cmp)
|
|
|
|
// inside the loop.
|
2012-10-19 23:27:19 +00:00
|
|
|
Builder.SetInsertPoint(VecBody->getFirstInsertionPt());
|
2012-10-17 18:25:06 +00:00
|
|
|
|
|
|
|
// Generate the induction variable.
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, getDebugLocFromInstOrOperands(OldInduction));
|
2012-10-19 23:27:19 +00:00
|
|
|
Induction = Builder.CreatePHI(IdxTy, 2, "index");
|
2013-01-03 00:52:27 +00:00
|
|
|
// The loop step is equal to the vectorization factor (num of SIMD elements)
|
|
|
|
// times the unroll factor (num of SIMD instructions).
|
|
|
|
Constant *Step = ConstantInt::get(IdxTy, VF * UF);
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-01-25 19:26:23 +00:00
|
|
|
// This is the IR builder that we use to add all of the logic for bypassing
|
|
|
|
// the new vector loop.
|
2013-01-25 19:43:15 +00:00
|
|
|
IRBuilder<> BypassBuilder(BypassBlock->getTerminator());
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(BypassBuilder,
|
|
|
|
getDebugLocFromInstOrOperands(OldInduction));
|
2013-01-25 19:26:23 +00:00
|
|
|
|
2012-11-25 08:41:35 +00:00
|
|
|
// We may need to extend the index in case there is a type mismatch.
|
|
|
|
// We know that the count starts at zero and does not overflow.
|
|
|
|
if (Count->getType() != IdxTy) {
|
|
|
|
// The exit count can be of pointer type. Convert it to the correct
|
|
|
|
// integer type.
|
|
|
|
if (ExitCount->getType()->isPointerTy())
|
2013-01-25 19:26:23 +00:00
|
|
|
Count = BypassBuilder.CreatePointerCast(Count, IdxTy, "ptrcnt.to.int");
|
2012-11-25 08:41:35 +00:00
|
|
|
else
|
2013-01-25 19:26:23 +00:00
|
|
|
Count = BypassBuilder.CreateZExtOrTrunc(Count, IdxTy, "cnt.cast");
|
2012-11-25 08:41:35 +00:00
|
|
|
}
|
2012-10-31 00:45:26 +00:00
|
|
|
|
|
|
|
// Add the start index to the loop count to get the new end index.
|
2013-01-25 19:26:23 +00:00
|
|
|
Value *IdxEnd = BypassBuilder.CreateAdd(Count, StartIdx, "end.idx");
|
2012-10-31 00:45:26 +00:00
|
|
|
|
2012-10-18 05:29:12 +00:00
|
|
|
// Now we need to generate the expression for N - (N % VF), which is
|
|
|
|
// the part that the vectorized body will execute.
|
2013-01-25 19:26:23 +00:00
|
|
|
Value *R = BypassBuilder.CreateURem(Count, Step, "n.mod.vf");
|
|
|
|
Value *CountRoundDown = BypassBuilder.CreateSub(Count, R, "n.vec");
|
|
|
|
Value *IdxEndRoundDown = BypassBuilder.CreateAdd(CountRoundDown, StartIdx,
|
|
|
|
"end.idx.rnd.down");
|
2012-10-18 05:29:12 +00:00
|
|
|
|
2012-11-25 08:41:35 +00:00
|
|
|
// Now, compare the new count to zero. If it is zero skip the vector loop and
|
|
|
|
// jump to the scalar loop.
|
2013-01-25 19:43:15 +00:00
|
|
|
Value *Cmp = BypassBuilder.CreateICmpEQ(IdxEndRoundDown, StartIdx,
|
|
|
|
"cmp.zero");
|
2013-01-25 19:26:23 +00:00
|
|
|
|
|
|
|
BasicBlock *LastBypassBlock = BypassBlock;
|
2012-11-09 07:09:44 +00:00
|
|
|
|
2013-01-19 13:57:58 +00:00
|
|
|
// Generate the code that checks in runtime if arrays overlap. We put the
|
|
|
|
// checks into a separate block to make the more common case of few elements
|
|
|
|
// faster.
|
2013-01-25 19:26:23 +00:00
|
|
|
Instruction *MemRuntimeCheck = addRuntimeCheck(Legal,
|
|
|
|
BypassBlock->getTerminator());
|
|
|
|
if (MemRuntimeCheck) {
|
2013-01-19 13:57:58 +00:00
|
|
|
// Create a new block containing the memory check.
|
2013-01-25 19:26:23 +00:00
|
|
|
BasicBlock *CheckBlock = BypassBlock->splitBasicBlock(MemRuntimeCheck,
|
2013-01-19 13:57:58 +00:00
|
|
|
"vector.memcheck");
|
2013-07-13 06:20:06 +00:00
|
|
|
if (ParentLoop)
|
|
|
|
ParentLoop->addBasicBlockToLoop(CheckBlock, LI->getBase());
|
2013-01-19 13:57:58 +00:00
|
|
|
LoopBypassBlocks.push_back(CheckBlock);
|
|
|
|
|
|
|
|
// Replace the branch into the memory check block with a conditional branch
|
|
|
|
// for the "few elements case".
|
|
|
|
Instruction *OldTerm = BypassBlock->getTerminator();
|
|
|
|
BranchInst::Create(MiddleBlock, CheckBlock, Cmp, OldTerm);
|
|
|
|
OldTerm->eraseFromParent();
|
|
|
|
|
2013-01-25 19:26:23 +00:00
|
|
|
Cmp = MemRuntimeCheck;
|
|
|
|
LastBypassBlock = CheckBlock;
|
2013-01-19 13:57:58 +00:00
|
|
|
}
|
2012-11-09 07:09:44 +00:00
|
|
|
|
2013-01-25 19:26:23 +00:00
|
|
|
LastBypassBlock->getTerminator()->eraseFromParent();
|
|
|
|
BranchInst::Create(MiddleBlock, VectorPH, Cmp,
|
|
|
|
LastBypassBlock);
|
2012-10-18 05:29:12 +00:00
|
|
|
|
2012-11-09 07:09:44 +00:00
|
|
|
// We are going to resume the execution of the scalar loop.
|
2012-11-17 00:27:03 +00:00
|
|
|
// Go over all of the induction variables that we found and fix the
|
|
|
|
// PHIs that are left in the scalar version of the loop.
|
|
|
|
// The starting values of PHI nodes depend on the counter of the last
|
|
|
|
// iteration in the vectorized loop.
|
2012-12-10 21:39:02 +00:00
|
|
|
// If we come from a bypass edge then we need to start from the original
|
|
|
|
// start value.
|
2012-11-17 00:27:03 +00:00
|
|
|
|
|
|
|
// This variable saves the new starting index for the scalar loop.
|
2012-11-25 08:41:35 +00:00
|
|
|
PHINode *ResumeIndex = 0;
|
2012-11-17 00:27:03 +00:00
|
|
|
LoopVectorizationLegality::InductionList::iterator I, E;
|
|
|
|
LoopVectorizationLegality::InductionList *List = Legal->getInductionVars();
|
2013-05-11 23:04:24 +00:00
|
|
|
// Set builder to point to last bypass block.
|
|
|
|
BypassBuilder.SetInsertPoint(LoopBypassBlocks.back()->getTerminator());
|
2012-11-17 00:27:03 +00:00
|
|
|
for (I = List->begin(), E = List->end(); I != E; ++I) {
|
|
|
|
PHINode *OrigPhi = I->first;
|
2012-12-10 19:25:06 +00:00
|
|
|
LoopVectorizationLegality::InductionInfo II = I->second;
|
2013-05-11 23:04:28 +00:00
|
|
|
|
|
|
|
Type *ResumeValTy = (OrigPhi == OldInduction) ? IdxTy : OrigPhi->getType();
|
|
|
|
PHINode *ResumeVal = PHINode::Create(ResumeValTy, 2, "resume.val",
|
2012-12-10 21:39:02 +00:00
|
|
|
MiddleBlock->getTerminator());
|
2013-05-11 23:04:28 +00:00
|
|
|
// We might have extended the type of the induction variable but we need a
|
|
|
|
// truncated version for the scalar loop.
|
|
|
|
PHINode *TruncResumeVal = (OrigPhi == OldInduction) ?
|
|
|
|
PHINode::Create(OrigPhi->getType(), 2, "trunc.resume.val",
|
|
|
|
MiddleBlock->getTerminator()) : 0;
|
|
|
|
|
2012-11-17 00:27:03 +00:00
|
|
|
Value *EndValue = 0;
|
2012-12-10 19:25:06 +00:00
|
|
|
switch (II.IK) {
|
2013-01-08 17:23:17 +00:00
|
|
|
case LoopVectorizationLegality::IK_NoInduction:
|
2012-12-10 19:25:06 +00:00
|
|
|
llvm_unreachable("Unknown induction");
|
2013-01-08 17:23:17 +00:00
|
|
|
case LoopVectorizationLegality::IK_IntInduction: {
|
2013-05-14 00:21:18 +00:00
|
|
|
// Handle the integer induction counter.
|
2012-12-10 19:25:06 +00:00
|
|
|
assert(OrigPhi->getType()->isIntegerTy() && "Invalid type");
|
2013-05-14 00:21:18 +00:00
|
|
|
|
|
|
|
// We have the canonical induction variable.
|
2013-05-11 23:04:28 +00:00
|
|
|
if (OrigPhi == OldInduction) {
|
|
|
|
// Create a truncated version of the resume value for the scalar loop,
|
|
|
|
// we might have promoted the type to a larger width.
|
|
|
|
EndValue =
|
|
|
|
BypassBuilder.CreateTrunc(IdxEndRoundDown, OrigPhi->getType());
|
|
|
|
// The new PHI merges the original incoming value, in case of a bypass,
|
|
|
|
// or the value at the end of the vectorized loop.
|
|
|
|
for (unsigned I = 0, E = LoopBypassBlocks.size(); I != E; ++I)
|
|
|
|
TruncResumeVal->addIncoming(II.StartValue, LoopBypassBlocks[I]);
|
|
|
|
TruncResumeVal->addIncoming(EndValue, VecBody);
|
2013-05-14 00:21:18 +00:00
|
|
|
|
|
|
|
// We know what the end value is.
|
|
|
|
EndValue = IdxEndRoundDown;
|
|
|
|
// We also know which PHI node holds it.
|
|
|
|
ResumeIndex = ResumeVal;
|
|
|
|
break;
|
2013-05-11 23:04:28 +00:00
|
|
|
}
|
2013-05-14 00:21:18 +00:00
|
|
|
|
|
|
|
// Not the canonical induction variable - add the vector loop count to the
|
|
|
|
// start value.
|
|
|
|
Value *CRD = BypassBuilder.CreateSExtOrTrunc(CountRoundDown,
|
|
|
|
II.StartValue->getType(),
|
|
|
|
"cast.crd");
|
|
|
|
EndValue = BypassBuilder.CreateAdd(CRD, II.StartValue , "ind.end");
|
2012-12-10 19:25:06 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-01-08 17:23:17 +00:00
|
|
|
case LoopVectorizationLegality::IK_ReverseIntInduction: {
|
2012-12-10 19:25:06 +00:00
|
|
|
// Convert the CountRoundDown variable to the PHI size.
|
2013-05-11 23:04:24 +00:00
|
|
|
Value *CRD = BypassBuilder.CreateSExtOrTrunc(CountRoundDown,
|
|
|
|
II.StartValue->getType(),
|
|
|
|
"cast.crd");
|
|
|
|
// Handle reverse integer induction counter.
|
|
|
|
EndValue = BypassBuilder.CreateSub(II.StartValue, CRD, "rev.ind.end");
|
2012-12-10 19:25:06 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-01-08 17:23:17 +00:00
|
|
|
case LoopVectorizationLegality::IK_PtrInduction: {
|
2012-11-17 00:27:03 +00:00
|
|
|
// For pointer induction variables, calculate the offset using
|
|
|
|
// the end index.
|
2013-05-11 23:04:24 +00:00
|
|
|
EndValue = BypassBuilder.CreateGEP(II.StartValue, CountRoundDown,
|
|
|
|
"ptr.ind.end");
|
2012-12-10 19:25:06 +00:00
|
|
|
break;
|
2012-11-17 00:27:03 +00:00
|
|
|
}
|
2013-01-23 01:35:00 +00:00
|
|
|
case LoopVectorizationLegality::IK_ReversePtrInduction: {
|
|
|
|
// The value at the end of the loop for the reverse pointer is calculated
|
|
|
|
// by creating a GEP with a negative index starting from the start value.
|
|
|
|
Value *Zero = ConstantInt::get(CountRoundDown->getType(), 0);
|
2013-05-11 23:04:24 +00:00
|
|
|
Value *NegIdx = BypassBuilder.CreateSub(Zero, CountRoundDown,
|
|
|
|
"rev.ind.end");
|
|
|
|
EndValue = BypassBuilder.CreateGEP(II.StartValue, NegIdx,
|
|
|
|
"rev.ptr.ind.end");
|
2013-01-23 01:35:00 +00:00
|
|
|
break;
|
|
|
|
}
|
2012-12-10 19:25:06 +00:00
|
|
|
}// end of case
|
2012-11-17 00:27:03 +00:00
|
|
|
|
|
|
|
// The new PHI merges the original incoming value, in case of a bypass,
|
|
|
|
// or the value at the end of the vectorized loop.
|
2013-05-11 23:04:28 +00:00
|
|
|
for (unsigned I = 0, E = LoopBypassBlocks.size(); I != E; ++I) {
|
|
|
|
if (OrigPhi == OldInduction)
|
|
|
|
ResumeVal->addIncoming(StartIdx, LoopBypassBlocks[I]);
|
|
|
|
else
|
|
|
|
ResumeVal->addIncoming(II.StartValue, LoopBypassBlocks[I]);
|
|
|
|
}
|
2012-11-17 00:27:03 +00:00
|
|
|
ResumeVal->addIncoming(EndValue, VecBody);
|
|
|
|
|
|
|
|
// Fix the scalar body counter (PHI node).
|
2012-11-25 08:41:35 +00:00
|
|
|
unsigned BlockIdx = OrigPhi->getBasicBlockIndex(ScalarPH);
|
2013-05-11 23:04:28 +00:00
|
|
|
// The old inductions phi node in the scalar body needs the truncated value.
|
|
|
|
if (OrigPhi == OldInduction)
|
|
|
|
OrigPhi->setIncomingValue(BlockIdx, TruncResumeVal);
|
|
|
|
else
|
|
|
|
OrigPhi->setIncomingValue(BlockIdx, ResumeVal);
|
2012-11-17 00:27:03 +00:00
|
|
|
}
|
|
|
|
|
2012-11-25 08:41:35 +00:00
|
|
|
// If we are generating a new induction variable then we also need to
|
|
|
|
// generate the code that calculates the exit value. This value is not
|
|
|
|
// simply the end of the counter because we may skip the vectorized body
|
|
|
|
// in case of a runtime check.
|
|
|
|
if (!OldInduction){
|
|
|
|
assert(!ResumeIndex && "Unexpected resume value found");
|
|
|
|
ResumeIndex = PHINode::Create(IdxTy, 2, "new.indc.resume.val",
|
|
|
|
MiddleBlock->getTerminator());
|
2013-01-19 13:57:58 +00:00
|
|
|
for (unsigned I = 0, E = LoopBypassBlocks.size(); I != E; ++I)
|
|
|
|
ResumeIndex->addIncoming(StartIdx, LoopBypassBlocks[I]);
|
2012-11-25 08:41:35 +00:00
|
|
|
ResumeIndex->addIncoming(IdxEndRoundDown, VecBody);
|
|
|
|
}
|
|
|
|
|
2012-11-17 00:27:03 +00:00
|
|
|
// Make sure that we found the index where scalar loop needs to continue.
|
|
|
|
assert(ResumeIndex && ResumeIndex->getType()->isIntegerTy() &&
|
|
|
|
"Invalid resume Index");
|
2012-11-09 07:09:44 +00:00
|
|
|
|
2012-10-18 05:29:12 +00:00
|
|
|
// Add a check in the middle block to see if we have completed
|
|
|
|
// all of the iterations in the first vector loop.
|
|
|
|
// If (N - N%VF) == N, then we *don't* need to run the remainder.
|
2012-10-31 00:45:26 +00:00
|
|
|
Value *CmpN = CmpInst::Create(Instruction::ICmp, CmpInst::ICMP_EQ, IdxEnd,
|
2012-11-09 07:09:44 +00:00
|
|
|
ResumeIndex, "cmp.n",
|
2012-10-18 05:29:12 +00:00
|
|
|
MiddleBlock->getTerminator());
|
|
|
|
|
|
|
|
BranchInst::Create(ExitBlock, ScalarPH, CmpN, MiddleBlock->getTerminator());
|
|
|
|
// Remove the old terminator.
|
|
|
|
MiddleBlock->getTerminator()->eraseFromParent();
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
// Create i+1 and fill the PHINode.
|
2012-10-19 23:27:19 +00:00
|
|
|
Value *NextIdx = Builder.CreateAdd(Induction, Step, "index.next");
|
2012-10-31 00:45:26 +00:00
|
|
|
Induction->addIncoming(StartIdx, VectorPH);
|
2012-10-18 05:29:12 +00:00
|
|
|
Induction->addIncoming(NextIdx, VecBody);
|
2012-10-17 18:25:06 +00:00
|
|
|
// Create the compare.
|
2012-10-31 00:45:26 +00:00
|
|
|
Value *ICmp = Builder.CreateICmpEQ(NextIdx, IdxEndRoundDown);
|
2012-10-19 23:27:19 +00:00
|
|
|
Builder.CreateCondBr(ICmp, MiddleBlock, VecBody);
|
2012-10-18 05:29:12 +00:00
|
|
|
|
|
|
|
// Now we have two terminators. Remove the old one from the block.
|
|
|
|
VecBody->getTerminator()->eraseFromParent();
|
|
|
|
|
|
|
|
// Get ready to start creating new instructions into the vectorized body.
|
2012-10-19 23:27:19 +00:00
|
|
|
Builder.SetInsertPoint(VecBody->getFirstInsertionPt());
|
2012-10-18 05:29:12 +00:00
|
|
|
|
2012-10-19 23:05:40 +00:00
|
|
|
// Save the state.
|
2012-10-29 21:52:38 +00:00
|
|
|
LoopVectorPreHeader = VectorPH;
|
|
|
|
LoopScalarPreHeader = ScalarPH;
|
2012-10-19 23:05:40 +00:00
|
|
|
LoopMiddleBlock = MiddleBlock;
|
|
|
|
LoopExitBlock = ExitBlock;
|
|
|
|
LoopVectorBody = VecBody;
|
|
|
|
LoopScalarBody = OldBasicBlock;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2012-10-30 22:06:26 +00:00
|
|
|
/// This function returns the identity element (or neutral element) for
|
|
|
|
/// the operation K.
|
2013-04-19 21:03:36 +00:00
|
|
|
Constant*
|
2013-05-05 01:54:42 +00:00
|
|
|
LoopVectorizationLegality::getReductionIdentity(ReductionKind K, Type *Tp) {
|
2012-10-30 18:12:36 +00:00
|
|
|
switch (K) {
|
2013-04-19 21:03:36 +00:00
|
|
|
case RK_IntegerXor:
|
|
|
|
case RK_IntegerAdd:
|
|
|
|
case RK_IntegerOr:
|
2012-10-30 22:06:26 +00:00
|
|
|
// Adding, Xoring, Oring zero to a number does not change it.
|
2013-01-07 23:13:00 +00:00
|
|
|
return ConstantInt::get(Tp, 0);
|
2013-04-19 21:03:36 +00:00
|
|
|
case RK_IntegerMult:
|
2012-10-30 22:06:26 +00:00
|
|
|
// Multiplying a number by 1 does not change it.
|
2013-01-07 23:13:00 +00:00
|
|
|
return ConstantInt::get(Tp, 1);
|
2013-04-19 21:03:36 +00:00
|
|
|
case RK_IntegerAnd:
|
2012-10-30 22:06:26 +00:00
|
|
|
// AND-ing a number with an all-1 value does not change it.
|
2013-01-07 23:13:00 +00:00
|
|
|
return ConstantInt::get(Tp, -1, true);
|
2013-04-19 21:03:36 +00:00
|
|
|
case RK_FloatMult:
|
2013-01-07 23:13:00 +00:00
|
|
|
// Multiplying a number by 1 does not change it.
|
|
|
|
return ConstantFP::get(Tp, 1.0L);
|
2013-04-19 21:03:36 +00:00
|
|
|
case RK_FloatAdd:
|
2013-01-07 23:13:00 +00:00
|
|
|
// Adding zero to a number does not change it.
|
|
|
|
return ConstantFP::get(Tp, 0.0L);
|
2012-10-30 18:12:36 +00:00
|
|
|
default:
|
|
|
|
llvm_unreachable("Unknown reduction kind");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-23 14:54:39 +00:00
|
|
|
static Intrinsic::ID checkUnaryFloatSignature(const CallInst &I,
|
|
|
|
Intrinsic::ID ValidIntrinsicID) {
|
|
|
|
if (I.getNumArgOperands() != 1 ||
|
|
|
|
!I.getArgOperand(0)->getType()->isFloatingPointTy() ||
|
|
|
|
I.getType() != I.getArgOperand(0)->getType() ||
|
|
|
|
!I.onlyReadsMemory())
|
|
|
|
return Intrinsic::not_intrinsic;
|
|
|
|
|
|
|
|
return ValidIntrinsicID;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Intrinsic::ID checkBinaryFloatSignature(const CallInst &I,
|
|
|
|
Intrinsic::ID ValidIntrinsicID) {
|
|
|
|
if (I.getNumArgOperands() != 2 ||
|
|
|
|
!I.getArgOperand(0)->getType()->isFloatingPointTy() ||
|
|
|
|
!I.getArgOperand(1)->getType()->isFloatingPointTy() ||
|
|
|
|
I.getType() != I.getArgOperand(0)->getType() ||
|
|
|
|
I.getType() != I.getArgOperand(1)->getType() ||
|
|
|
|
!I.onlyReadsMemory())
|
|
|
|
return Intrinsic::not_intrinsic;
|
|
|
|
|
|
|
|
return ValidIntrinsicID;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-27 15:24:19 +00:00
|
|
|
static Intrinsic::ID
|
|
|
|
getIntrinsicIDForCall(CallInst *CI, const TargetLibraryInfo *TLI) {
|
|
|
|
// If we have an intrinsic call, check if it is trivially vectorizable.
|
|
|
|
if (IntrinsicInst *II = dyn_cast<IntrinsicInst>(CI)) {
|
|
|
|
switch (II->getIntrinsicID()) {
|
|
|
|
case Intrinsic::sqrt:
|
|
|
|
case Intrinsic::sin:
|
|
|
|
case Intrinsic::cos:
|
|
|
|
case Intrinsic::exp:
|
|
|
|
case Intrinsic::exp2:
|
|
|
|
case Intrinsic::log:
|
|
|
|
case Intrinsic::log10:
|
|
|
|
case Intrinsic::log2:
|
|
|
|
case Intrinsic::fabs:
|
Add a llvm.copysign intrinsic
This adds a llvm.copysign intrinsic; We already have Libfunc recognition for
copysign (which is turned into the FCOPYSIGN SDAG node). In order to
autovectorize calls to copysign in the loop vectorizer, we need a corresponding
intrinsic as well.
In addition to the expected changes to the language reference, the loop
vectorizer, BasicTTI, and the SDAG builder (the intrinsic is transformed into
an FCOPYSIGN node, just like the function call), this also adds FCOPYSIGN to a
few lists in LegalizeVector{Ops,Types} so that vector copysigns can be
expanded.
In TargetLoweringBase::initActions, I've made the default action for FCOPYSIGN
be Expand for vector types. This seems correct for all in-tree targets, and I
think is the right thing to do because, previously, there was no way to generate
vector-values FCOPYSIGN nodes (and most targets don't specify an action for
vector-typed FCOPYSIGN).
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@188728 91177308-0d34-0410-b5e6-96231b3b80d8
2013-08-19 23:35:46 +00:00
|
|
|
case Intrinsic::copysign:
|
2013-02-27 15:24:19 +00:00
|
|
|
case Intrinsic::floor:
|
|
|
|
case Intrinsic::ceil:
|
|
|
|
case Intrinsic::trunc:
|
|
|
|
case Intrinsic::rint:
|
|
|
|
case Intrinsic::nearbyint:
|
2013-08-07 22:49:12 +00:00
|
|
|
case Intrinsic::round:
|
2013-02-27 15:24:19 +00:00
|
|
|
case Intrinsic::pow:
|
|
|
|
case Intrinsic::fma:
|
|
|
|
case Intrinsic::fmuladd:
|
2013-08-06 22:37:52 +00:00
|
|
|
case Intrinsic::lifetime_start:
|
|
|
|
case Intrinsic::lifetime_end:
|
2013-02-27 15:24:19 +00:00
|
|
|
return II->getIntrinsicID();
|
|
|
|
default:
|
|
|
|
return Intrinsic::not_intrinsic;
|
|
|
|
}
|
2012-12-09 20:42:17 +00:00
|
|
|
}
|
2013-02-27 15:24:19 +00:00
|
|
|
|
|
|
|
if (!TLI)
|
|
|
|
return Intrinsic::not_intrinsic;
|
|
|
|
|
|
|
|
LibFunc::Func Func;
|
|
|
|
Function *F = CI->getCalledFunction();
|
|
|
|
// We're going to make assumptions on the semantics of the functions, check
|
2013-09-23 14:54:39 +00:00
|
|
|
// that the target knows that it's available in this environment and it does
|
|
|
|
// not have local linkage.
|
|
|
|
if (!F || F->hasLocalLinkage() || !TLI->getLibFunc(F->getName(), Func))
|
2013-02-27 15:24:19 +00:00
|
|
|
return Intrinsic::not_intrinsic;
|
|
|
|
|
|
|
|
// Otherwise check if we have a call to a function that can be turned into a
|
|
|
|
// vector intrinsic.
|
|
|
|
switch (Func) {
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
case LibFunc::sin:
|
|
|
|
case LibFunc::sinf:
|
|
|
|
case LibFunc::sinl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::sin);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::cos:
|
|
|
|
case LibFunc::cosf:
|
|
|
|
case LibFunc::cosl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::cos);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::exp:
|
|
|
|
case LibFunc::expf:
|
|
|
|
case LibFunc::expl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::exp);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::exp2:
|
|
|
|
case LibFunc::exp2f:
|
|
|
|
case LibFunc::exp2l:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::exp2);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::log:
|
|
|
|
case LibFunc::logf:
|
|
|
|
case LibFunc::logl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::log);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::log10:
|
|
|
|
case LibFunc::log10f:
|
|
|
|
case LibFunc::log10l:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::log10);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::log2:
|
|
|
|
case LibFunc::log2f:
|
|
|
|
case LibFunc::log2l:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::log2);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::fabs:
|
|
|
|
case LibFunc::fabsf:
|
|
|
|
case LibFunc::fabsl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::fabs);
|
Add a llvm.copysign intrinsic
This adds a llvm.copysign intrinsic; We already have Libfunc recognition for
copysign (which is turned into the FCOPYSIGN SDAG node). In order to
autovectorize calls to copysign in the loop vectorizer, we need a corresponding
intrinsic as well.
In addition to the expected changes to the language reference, the loop
vectorizer, BasicTTI, and the SDAG builder (the intrinsic is transformed into
an FCOPYSIGN node, just like the function call), this also adds FCOPYSIGN to a
few lists in LegalizeVector{Ops,Types} so that vector copysigns can be
expanded.
In TargetLoweringBase::initActions, I've made the default action for FCOPYSIGN
be Expand for vector types. This seems correct for all in-tree targets, and I
think is the right thing to do because, previously, there was no way to generate
vector-values FCOPYSIGN nodes (and most targets don't specify an action for
vector-typed FCOPYSIGN).
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@188728 91177308-0d34-0410-b5e6-96231b3b80d8
2013-08-19 23:35:46 +00:00
|
|
|
case LibFunc::copysign:
|
|
|
|
case LibFunc::copysignf:
|
|
|
|
case LibFunc::copysignl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkBinaryFloatSignature(*CI, Intrinsic::copysign);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::floor:
|
|
|
|
case LibFunc::floorf:
|
|
|
|
case LibFunc::floorl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::floor);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::ceil:
|
|
|
|
case LibFunc::ceilf:
|
|
|
|
case LibFunc::ceill:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::ceil);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::trunc:
|
|
|
|
case LibFunc::truncf:
|
|
|
|
case LibFunc::truncl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::trunc);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::rint:
|
|
|
|
case LibFunc::rintf:
|
|
|
|
case LibFunc::rintl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::rint);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::nearbyint:
|
|
|
|
case LibFunc::nearbyintf:
|
|
|
|
case LibFunc::nearbyintl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::nearbyint);
|
2013-08-07 22:49:12 +00:00
|
|
|
case LibFunc::round:
|
|
|
|
case LibFunc::roundf:
|
|
|
|
case LibFunc::roundl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkUnaryFloatSignature(*CI, Intrinsic::round);
|
2013-02-27 15:24:19 +00:00
|
|
|
case LibFunc::pow:
|
|
|
|
case LibFunc::powf:
|
|
|
|
case LibFunc::powl:
|
2013-09-23 14:54:39 +00:00
|
|
|
return checkBinaryFloatSignature(*CI, Intrinsic::pow);
|
2013-02-27 15:24:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return Intrinsic::not_intrinsic;
|
2012-12-09 20:42:17 +00:00
|
|
|
}
|
|
|
|
|
2013-01-08 17:37:45 +00:00
|
|
|
/// This function translates the reduction kind to an LLVM binary operator.
|
2013-04-18 17:22:34 +00:00
|
|
|
static unsigned
|
2013-01-08 17:37:45 +00:00
|
|
|
getReductionBinOp(LoopVectorizationLegality::ReductionKind Kind) {
|
|
|
|
switch (Kind) {
|
|
|
|
case LoopVectorizationLegality::RK_IntegerAdd:
|
|
|
|
return Instruction::Add;
|
|
|
|
case LoopVectorizationLegality::RK_IntegerMult:
|
|
|
|
return Instruction::Mul;
|
|
|
|
case LoopVectorizationLegality::RK_IntegerOr:
|
|
|
|
return Instruction::Or;
|
|
|
|
case LoopVectorizationLegality::RK_IntegerAnd:
|
|
|
|
return Instruction::And;
|
|
|
|
case LoopVectorizationLegality::RK_IntegerXor:
|
|
|
|
return Instruction::Xor;
|
|
|
|
case LoopVectorizationLegality::RK_FloatMult:
|
|
|
|
return Instruction::FMul;
|
|
|
|
case LoopVectorizationLegality::RK_FloatAdd:
|
|
|
|
return Instruction::FAdd;
|
2013-04-18 17:22:34 +00:00
|
|
|
case LoopVectorizationLegality::RK_IntegerMinMax:
|
|
|
|
return Instruction::ICmp;
|
2013-05-05 01:54:48 +00:00
|
|
|
case LoopVectorizationLegality::RK_FloatMinMax:
|
|
|
|
return Instruction::FCmp;
|
2013-01-08 17:37:45 +00:00
|
|
|
default:
|
|
|
|
llvm_unreachable("Unknown reduction operation");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-19 21:03:36 +00:00
|
|
|
Value *createMinMaxOp(IRBuilder<> &Builder,
|
|
|
|
LoopVectorizationLegality::MinMaxReductionKind RK,
|
|
|
|
Value *Left,
|
2013-04-18 17:22:34 +00:00
|
|
|
Value *Right) {
|
2013-04-19 21:03:36 +00:00
|
|
|
CmpInst::Predicate P = CmpInst::ICMP_NE;
|
|
|
|
switch (RK) {
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Unknown min/max reduction kind");
|
|
|
|
case LoopVectorizationLegality::MRK_UIntMin:
|
|
|
|
P = CmpInst::ICMP_ULT;
|
|
|
|
break;
|
|
|
|
case LoopVectorizationLegality::MRK_UIntMax:
|
|
|
|
P = CmpInst::ICMP_UGT;
|
|
|
|
break;
|
|
|
|
case LoopVectorizationLegality::MRK_SIntMin:
|
|
|
|
P = CmpInst::ICMP_SLT;
|
|
|
|
break;
|
|
|
|
case LoopVectorizationLegality::MRK_SIntMax:
|
|
|
|
P = CmpInst::ICMP_SGT;
|
2013-05-05 01:54:48 +00:00
|
|
|
break;
|
|
|
|
case LoopVectorizationLegality::MRK_FloatMin:
|
|
|
|
P = CmpInst::FCMP_OLT;
|
|
|
|
break;
|
|
|
|
case LoopVectorizationLegality::MRK_FloatMax:
|
|
|
|
P = CmpInst::FCMP_OGT;
|
|
|
|
break;
|
2013-04-19 21:03:36 +00:00
|
|
|
}
|
2013-05-05 01:54:48 +00:00
|
|
|
|
|
|
|
Value *Cmp;
|
2013-07-31 22:17:45 +00:00
|
|
|
if (RK == LoopVectorizationLegality::MRK_FloatMin ||
|
|
|
|
RK == LoopVectorizationLegality::MRK_FloatMax)
|
2013-05-05 01:54:48 +00:00
|
|
|
Cmp = Builder.CreateFCmp(P, Left, Right, "rdx.minmax.cmp");
|
|
|
|
else
|
|
|
|
Cmp = Builder.CreateICmp(P, Left, Right, "rdx.minmax.cmp");
|
|
|
|
|
2013-04-18 17:22:34 +00:00
|
|
|
Value *Select = Builder.CreateSelect(Cmp, Left, Right, "rdx.minmax.select");
|
|
|
|
return Select;
|
|
|
|
}
|
|
|
|
|
2012-10-19 23:05:40 +00:00
|
|
|
void
|
2012-12-03 21:33:08 +00:00
|
|
|
InnerLoopVectorizer::vectorizeLoop(LoopVectorizationLegality *Legal) {
|
2012-10-24 20:36:32 +00:00
|
|
|
//===------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Notice: any optimization or new instruction that go
|
|
|
|
// into the code below should be also be implemented in
|
|
|
|
// the cost-model.
|
|
|
|
//
|
|
|
|
//===------------------------------------------------===//
|
2013-01-25 21:47:42 +00:00
|
|
|
Constant *Zero = Builder.getInt32(0);
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2012-10-19 23:05:40 +00:00
|
|
|
// In order to support reduction variables we need to be able to vectorize
|
|
|
|
// Phi nodes. Phi nodes have cycles, so we need to vectorize them in two
|
2012-11-29 19:25:41 +00:00
|
|
|
// stages. First, we create a new vector PHI node with no incoming edges.
|
2012-10-19 23:05:40 +00:00
|
|
|
// We use this value when we vectorize all of the instructions that use the
|
|
|
|
// PHI. Next, after all of the instructions in the block are complete we
|
|
|
|
// add the new incoming edges to the PHI. At this point all of the
|
|
|
|
// instructions in the basic block are vectorized, so we can use them to
|
|
|
|
// construct the PHI.
|
2012-11-17 00:27:03 +00:00
|
|
|
PhiVector RdxPHIsToFix;
|
2012-10-19 23:05:40 +00:00
|
|
|
|
2012-12-04 06:15:11 +00:00
|
|
|
// Scan the loop in a topological order to ensure that defs are vectorized
|
|
|
|
// before users.
|
|
|
|
LoopBlocksDFS DFS(OrigLoop);
|
|
|
|
DFS.perform(LI);
|
|
|
|
|
|
|
|
// Vectorize all of the blocks in the original loop.
|
|
|
|
for (LoopBlocksDFS::RPOIterator bb = DFS.beginRPO(),
|
|
|
|
be = DFS.endRPO(); bb != be; ++bb)
|
|
|
|
vectorizeBlockInLoop(Legal, *bb, &RdxPHIsToFix);
|
|
|
|
|
|
|
|
// At this point every instruction in the original loop is widened to
|
|
|
|
// a vector form. We are almost done. Now, we need to fix the PHI nodes
|
|
|
|
// that we vectorized. The PHI nodes are currently empty because we did
|
|
|
|
// not want to introduce cycles. Notice that the remaining PHI nodes
|
|
|
|
// that we need to fix are reduction variables.
|
|
|
|
|
|
|
|
// Create the 'reduced' values for each of the induction vars.
|
|
|
|
// The reduced values are the vector values that we scalarize and combine
|
|
|
|
// after the loop is finished.
|
|
|
|
for (PhiVector::iterator it = RdxPHIsToFix.begin(), e = RdxPHIsToFix.end();
|
|
|
|
it != e; ++it) {
|
|
|
|
PHINode *RdxPhi = *it;
|
|
|
|
assert(RdxPhi && "Unable to recover vectorized PHI");
|
|
|
|
|
|
|
|
// Find the reduction variable descriptor.
|
|
|
|
assert(Legal->getReductionVars()->count(RdxPhi) &&
|
|
|
|
"Unable to find the reduction variable");
|
|
|
|
LoopVectorizationLegality::ReductionDescriptor RdxDesc =
|
2012-12-10 21:39:02 +00:00
|
|
|
(*Legal->getReductionVars())[RdxPhi];
|
2012-12-04 06:15:11 +00:00
|
|
|
|
2013-06-28 17:14:48 +00:00
|
|
|
setDebugLocFromInst(Builder, RdxDesc.StartValue);
|
2013-06-28 16:26:54 +00:00
|
|
|
|
2012-12-04 06:15:11 +00:00
|
|
|
// We need to generate a reduction vector from the incoming scalar.
|
|
|
|
// To do so, we need to generate the 'identity' vector and overide
|
|
|
|
// one of the elements with the incoming scalar reduction. We need
|
|
|
|
// to do it in the vector-loop preheader.
|
2013-03-08 16:58:37 +00:00
|
|
|
Builder.SetInsertPoint(LoopBypassBlocks.front()->getTerminator());
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
// This is the vector-clone of the value that leaves the loop.
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts &VectorExit = getVectorValue(RdxDesc.LoopExitInstr);
|
|
|
|
Type *VecTy = VectorExit[0]->getType();
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
// Find the reduction identity variable. Zero for addition, or, xor,
|
|
|
|
// one for multiplication, -1 for And.
|
2013-05-05 01:54:42 +00:00
|
|
|
Value *Identity;
|
|
|
|
Value *VectorStart;
|
2013-05-05 01:54:48 +00:00
|
|
|
if (RdxDesc.Kind == LoopVectorizationLegality::RK_IntegerMinMax ||
|
|
|
|
RdxDesc.Kind == LoopVectorizationLegality::RK_FloatMinMax) {
|
2013-05-05 01:54:42 +00:00
|
|
|
// MinMax reduction have the start value as their identify.
|
2013-08-27 18:52:47 +00:00
|
|
|
if (VF == 1) {
|
|
|
|
VectorStart = Identity = RdxDesc.StartValue;
|
|
|
|
} else {
|
|
|
|
VectorStart = Identity = Builder.CreateVectorSplat(VF,
|
|
|
|
RdxDesc.StartValue,
|
|
|
|
"minmax.ident");
|
|
|
|
}
|
2013-05-05 01:54:48 +00:00
|
|
|
} else {
|
2013-08-27 18:52:47 +00:00
|
|
|
// Handle other reduction kinds:
|
2013-05-05 01:54:42 +00:00
|
|
|
Constant *Iden =
|
2013-08-27 18:52:47 +00:00
|
|
|
LoopVectorizationLegality::getReductionIdentity(RdxDesc.Kind,
|
|
|
|
VecTy->getScalarType());
|
|
|
|
if (VF == 1) {
|
|
|
|
Identity = Iden;
|
|
|
|
// This vector is the Identity vector where the first element is the
|
|
|
|
// incoming scalar reduction.
|
|
|
|
VectorStart = RdxDesc.StartValue;
|
|
|
|
} else {
|
|
|
|
Identity = ConstantVector::getSplat(VF, Iden);
|
|
|
|
|
|
|
|
// This vector is the Identity vector where the first element is the
|
|
|
|
// incoming scalar reduction.
|
|
|
|
VectorStart = Builder.CreateInsertElement(Identity,
|
|
|
|
RdxDesc.StartValue, Zero);
|
|
|
|
}
|
2013-05-05 01:54:42 +00:00
|
|
|
}
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
// Fix the vector-loop phi.
|
|
|
|
// We created the induction variable so we know that the
|
|
|
|
// preheader is the first entry.
|
|
|
|
BasicBlock *VecPreheader = Induction->getIncomingBlock(0);
|
|
|
|
|
|
|
|
// Reductions do not have to start at zero. They can start with
|
|
|
|
// any loop invariant values.
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts &VecRdxPhi = WidenMap.get(RdxPhi);
|
|
|
|
BasicBlock *Latch = OrigLoop->getLoopLatch();
|
|
|
|
Value *LoopVal = RdxPhi->getIncomingValueForBlock(Latch);
|
|
|
|
VectorParts &Val = getVectorValue(LoopVal);
|
|
|
|
for (unsigned part = 0; part < UF; ++part) {
|
2013-05-29 03:13:41 +00:00
|
|
|
// Make sure to add the reduction stat value only to the
|
2013-01-03 00:52:27 +00:00
|
|
|
// first unroll part.
|
|
|
|
Value *StartVal = (part == 0) ? VectorStart : Identity;
|
|
|
|
cast<PHINode>(VecRdxPhi[part])->addIncoming(StartVal, VecPreheader);
|
|
|
|
cast<PHINode>(VecRdxPhi[part])->addIncoming(Val[part], LoopVectorBody);
|
|
|
|
}
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
// Before each round, move the insertion point right between
|
|
|
|
// the PHIs and the values we are going to write.
|
|
|
|
// This allows us to write both PHINodes and the extractelement
|
|
|
|
// instructions.
|
|
|
|
Builder.SetInsertPoint(LoopMiddleBlock->getFirstInsertionPt());
|
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts RdxParts;
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, RdxDesc.LoopExitInstr);
|
2013-01-03 00:52:27 +00:00
|
|
|
for (unsigned part = 0; part < UF; ++part) {
|
|
|
|
// This PHINode contains the vectorized reduction variable, or
|
|
|
|
// the initial value vector, if we bypass the vector loop.
|
|
|
|
VectorParts &RdxExitVal = getVectorValue(RdxDesc.LoopExitInstr);
|
|
|
|
PHINode *NewPhi = Builder.CreatePHI(VecTy, 2, "rdx.vec.exit.phi");
|
|
|
|
Value *StartVal = (part == 0) ? VectorStart : Identity;
|
2013-01-19 13:57:58 +00:00
|
|
|
for (unsigned I = 0, E = LoopBypassBlocks.size(); I != E; ++I)
|
|
|
|
NewPhi->addIncoming(StartVal, LoopBypassBlocks[I]);
|
2013-01-03 00:52:27 +00:00
|
|
|
NewPhi->addIncoming(RdxExitVal[part], LoopVectorBody);
|
|
|
|
RdxParts.push_back(NewPhi);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reduce all of the unrolled parts into a single vector.
|
|
|
|
Value *ReducedPartRdx = RdxParts[0];
|
2013-04-18 17:22:34 +00:00
|
|
|
unsigned Op = getReductionBinOp(RdxDesc.Kind);
|
2013-06-28 17:14:48 +00:00
|
|
|
setDebugLocFromInst(Builder, ReducedPartRdx);
|
2013-01-03 00:52:27 +00:00
|
|
|
for (unsigned part = 1; part < UF; ++part) {
|
2013-05-05 01:54:48 +00:00
|
|
|
if (Op != Instruction::ICmp && Op != Instruction::FCmp)
|
2013-04-18 17:22:34 +00:00
|
|
|
ReducedPartRdx = Builder.CreateBinOp((Instruction::BinaryOps)Op,
|
|
|
|
RdxParts[part], ReducedPartRdx,
|
|
|
|
"bin.rdx");
|
|
|
|
else
|
2013-04-19 21:03:36 +00:00
|
|
|
ReducedPartRdx = createMinMaxOp(Builder, RdxDesc.MinMaxKind,
|
2013-04-18 17:22:34 +00:00
|
|
|
ReducedPartRdx, RdxParts[part]);
|
2013-01-03 00:52:27 +00:00
|
|
|
}
|
2012-12-04 06:15:11 +00:00
|
|
|
|
2013-08-27 18:52:47 +00:00
|
|
|
if (VF > 1) {
|
|
|
|
// VF is a power of 2 so we can emit the reduction using log2(VF) shuffles
|
|
|
|
// and vector ops, reducing the set of values being computed by half each
|
|
|
|
// round.
|
|
|
|
assert(isPowerOf2_32(VF) &&
|
|
|
|
"Reduction emission only supported for pow2 vectors!");
|
|
|
|
Value *TmpVec = ReducedPartRdx;
|
|
|
|
SmallVector<Constant*, 32> ShuffleMask(VF, 0);
|
|
|
|
for (unsigned i = VF; i != 1; i >>= 1) {
|
|
|
|
// Move the upper half of the vector to the lower half.
|
|
|
|
for (unsigned j = 0; j != i/2; ++j)
|
|
|
|
ShuffleMask[j] = Builder.getInt32(i/2 + j);
|
|
|
|
|
|
|
|
// Fill the rest of the mask with undef.
|
|
|
|
std::fill(&ShuffleMask[i/2], ShuffleMask.end(),
|
|
|
|
UndefValue::get(Builder.getInt32Ty()));
|
|
|
|
|
|
|
|
Value *Shuf =
|
LoopVectorize: Emit reductions as log2(vectorsize) shuffles + vector ops instead of scalar operations.
For example on x86 with SSE4.2 a <8 x i8> add reduction becomes
movdqa %xmm0, %xmm1
movhlps %xmm1, %xmm1 ## xmm1 = xmm1[1,1]
paddw %xmm0, %xmm1
pshufd $1, %xmm1, %xmm0 ## xmm0 = xmm1[1,0,0,0]
paddw %xmm1, %xmm0
phaddw %xmm0, %xmm0
pextrb $0, %xmm0, %edx
instead of
pextrb $2, %xmm0, %esi
pextrb $0, %xmm0, %edx
addb %sil, %dl
pextrb $4, %xmm0, %esi
addb %dl, %sil
pextrb $6, %xmm0, %edx
addb %sil, %dl
pextrb $8, %xmm0, %esi
addb %dl, %sil
pextrb $10, %xmm0, %edi
pextrb $14, %xmm0, %edx
addb %sil, %dil
pextrb $12, %xmm0, %esi
addb %dil, %sil
addb %sil, %dl
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@170439 91177308-0d34-0410-b5e6-96231b3b80d8
2012-12-18 18:40:20 +00:00
|
|
|
Builder.CreateShuffleVector(TmpVec,
|
|
|
|
UndefValue::get(TmpVec->getType()),
|
|
|
|
ConstantVector::get(ShuffleMask),
|
|
|
|
"rdx.shuf");
|
|
|
|
|
2013-08-27 18:52:47 +00:00
|
|
|
if (Op != Instruction::ICmp && Op != Instruction::FCmp)
|
|
|
|
TmpVec = Builder.CreateBinOp((Instruction::BinaryOps)Op, TmpVec, Shuf,
|
|
|
|
"bin.rdx");
|
|
|
|
else
|
|
|
|
TmpVec = createMinMaxOp(Builder, RdxDesc.MinMaxKind, TmpVec, Shuf);
|
|
|
|
}
|
2012-12-04 06:15:11 +00:00
|
|
|
|
2013-08-27 18:52:47 +00:00
|
|
|
// The result is in the first element of the vector.
|
|
|
|
ReducedPartRdx = Builder.CreateExtractElement(TmpVec,
|
|
|
|
Builder.getInt32(0));
|
|
|
|
}
|
LoopVectorize: Emit reductions as log2(vectorsize) shuffles + vector ops instead of scalar operations.
For example on x86 with SSE4.2 a <8 x i8> add reduction becomes
movdqa %xmm0, %xmm1
movhlps %xmm1, %xmm1 ## xmm1 = xmm1[1,1]
paddw %xmm0, %xmm1
pshufd $1, %xmm1, %xmm0 ## xmm0 = xmm1[1,0,0,0]
paddw %xmm1, %xmm0
phaddw %xmm0, %xmm0
pextrb $0, %xmm0, %edx
instead of
pextrb $2, %xmm0, %esi
pextrb $0, %xmm0, %edx
addb %sil, %dl
pextrb $4, %xmm0, %esi
addb %dl, %sil
pextrb $6, %xmm0, %edx
addb %sil, %dl
pextrb $8, %xmm0, %esi
addb %dl, %sil
pextrb $10, %xmm0, %edi
pextrb $14, %xmm0, %edx
addb %sil, %dil
pextrb $12, %xmm0, %esi
addb %dil, %sil
addb %sil, %dl
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@170439 91177308-0d34-0410-b5e6-96231b3b80d8
2012-12-18 18:40:20 +00:00
|
|
|
|
2012-12-04 06:15:11 +00:00
|
|
|
// Now, we need to fix the users of the reduction variable
|
|
|
|
// inside and outside of the scalar remainder loop.
|
|
|
|
// We know that the loop is in LCSSA form. We need to update the
|
|
|
|
// PHI nodes in the exit blocks.
|
|
|
|
for (BasicBlock::iterator LEI = LoopExitBlock->begin(),
|
|
|
|
LEE = LoopExitBlock->end(); LEI != LEE; ++LEI) {
|
|
|
|
PHINode *LCSSAPhi = dyn_cast<PHINode>(LEI);
|
2013-09-10 18:46:15 +00:00
|
|
|
if (!LCSSAPhi) break;
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
// All PHINodes need to have a single entry edge, or two if
|
|
|
|
// we already fixed them.
|
|
|
|
assert(LCSSAPhi->getNumIncomingValues() < 3 && "Invalid LCSSA PHI");
|
|
|
|
|
|
|
|
// We found our reduction value exit-PHI. Update it with the
|
|
|
|
// incoming bypass edge.
|
|
|
|
if (LCSSAPhi->getIncomingValue(0) == RdxDesc.LoopExitInstr) {
|
|
|
|
// Add an edge coming from the bypass.
|
2013-08-27 18:52:47 +00:00
|
|
|
LCSSAPhi->addIncoming(ReducedPartRdx, LoopMiddleBlock);
|
2012-12-04 06:15:11 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}// end of the LCSSA phi scan.
|
|
|
|
|
|
|
|
// Fix the scalar loop reduction variable with the incoming reduction sum
|
|
|
|
// from the vector body and from the backedge value.
|
2012-12-04 18:17:33 +00:00
|
|
|
int IncomingEdgeBlockIdx =
|
|
|
|
(RdxPhi)->getBasicBlockIndex(OrigLoop->getLoopLatch());
|
|
|
|
assert(IncomingEdgeBlockIdx >= 0 && "Invalid block index");
|
|
|
|
// Pick the other block.
|
|
|
|
int SelfEdgeBlockIdx = (IncomingEdgeBlockIdx ? 0 : 1);
|
2013-08-27 18:52:47 +00:00
|
|
|
(RdxPhi)->setIncomingValue(SelfEdgeBlockIdx, ReducedPartRdx);
|
2012-12-04 06:15:11 +00:00
|
|
|
(RdxPhi)->setIncomingValue(IncomingEdgeBlockIdx, RdxDesc.LoopExitInstr);
|
|
|
|
}// end of for each redux variable.
|
2013-08-26 22:33:26 +00:00
|
|
|
|
|
|
|
fixLCSSAPHIs();
|
|
|
|
}
|
2012-12-30 07:47:00 +00:00
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
void InnerLoopVectorizer::fixLCSSAPHIs() {
|
2012-12-30 07:47:00 +00:00
|
|
|
for (BasicBlock::iterator LEI = LoopExitBlock->begin(),
|
|
|
|
LEE = LoopExitBlock->end(); LEI != LEE; ++LEI) {
|
|
|
|
PHINode *LCSSAPhi = dyn_cast<PHINode>(LEI);
|
2013-09-10 18:46:15 +00:00
|
|
|
if (!LCSSAPhi) break;
|
2012-12-30 07:47:00 +00:00
|
|
|
if (LCSSAPhi->getNumIncomingValues() == 1)
|
|
|
|
LCSSAPhi->addIncoming(UndefValue::get(LCSSAPhi->getType()),
|
|
|
|
LoopMiddleBlock);
|
|
|
|
}
|
2013-08-26 22:33:26 +00:00
|
|
|
}
|
2012-12-04 06:15:11 +00:00
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
InnerLoopVectorizer::VectorParts
|
|
|
|
InnerLoopVectorizer::createEdgeMask(BasicBlock *Src, BasicBlock *Dst) {
|
2012-12-04 06:15:11 +00:00
|
|
|
assert(std::find(pred_begin(Dst), pred_end(Dst), Src) != pred_end(Dst) &&
|
|
|
|
"Invalid edge");
|
|
|
|
|
2013-06-27 20:31:06 +00:00
|
|
|
// Look for cached value.
|
|
|
|
std::pair<BasicBlock*, BasicBlock*> Edge(Src, Dst);
|
|
|
|
EdgeMaskCache::iterator ECEntryIt = MaskCache.find(Edge);
|
|
|
|
if (ECEntryIt != MaskCache.end())
|
|
|
|
return ECEntryIt->second;
|
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts SrcMask = createBlockInMask(Src);
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
// The terminator has to be a branch inst!
|
|
|
|
BranchInst *BI = dyn_cast<BranchInst>(Src->getTerminator());
|
|
|
|
assert(BI && "Unexpected terminator found");
|
|
|
|
|
|
|
|
if (BI->isConditional()) {
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts EdgeMask = getVectorValue(BI->getCondition());
|
|
|
|
|
2012-12-04 06:15:11 +00:00
|
|
|
if (BI->getSuccessor(0) != Dst)
|
2013-01-03 00:52:27 +00:00
|
|
|
for (unsigned part = 0; part < UF; ++part)
|
|
|
|
EdgeMask[part] = Builder.CreateNot(EdgeMask[part]);
|
|
|
|
|
|
|
|
for (unsigned part = 0; part < UF; ++part)
|
|
|
|
EdgeMask[part] = Builder.CreateAnd(EdgeMask[part], SrcMask[part]);
|
2013-06-27 20:31:06 +00:00
|
|
|
|
|
|
|
MaskCache[Edge] = EdgeMask;
|
2013-01-03 00:52:27 +00:00
|
|
|
return EdgeMask;
|
2012-12-04 06:15:11 +00:00
|
|
|
}
|
|
|
|
|
2013-06-27 20:31:06 +00:00
|
|
|
MaskCache[Edge] = SrcMask;
|
2013-01-03 00:52:27 +00:00
|
|
|
return SrcMask;
|
2012-12-04 06:15:11 +00:00
|
|
|
}
|
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
InnerLoopVectorizer::VectorParts
|
|
|
|
InnerLoopVectorizer::createBlockInMask(BasicBlock *BB) {
|
2012-12-04 06:15:11 +00:00
|
|
|
assert(OrigLoop->contains(BB) && "Block is not a part of a loop");
|
|
|
|
|
|
|
|
// Loop incoming mask is all-one.
|
2012-12-10 21:39:02 +00:00
|
|
|
if (OrigLoop->getHeader() == BB) {
|
|
|
|
Value *C = ConstantInt::get(IntegerType::getInt1Ty(BB->getContext()), 1);
|
|
|
|
return getVectorValue(C);
|
|
|
|
}
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
// This is the block mask. We OR all incoming edges, and with zero.
|
2012-12-10 21:39:02 +00:00
|
|
|
Value *Zero = ConstantInt::get(IntegerType::getInt1Ty(BB->getContext()), 0);
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts BlockMask = getVectorValue(Zero);
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
// For each pred:
|
2013-01-03 00:52:27 +00:00
|
|
|
for (pred_iterator it = pred_begin(BB), e = pred_end(BB); it != e; ++it) {
|
|
|
|
VectorParts EM = createEdgeMask(*it, BB);
|
|
|
|
for (unsigned part = 0; part < UF; ++part)
|
|
|
|
BlockMask[part] = Builder.CreateOr(BlockMask[part], EM[part]);
|
|
|
|
}
|
2012-12-04 06:15:11 +00:00
|
|
|
|
|
|
|
return BlockMask;
|
|
|
|
}
|
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
void InnerLoopVectorizer::widenPHIInstruction(Instruction *PN,
|
|
|
|
InnerLoopVectorizer::VectorParts &Entry,
|
|
|
|
LoopVectorizationLegality *Legal,
|
|
|
|
unsigned UF, unsigned VF, PhiVector *PV) {
|
|
|
|
PHINode* P = cast<PHINode>(PN);
|
|
|
|
// Handle reduction variables:
|
|
|
|
if (Legal->getReductionVars()->count(P)) {
|
|
|
|
for (unsigned part = 0; part < UF; ++part) {
|
|
|
|
// This is phase one of vectorizing PHIs.
|
|
|
|
Type *VecTy = (VF == 1) ? PN->getType() :
|
|
|
|
VectorType::get(PN->getType(), VF);
|
|
|
|
Entry[part] = PHINode::Create(VecTy, 2, "vec.phi",
|
|
|
|
LoopVectorBody-> getFirstInsertionPt());
|
|
|
|
}
|
|
|
|
PV->push_back(P);
|
|
|
|
return;
|
|
|
|
}
|
2012-11-25 08:41:35 +00:00
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
setDebugLocFromInst(Builder, P);
|
|
|
|
// Check for PHI nodes that are lowered to vector selects.
|
|
|
|
if (P->getParent() != OrigLoop->getHeader()) {
|
|
|
|
// We know that all PHIs in non header blocks are converted into
|
|
|
|
// selects, so we don't have to worry about the insertion order and we
|
|
|
|
// can just use the builder.
|
|
|
|
// At this point we generate the predication tree. There may be
|
|
|
|
// duplications since this is a simple recursive scan, but future
|
|
|
|
// optimizations will clean it up.
|
|
|
|
|
|
|
|
unsigned NumIncoming = P->getNumIncomingValues();
|
|
|
|
|
|
|
|
// Generate a sequence of selects of the form:
|
|
|
|
// SELECT(Mask3, In3,
|
|
|
|
// SELECT(Mask2, In2,
|
|
|
|
// ( ...)))
|
|
|
|
for (unsigned In = 0; In < NumIncoming; In++) {
|
|
|
|
VectorParts Cond = createEdgeMask(P->getIncomingBlock(In),
|
|
|
|
P->getParent());
|
|
|
|
VectorParts &In0 = getVectorValue(P->getIncomingValue(In));
|
|
|
|
|
|
|
|
for (unsigned part = 0; part < UF; ++part) {
|
|
|
|
// We might have single edge PHIs (blocks) - use an identity
|
|
|
|
// 'select' for the first PHI operand.
|
|
|
|
if (In == 0)
|
|
|
|
Entry[part] = Builder.CreateSelect(Cond[part], In0[part],
|
|
|
|
In0[part]);
|
|
|
|
else
|
|
|
|
// Select between the current value and the previous incoming edge
|
|
|
|
// based on the incoming mask.
|
|
|
|
Entry[part] = Builder.CreateSelect(Cond[part], In0[part],
|
|
|
|
Entry[part], "predphi");
|
2012-12-10 21:39:02 +00:00
|
|
|
}
|
2013-08-26 22:33:26 +00:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2012-12-10 19:25:06 +00:00
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
// This PHINode must be an induction variable.
|
|
|
|
// Make sure that we know about it.
|
|
|
|
assert(Legal->getInductionVars()->count(P) &&
|
|
|
|
"Not an induction variable");
|
|
|
|
|
|
|
|
LoopVectorizationLegality::InductionInfo II =
|
|
|
|
Legal->getInductionVars()->lookup(P);
|
|
|
|
|
|
|
|
switch (II.IK) {
|
|
|
|
case LoopVectorizationLegality::IK_NoInduction:
|
|
|
|
llvm_unreachable("Unknown induction");
|
|
|
|
case LoopVectorizationLegality::IK_IntInduction: {
|
|
|
|
assert(P->getType() == II.StartValue->getType() && "Types must match");
|
|
|
|
Type *PhiTy = P->getType();
|
|
|
|
Value *Broadcasted;
|
|
|
|
if (P == OldInduction) {
|
|
|
|
// Handle the canonical induction variable. We might have had to
|
|
|
|
// extend the type.
|
|
|
|
Broadcasted = Builder.CreateTrunc(Induction, PhiTy);
|
|
|
|
} else {
|
|
|
|
// Handle other induction variables that are now based on the
|
|
|
|
// canonical one.
|
|
|
|
Value *NormalizedIdx = Builder.CreateSub(Induction, ExtendedIdx,
|
|
|
|
"normalized.idx");
|
|
|
|
NormalizedIdx = Builder.CreateSExtOrTrunc(NormalizedIdx, PhiTy);
|
|
|
|
Broadcasted = Builder.CreateAdd(II.StartValue, NormalizedIdx,
|
|
|
|
"offset.idx");
|
|
|
|
}
|
|
|
|
Broadcasted = getBroadcastInstrs(Broadcasted);
|
|
|
|
// After broadcasting the induction variable we need to make the vector
|
|
|
|
// consecutive by adding 0, 1, 2, etc.
|
|
|
|
for (unsigned part = 0; part < UF; ++part)
|
|
|
|
Entry[part] = getConsecutiveVector(Broadcasted, VF * part, false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case LoopVectorizationLegality::IK_ReverseIntInduction:
|
|
|
|
case LoopVectorizationLegality::IK_PtrInduction:
|
|
|
|
case LoopVectorizationLegality::IK_ReversePtrInduction:
|
|
|
|
// Handle reverse integer and pointer inductions.
|
|
|
|
Value *StartIdx = ExtendedIdx;
|
|
|
|
// This is the normalized GEP that starts counting at zero.
|
|
|
|
Value *NormalizedIdx = Builder.CreateSub(Induction, StartIdx,
|
|
|
|
"normalized.idx");
|
|
|
|
|
|
|
|
// Handle the reverse integer induction variable case.
|
|
|
|
if (LoopVectorizationLegality::IK_ReverseIntInduction == II.IK) {
|
|
|
|
IntegerType *DstTy = cast<IntegerType>(II.StartValue->getType());
|
|
|
|
Value *CNI = Builder.CreateSExtOrTrunc(NormalizedIdx, DstTy,
|
|
|
|
"resize.norm.idx");
|
|
|
|
Value *ReverseInd = Builder.CreateSub(II.StartValue, CNI,
|
|
|
|
"reverse.idx");
|
|
|
|
|
|
|
|
// This is a new value so do not hoist it out.
|
|
|
|
Value *Broadcasted = getBroadcastInstrs(ReverseInd);
|
|
|
|
// After broadcasting the induction variable we need to make the
|
|
|
|
// vector consecutive by adding ... -3, -2, -1, 0.
|
2013-01-03 00:52:27 +00:00
|
|
|
for (unsigned part = 0; part < UF; ++part)
|
2013-08-26 22:33:26 +00:00
|
|
|
Entry[part] = getConsecutiveVector(Broadcasted, -(int)VF * part,
|
|
|
|
true);
|
|
|
|
return;
|
2012-12-10 21:39:02 +00:00
|
|
|
}
|
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
// Handle the pointer induction variable case.
|
|
|
|
assert(P->getType()->isPointerTy() && "Unexpected type.");
|
|
|
|
|
|
|
|
// Is this a reverse induction ptr or a consecutive induction ptr.
|
|
|
|
bool Reverse = (LoopVectorizationLegality::IK_ReversePtrInduction ==
|
|
|
|
II.IK);
|
|
|
|
|
|
|
|
// This is the vector of results. Notice that we don't generate
|
|
|
|
// vector geps because scalar geps result in better code.
|
|
|
|
for (unsigned part = 0; part < UF; ++part) {
|
|
|
|
if (VF == 1) {
|
|
|
|
int EltIndex = (part) * (Reverse ? -1 : 1);
|
|
|
|
Constant *Idx = ConstantInt::get(Induction->getType(), EltIndex);
|
|
|
|
Value *GlobalIdx;
|
|
|
|
if (Reverse)
|
|
|
|
GlobalIdx = Builder.CreateSub(Idx, NormalizedIdx, "gep.ridx");
|
|
|
|
else
|
|
|
|
GlobalIdx = Builder.CreateAdd(NormalizedIdx, Idx, "gep.idx");
|
|
|
|
|
|
|
|
Value *SclrGep = Builder.CreateGEP(II.StartValue, GlobalIdx,
|
|
|
|
"next.gep");
|
|
|
|
Entry[part] = SclrGep;
|
2012-11-17 00:27:03 +00:00
|
|
|
continue;
|
|
|
|
}
|
2012-12-10 19:25:06 +00:00
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
Value *VecVal = UndefValue::get(VectorType::get(P->getType(), VF));
|
|
|
|
for (unsigned int i = 0; i < VF; ++i) {
|
|
|
|
int EltIndex = (i + part * VF) * (Reverse ? -1 : 1);
|
|
|
|
Constant *Idx = ConstantInt::get(Induction->getType(), EltIndex);
|
|
|
|
Value *GlobalIdx;
|
|
|
|
if (!Reverse)
|
|
|
|
GlobalIdx = Builder.CreateAdd(NormalizedIdx, Idx, "gep.idx");
|
|
|
|
else
|
|
|
|
GlobalIdx = Builder.CreateSub(Idx, NormalizedIdx, "gep.ridx");
|
|
|
|
|
|
|
|
Value *SclrGep = Builder.CreateGEP(II.StartValue, GlobalIdx,
|
|
|
|
"next.gep");
|
|
|
|
VecVal = Builder.CreateInsertElement(VecVal, SclrGep,
|
|
|
|
Builder.getInt32(i),
|
|
|
|
"insert.gep");
|
2012-11-25 08:41:35 +00:00
|
|
|
}
|
2013-08-26 22:33:26 +00:00
|
|
|
Entry[part] = VecVal;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
2013-08-26 22:33:26 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
void
|
|
|
|
InnerLoopVectorizer::vectorizeBlockInLoop(LoopVectorizationLegality *Legal,
|
|
|
|
BasicBlock *BB, PhiVector *PV) {
|
|
|
|
// For each instruction in the old loop.
|
|
|
|
for (BasicBlock::iterator it = BB->begin(), e = BB->end(); it != e; ++it) {
|
|
|
|
VectorParts &Entry = WidenMap.get(it);
|
|
|
|
switch (it->getOpcode()) {
|
|
|
|
case Instruction::Br:
|
|
|
|
// Nothing to do for PHIs and BR, since we already took care of the
|
|
|
|
// loop control flow instructions.
|
|
|
|
continue;
|
|
|
|
case Instruction::PHI:{
|
|
|
|
// Vectorize PHINodes.
|
|
|
|
widenPHIInstruction(it, Entry, Legal, UF, VF, PV);
|
|
|
|
continue;
|
2012-12-10 21:39:02 +00:00
|
|
|
}// End of PHI.
|
|
|
|
|
|
|
|
case Instruction::Add:
|
|
|
|
case Instruction::FAdd:
|
|
|
|
case Instruction::Sub:
|
|
|
|
case Instruction::FSub:
|
|
|
|
case Instruction::Mul:
|
|
|
|
case Instruction::FMul:
|
|
|
|
case Instruction::UDiv:
|
|
|
|
case Instruction::SDiv:
|
|
|
|
case Instruction::FDiv:
|
|
|
|
case Instruction::URem:
|
|
|
|
case Instruction::SRem:
|
|
|
|
case Instruction::FRem:
|
|
|
|
case Instruction::Shl:
|
|
|
|
case Instruction::LShr:
|
|
|
|
case Instruction::AShr:
|
|
|
|
case Instruction::And:
|
|
|
|
case Instruction::Or:
|
|
|
|
case Instruction::Xor: {
|
|
|
|
// Just widen binops.
|
|
|
|
BinaryOperator *BinOp = dyn_cast<BinaryOperator>(it);
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, BinOp);
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts &A = getVectorValue(it->getOperand(0));
|
|
|
|
VectorParts &B = getVectorValue(it->getOperand(1));
|
2012-12-10 21:39:02 +00:00
|
|
|
|
|
|
|
// Use this vector value for all users of the original instruction.
|
2013-01-03 00:52:27 +00:00
|
|
|
for (unsigned Part = 0; Part < UF; ++Part) {
|
|
|
|
Value *V = Builder.CreateBinOp(BinOp->getOpcode(), A[Part], B[Part]);
|
|
|
|
|
2013-01-10 17:34:39 +00:00
|
|
|
// Update the NSW, NUW and Exact flags. Notice: V can be an Undef.
|
|
|
|
BinaryOperator *VecOp = dyn_cast<BinaryOperator>(V);
|
|
|
|
if (VecOp && isa<OverflowingBinaryOperator>(BinOp)) {
|
2013-01-03 00:52:27 +00:00
|
|
|
VecOp->setHasNoSignedWrap(BinOp->hasNoSignedWrap());
|
|
|
|
VecOp->setHasNoUnsignedWrap(BinOp->hasNoUnsignedWrap());
|
|
|
|
}
|
2013-01-10 17:34:39 +00:00
|
|
|
if (VecOp && isa<PossiblyExactOperator>(VecOp))
|
2013-01-03 00:52:27 +00:00
|
|
|
VecOp->setIsExact(BinOp->isExact());
|
|
|
|
|
|
|
|
Entry[Part] = V;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
2012-12-10 21:39:02 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Instruction::Select: {
|
|
|
|
// Widen selects.
|
|
|
|
// If the selector is loop invariant we can create a select
|
|
|
|
// instruction with a scalar condition. Otherwise, use vector-select.
|
2013-01-03 00:52:27 +00:00
|
|
|
bool InvariantCond = SE->isLoopInvariant(SE->getSCEV(it->getOperand(0)),
|
|
|
|
OrigLoop);
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, it);
|
2012-12-10 21:39:02 +00:00
|
|
|
|
|
|
|
// The condition can be loop invariant but still defined inside the
|
|
|
|
// loop. This means that we can't just use the original 'cond' value.
|
|
|
|
// We have to take the 'vectorized' value and pick the first lane.
|
|
|
|
// Instcombine will make this a no-op.
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts &Cond = getVectorValue(it->getOperand(0));
|
|
|
|
VectorParts &Op0 = getVectorValue(it->getOperand(1));
|
|
|
|
VectorParts &Op1 = getVectorValue(it->getOperand(2));
|
2013-08-26 22:33:26 +00:00
|
|
|
|
|
|
|
Value *ScalarCond = (VF == 1) ? Cond[0] :
|
|
|
|
Builder.CreateExtractElement(Cond[0], Builder.getInt32(0));
|
|
|
|
|
2013-01-03 00:52:27 +00:00
|
|
|
for (unsigned Part = 0; Part < UF; ++Part) {
|
|
|
|
Entry[Part] = Builder.CreateSelect(
|
|
|
|
InvariantCond ? ScalarCond : Cond[Part],
|
|
|
|
Op0[Part],
|
|
|
|
Op1[Part]);
|
|
|
|
}
|
2012-12-10 21:39:02 +00:00
|
|
|
break;
|
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::ICmp:
|
|
|
|
case Instruction::FCmp: {
|
|
|
|
// Widen compares. Generate vector compares.
|
|
|
|
bool FCmp = (it->getOpcode() == Instruction::FCmp);
|
|
|
|
CmpInst *Cmp = dyn_cast<CmpInst>(it);
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, it);
|
2013-01-03 00:52:27 +00:00
|
|
|
VectorParts &A = getVectorValue(it->getOperand(0));
|
|
|
|
VectorParts &B = getVectorValue(it->getOperand(1));
|
|
|
|
for (unsigned Part = 0; Part < UF; ++Part) {
|
|
|
|
Value *C = 0;
|
|
|
|
if (FCmp)
|
|
|
|
C = Builder.CreateFCmp(Cmp->getPredicate(), A[Part], B[Part]);
|
|
|
|
else
|
|
|
|
C = Builder.CreateICmp(Cmp->getPredicate(), A[Part], B[Part]);
|
|
|
|
Entry[Part] = C;
|
|
|
|
}
|
2012-12-10 21:39:02 +00:00
|
|
|
break;
|
|
|
|
}
|
2012-11-09 07:09:44 +00:00
|
|
|
|
2013-01-25 21:47:42 +00:00
|
|
|
case Instruction::Store:
|
|
|
|
case Instruction::Load:
|
|
|
|
vectorizeMemoryInstruction(it, Legal);
|
2012-10-17 18:25:06 +00:00
|
|
|
break;
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::ZExt:
|
|
|
|
case Instruction::SExt:
|
|
|
|
case Instruction::FPToUI:
|
|
|
|
case Instruction::FPToSI:
|
|
|
|
case Instruction::FPExt:
|
|
|
|
case Instruction::PtrToInt:
|
|
|
|
case Instruction::IntToPtr:
|
|
|
|
case Instruction::SIToFP:
|
|
|
|
case Instruction::UIToFP:
|
|
|
|
case Instruction::Trunc:
|
|
|
|
case Instruction::FPTrunc:
|
|
|
|
case Instruction::BitCast: {
|
|
|
|
CastInst *CI = dyn_cast<CastInst>(it);
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, it);
|
2012-12-11 18:58:10 +00:00
|
|
|
/// Optimize the special case where the source is the induction
|
|
|
|
/// variable. Notice that we can only optimize the 'trunc' case
|
|
|
|
/// because: a. FP conversions lose precision, b. sext/zext may wrap,
|
|
|
|
/// c. other casts depend on pointer size.
|
|
|
|
if (CI->getOperand(0) == OldInduction &&
|
|
|
|
it->getOpcode() == Instruction::Trunc) {
|
|
|
|
Value *ScalarCast = Builder.CreateCast(CI->getOpcode(), Induction,
|
|
|
|
CI->getType());
|
|
|
|
Value *Broadcasted = getBroadcastInstrs(ScalarCast);
|
2013-01-03 00:52:27 +00:00
|
|
|
for (unsigned Part = 0; Part < UF; ++Part)
|
|
|
|
Entry[Part] = getConsecutiveVector(Broadcasted, VF * Part, false);
|
2012-12-11 18:58:10 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
/// Vectorize casts.
|
2013-08-26 22:33:26 +00:00
|
|
|
Type *DestTy = (VF == 1) ? CI->getType() :
|
|
|
|
VectorType::get(CI->getType(), VF);
|
2013-01-03 00:52:27 +00:00
|
|
|
|
|
|
|
VectorParts &A = getVectorValue(it->getOperand(0));
|
|
|
|
for (unsigned Part = 0; Part < UF; ++Part)
|
|
|
|
Entry[Part] = Builder.CreateCast(CI->getOpcode(), A[Part], DestTy);
|
2012-12-10 21:39:02 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Instruction::Call: {
|
2013-03-09 16:27:27 +00:00
|
|
|
// Ignore dbg intrinsics.
|
|
|
|
if (isa<DbgInfoIntrinsic>(it))
|
2013-03-09 15:56:34 +00:00
|
|
|
break;
|
2013-06-28 16:26:54 +00:00
|
|
|
setDebugLocFromInst(Builder, it);
|
2013-06-28 00:38:54 +00:00
|
|
|
|
2012-12-10 21:39:02 +00:00
|
|
|
Module *M = BB->getParent()->getParent();
|
2013-02-27 15:24:19 +00:00
|
|
|
CallInst *CI = cast<CallInst>(it);
|
|
|
|
Intrinsic::ID ID = getIntrinsicIDForCall(CI, TLI);
|
|
|
|
assert(ID && "Not an intrinsic call!");
|
2013-08-06 22:37:52 +00:00
|
|
|
switch (ID) {
|
|
|
|
case Intrinsic::lifetime_end:
|
|
|
|
case Intrinsic::lifetime_start:
|
|
|
|
scalarizeInstruction(it);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
for (unsigned Part = 0; Part < UF; ++Part) {
|
|
|
|
SmallVector<Value *, 4> Args;
|
|
|
|
for (unsigned i = 0, ie = CI->getNumArgOperands(); i != ie; ++i) {
|
|
|
|
VectorParts &Arg = getVectorValue(CI->getArgOperand(i));
|
|
|
|
Args.push_back(Arg[Part]);
|
|
|
|
}
|
2013-08-26 22:33:26 +00:00
|
|
|
Type *Tys[] = {CI->getType()};
|
|
|
|
if (VF > 1)
|
|
|
|
Tys[0] = VectorType::get(CI->getType()->getScalarType(), VF);
|
|
|
|
|
2013-08-06 22:37:52 +00:00
|
|
|
Function *F = Intrinsic::getDeclaration(M, ID, Tys);
|
|
|
|
Entry[Part] = Builder.CreateCall(F, Args);
|
2013-01-03 00:52:27 +00:00
|
|
|
}
|
2013-08-06 22:37:52 +00:00
|
|
|
break;
|
2013-01-03 00:52:27 +00:00
|
|
|
}
|
2012-12-10 21:39:02 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
// All other instructions are unsupported. Scalarize them.
|
|
|
|
scalarizeInstruction(it);
|
|
|
|
break;
|
2012-10-17 18:25:06 +00:00
|
|
|
}// end of switch.
|
|
|
|
}// end of for_each instr.
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:33:08 +00:00
|
|
|
void InnerLoopVectorizer::updateAnalysis() {
|
2012-11-29 19:25:41 +00:00
|
|
|
// Forget the original basic block.
|
2012-10-22 04:53:05 +00:00
|
|
|
SE->forgetLoop(OrigLoop);
|
2012-10-29 21:52:38 +00:00
|
|
|
|
|
|
|
// Update the dominator tree information.
|
2013-01-19 13:57:58 +00:00
|
|
|
assert(DT->properlyDominates(LoopBypassBlocks.front(), LoopExitBlock) &&
|
2012-10-29 21:52:38 +00:00
|
|
|
"Entry does not dominate exit.");
|
|
|
|
|
2013-01-19 13:57:58 +00:00
|
|
|
for (unsigned I = 1, E = LoopBypassBlocks.size(); I != E; ++I)
|
|
|
|
DT->addNewBlock(LoopBypassBlocks[I], LoopBypassBlocks[I-1]);
|
|
|
|
DT->addNewBlock(LoopVectorPreHeader, LoopBypassBlocks.back());
|
2012-10-29 21:52:38 +00:00
|
|
|
DT->addNewBlock(LoopVectorBody, LoopVectorPreHeader);
|
2013-01-19 13:57:58 +00:00
|
|
|
DT->addNewBlock(LoopMiddleBlock, LoopBypassBlocks.front());
|
2012-10-29 21:52:38 +00:00
|
|
|
DT->addNewBlock(LoopScalarPreHeader, LoopMiddleBlock);
|
|
|
|
DT->changeImmediateDominator(LoopScalarBody, LoopScalarPreHeader);
|
|
|
|
DT->changeImmediateDominator(LoopExitBlock, LoopMiddleBlock);
|
|
|
|
|
|
|
|
DEBUG(DT->verifyAnalysis());
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
bool LoopVectorizationLegality::canVectorizeWithIfConvert() {
|
|
|
|
if (!EnableIfConversion)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
assert(TheLoop->getNumBlocks() > 1 && "Single block loops are vectorizable");
|
|
|
|
std::vector<BasicBlock*> &LoopBlocks = TheLoop->getBlocksVector();
|
|
|
|
|
2013-06-28 20:46:27 +00:00
|
|
|
// A list of pointers that we can safely read and write to.
|
|
|
|
SmallPtrSet<Value *, 8> SafePointes;
|
|
|
|
|
|
|
|
// Collect safe addresses.
|
|
|
|
for (unsigned i = 0, e = LoopBlocks.size(); i < e; ++i) {
|
|
|
|
BasicBlock *BB = LoopBlocks[i];
|
|
|
|
|
|
|
|
if (blockNeedsPredication(BB))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (BasicBlock::iterator I = BB->begin(), E = BB->end(); I != E; ++I) {
|
|
|
|
if (LoadInst *LI = dyn_cast<LoadInst>(I))
|
|
|
|
SafePointes.insert(LI->getPointerOperand());
|
|
|
|
else if (StoreInst *SI = dyn_cast<StoreInst>(I))
|
|
|
|
SafePointes.insert(SI->getPointerOperand());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// Collect the blocks that need predication.
|
|
|
|
for (unsigned i = 0, e = LoopBlocks.size(); i < e; ++i) {
|
|
|
|
BasicBlock *BB = LoopBlocks[i];
|
|
|
|
|
2012-12-11 04:55:10 +00:00
|
|
|
// We don't support switch statements inside loops.
|
|
|
|
if (!isa<BranchInst>(BB->getTerminator()))
|
|
|
|
return false;
|
|
|
|
|
2012-12-03 22:57:09 +00:00
|
|
|
// We must be able to predicate all blocks that need to be predicated.
|
2013-06-28 20:46:27 +00:00
|
|
|
if (blockNeedsPredication(BB) && !blockCanBePredicated(BB, SafePointes))
|
2012-12-03 21:06:35 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can if-convert this loop.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-10-24 20:36:32 +00:00
|
|
|
bool LoopVectorizationLegality::canVectorize() {
|
2013-05-24 18:05:35 +00:00
|
|
|
// We must have a loop in canonical form. Loops with indirectbr in them cannot
|
|
|
|
// be canonicalized.
|
|
|
|
if (!TheLoop->getLoopPreheader())
|
|
|
|
return false;
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// We can only vectorize innermost loops.
|
|
|
|
if (TheLoop->getSubLoopsVector().size())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// We must have a single backedge.
|
|
|
|
if (TheLoop->getNumBackEdges() != 1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// We must have a single exiting block.
|
|
|
|
if (!TheLoop->getExitingBlock())
|
|
|
|
return false;
|
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
unsigned NumBlocks = TheLoop->getNumBlocks();
|
2012-12-03 21:06:35 +00:00
|
|
|
|
|
|
|
// Check if we can if-convert non single-bb loops.
|
|
|
|
if (NumBlocks != 1 && !canVectorizeWithIfConvert()) {
|
|
|
|
DEBUG(dbgs() << "LV: Can't if-convert the loop.\n");
|
2012-10-24 20:36:32 +00:00
|
|
|
return false;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// We need to have a loop header.
|
2012-12-03 21:06:35 +00:00
|
|
|
BasicBlock *Latch = TheLoop->getLoopLatch();
|
2012-12-04 00:49:34 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a loop: " <<
|
|
|
|
TheLoop->getHeader()->getName() << "\n");
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2012-10-18 05:29:12 +00:00
|
|
|
// ScalarEvolution needs to be able to find the exit count.
|
2013-05-31 21:48:56 +00:00
|
|
|
const SCEV *ExitCount = SE->getBackedgeTakenCount(TheLoop);
|
2012-10-18 05:29:12 +00:00
|
|
|
if (ExitCount == SE->getCouldNotCompute()) {
|
|
|
|
DEBUG(dbgs() << "LV: SCEV could not compute the loop exit count.\n");
|
2012-10-24 20:36:32 +00:00
|
|
|
return false;
|
2012-10-18 05:29:12 +00:00
|
|
|
}
|
|
|
|
|
2012-10-31 03:31:07 +00:00
|
|
|
// Do not loop-vectorize loops with a tiny trip count.
|
2012-12-03 21:06:35 +00:00
|
|
|
unsigned TC = SE->getSmallConstantTripCount(TheLoop, Latch);
|
2013-01-07 21:54:51 +00:00
|
|
|
if (TC > 0u && TC < TinyTripCountVectorThreshold) {
|
2012-10-31 03:31:07 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a loop with a very small trip count. " <<
|
|
|
|
"This loop is not worth vectorizing.\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// Check if we can vectorize the instructions and CFG in this loop.
|
2012-12-03 22:57:09 +00:00
|
|
|
if (!canVectorizeInstrs()) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Can't vectorize the instructions or CFG\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-11-09 07:09:44 +00:00
|
|
|
// Go over each instruction and look at memory deps.
|
2012-12-03 21:06:35 +00:00
|
|
|
if (!canVectorizeMemory()) {
|
|
|
|
DEBUG(dbgs() << "LV: Can't vectorize due to memory conflicts\n");
|
2012-11-09 07:09:44 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-12-03 22:57:09 +00:00
|
|
|
// Collect all of the variables that remain uniform after vectorization.
|
2012-12-03 21:06:35 +00:00
|
|
|
collectLoopUniforms();
|
|
|
|
|
2012-11-09 07:09:44 +00:00
|
|
|
DEBUG(dbgs() << "LV: We can vectorize this loop" <<
|
|
|
|
(PtrRtCheck.Need ? " (with a runtime bound check)" : "")
|
|
|
|
<<"!\n");
|
2012-10-18 05:29:12 +00:00
|
|
|
|
|
|
|
// Okay! We can vectorize. At this point we don't have any other mem analysis
|
2012-10-24 20:36:32 +00:00
|
|
|
// which may limit our maximum vectorization factor, so just return true with
|
|
|
|
// no restrictions.
|
|
|
|
return true;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2013-05-11 23:04:28 +00:00
|
|
|
static Type *convertPointerToIntegerType(DataLayout &DL, Type *Ty) {
|
|
|
|
if (Ty->isPointerTy())
|
2013-08-22 02:42:55 +00:00
|
|
|
return DL.getIntPtrType(Ty);
|
|
|
|
|
2013-05-11 23:04:28 +00:00
|
|
|
return Ty;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Type* getWiderType(DataLayout &DL, Type *Ty0, Type *Ty1) {
|
|
|
|
Ty0 = convertPointerToIntegerType(DL, Ty0);
|
|
|
|
Ty1 = convertPointerToIntegerType(DL, Ty1);
|
|
|
|
if (Ty0->getScalarSizeInBits() > Ty1->getScalarSizeInBits())
|
|
|
|
return Ty0;
|
|
|
|
return Ty1;
|
|
|
|
}
|
|
|
|
|
2013-05-31 19:53:50 +00:00
|
|
|
/// \brief Check that the instruction has outside loop users and is not an
|
|
|
|
/// identified reduction variable.
|
|
|
|
static bool hasOutsideLoopUser(const Loop *TheLoop, Instruction *Inst,
|
|
|
|
SmallPtrSet<Value *, 4> &Reductions) {
|
|
|
|
// Reduction instructions are allowed to have exit users. All other
|
|
|
|
// instructions must not have external users.
|
|
|
|
if (!Reductions.count(Inst))
|
|
|
|
//Check that all of the users of the loop are inside the BB.
|
|
|
|
for (Value::use_iterator I = Inst->use_begin(), E = Inst->use_end();
|
|
|
|
I != E; ++I) {
|
|
|
|
Instruction *U = cast<Instruction>(*I);
|
|
|
|
// This user may be a reduction exit value.
|
|
|
|
if (!TheLoop->contains(U)) {
|
|
|
|
DEBUG(dbgs() << "LV: Found an outside user for : "<< *U << "\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-12-03 22:57:09 +00:00
|
|
|
bool LoopVectorizationLegality::canVectorizeInstrs() {
|
2012-11-17 00:27:03 +00:00
|
|
|
BasicBlock *PreHeader = TheLoop->getLoopPreheader();
|
2012-12-03 21:06:35 +00:00
|
|
|
BasicBlock *Header = TheLoop->getHeader();
|
2012-11-17 00:27:03 +00:00
|
|
|
|
2013-05-05 01:54:48 +00:00
|
|
|
// Look for the attribute signaling the absence of NaNs.
|
|
|
|
Function &F = *Header->getParent();
|
|
|
|
if (F.hasFnAttribute("no-nans-fp-math"))
|
|
|
|
HasFunNoNaNAttr = F.getAttributes().getAttribute(
|
|
|
|
AttributeSet::FunctionIndex,
|
|
|
|
"no-nans-fp-math").getValueAsString() == "true";
|
|
|
|
|
2012-12-03 22:57:09 +00:00
|
|
|
// For each block in the loop.
|
2012-12-03 21:06:35 +00:00
|
|
|
for (Loop::block_iterator bb = TheLoop->block_begin(),
|
|
|
|
be = TheLoop->block_end(); bb != be; ++bb) {
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// Scan the instructions in the block and look for hazards.
|
2012-12-03 22:57:09 +00:00
|
|
|
for (BasicBlock::iterator it = (*bb)->begin(), e = (*bb)->end(); it != e;
|
|
|
|
++it) {
|
2012-11-17 00:27:03 +00:00
|
|
|
|
2012-12-03 22:57:09 +00:00
|
|
|
if (PHINode *Phi = dyn_cast<PHINode>(it)) {
|
2013-05-11 23:04:26 +00:00
|
|
|
Type *PhiTy = Phi->getType();
|
2012-12-10 19:25:06 +00:00
|
|
|
// Check that this PHI type is allowed.
|
2013-05-11 23:04:26 +00:00
|
|
|
if (!PhiTy->isIntegerTy() &&
|
|
|
|
!PhiTy->isFloatingPointTy() &&
|
|
|
|
!PhiTy->isPointerTy()) {
|
2012-12-10 19:25:06 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found an non-int non-pointer PHI.\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// If this PHINode is not in the header block, then we know that we
|
2012-12-10 19:25:06 +00:00
|
|
|
// can convert it to select during if-conversion. No need to check if
|
|
|
|
// the PHIs in this block are induction or reduction variables.
|
2013-05-31 19:53:50 +00:00
|
|
|
if (*bb != Header) {
|
|
|
|
// Check that this instruction has no outside users or is an
|
|
|
|
// identified reduction value with an outside user.
|
|
|
|
if(!hasOutsideLoopUser(TheLoop, it, AllowedExit))
|
|
|
|
continue;
|
|
|
|
return false;
|
|
|
|
}
|
2012-10-20 08:26:33 +00:00
|
|
|
|
2013-05-03 17:42:55 +00:00
|
|
|
// We only allow if-converted PHIs with more than two incoming values.
|
|
|
|
if (Phi->getNumIncomingValues() != 2) {
|
|
|
|
DEBUG(dbgs() << "LV: Found an invalid PHI.\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// This is the value coming from the preheader.
|
|
|
|
Value *StartValue = Phi->getIncomingValueForBlock(PreHeader);
|
2012-12-10 19:25:06 +00:00
|
|
|
// Check if this is an induction variable.
|
|
|
|
InductionKind IK = isInductionVariable(Phi);
|
|
|
|
|
2013-01-08 17:23:17 +00:00
|
|
|
if (IK_NoInduction != IK) {
|
2013-05-11 23:04:28 +00:00
|
|
|
// Get the widest type.
|
|
|
|
if (!WidestIndTy)
|
|
|
|
WidestIndTy = convertPointerToIntegerType(*DL, PhiTy);
|
|
|
|
else
|
|
|
|
WidestIndTy = getWiderType(*DL, PhiTy, WidestIndTy);
|
|
|
|
|
2012-12-10 19:25:06 +00:00
|
|
|
// Int inductions are special because we only allow one IV.
|
2013-01-08 17:23:17 +00:00
|
|
|
if (IK == IK_IntInduction) {
|
2013-05-14 00:21:18 +00:00
|
|
|
// Use the phi node with the widest type as induction. Use the last
|
|
|
|
// one if there are multiple (no good reason for doing this other
|
|
|
|
// than it is expedient).
|
|
|
|
if (!Induction || PhiTy == WidestIndTy)
|
|
|
|
Induction = Phi;
|
2012-12-10 19:25:06 +00:00
|
|
|
}
|
2012-12-03 21:06:35 +00:00
|
|
|
|
2012-12-10 19:25:06 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found an induction variable.\n");
|
|
|
|
Inductions[Phi] = InductionInfo(StartValue, IK);
|
2013-09-16 16:17:24 +00:00
|
|
|
|
|
|
|
// Until we explicitly handle the case of an induction variable with
|
|
|
|
// an outside loop user we have to give up vectorizing this loop.
|
|
|
|
if (hasOutsideLoopUser(TheLoop, it, AllowedExit))
|
|
|
|
return false;
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
continue;
|
2012-10-20 08:26:33 +00:00
|
|
|
}
|
2012-10-21 02:38:01 +00:00
|
|
|
|
2013-01-07 23:13:00 +00:00
|
|
|
if (AddReductionVar(Phi, RK_IntegerAdd)) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found an ADD reduction PHI."<< *Phi <<"\n");
|
|
|
|
continue;
|
|
|
|
}
|
2013-01-07 23:13:00 +00:00
|
|
|
if (AddReductionVar(Phi, RK_IntegerMult)) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a MUL reduction PHI."<< *Phi <<"\n");
|
|
|
|
continue;
|
|
|
|
}
|
2013-01-07 23:13:00 +00:00
|
|
|
if (AddReductionVar(Phi, RK_IntegerOr)) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found an OR reduction PHI."<< *Phi <<"\n");
|
|
|
|
continue;
|
|
|
|
}
|
2013-01-07 23:13:00 +00:00
|
|
|
if (AddReductionVar(Phi, RK_IntegerAnd)) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found an AND reduction PHI."<< *Phi <<"\n");
|
|
|
|
continue;
|
|
|
|
}
|
2013-01-07 23:13:00 +00:00
|
|
|
if (AddReductionVar(Phi, RK_IntegerXor)) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a XOR reduction PHI."<< *Phi <<"\n");
|
|
|
|
continue;
|
|
|
|
}
|
2013-04-18 17:22:34 +00:00
|
|
|
if (AddReductionVar(Phi, RK_IntegerMinMax)) {
|
|
|
|
DEBUG(dbgs() << "LV: Found a MINMAX reduction PHI."<< *Phi <<"\n");
|
|
|
|
continue;
|
|
|
|
}
|
2013-01-07 23:13:00 +00:00
|
|
|
if (AddReductionVar(Phi, RK_FloatMult)) {
|
|
|
|
DEBUG(dbgs() << "LV: Found an FMult reduction PHI."<< *Phi <<"\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (AddReductionVar(Phi, RK_FloatAdd)) {
|
|
|
|
DEBUG(dbgs() << "LV: Found an FAdd reduction PHI."<< *Phi <<"\n");
|
|
|
|
continue;
|
|
|
|
}
|
2013-05-05 01:54:48 +00:00
|
|
|
if (AddReductionVar(Phi, RK_FloatMinMax)) {
|
2013-07-31 22:17:45 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found an float MINMAX reduction PHI."<< *Phi <<
|
|
|
|
"\n");
|
2013-05-05 01:54:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found an unidentified PHI."<< *Phi <<"\n");
|
|
|
|
return false;
|
|
|
|
}// end of PHI handling
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2013-03-09 16:27:27 +00:00
|
|
|
// We still don't handle functions. However, we can ignore dbg intrinsic
|
2013-03-09 15:56:34 +00:00
|
|
|
// calls and we do handle certain intrinsic and libm functions.
|
2012-12-03 22:57:09 +00:00
|
|
|
CallInst *CI = dyn_cast<CallInst>(it);
|
2013-09-23 14:54:39 +00:00
|
|
|
if (CI && !getIntrinsicIDForCall(CI, TLI) && !isa<DbgInfoIntrinsic>(CI)) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a call site.\n");
|
2013-09-23 14:54:39 +00:00
|
|
|
return false;
|
2012-12-03 21:06:35 +00:00
|
|
|
}
|
2012-10-19 23:05:40 +00:00
|
|
|
|
2012-12-24 09:14:18 +00:00
|
|
|
// Check that the instruction return type is vectorizable.
|
2012-12-03 22:57:09 +00:00
|
|
|
if (!VectorType::isValidElementType(it->getType()) &&
|
|
|
|
!it->getType()->isVoidTy()) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found unvectorizable type." << "\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-12-24 09:14:18 +00:00
|
|
|
// Check that the stored type is vectorizable.
|
|
|
|
if (StoreInst *ST = dyn_cast<StoreInst>(it)) {
|
|
|
|
Type *T = ST->getValueOperand()->getType();
|
|
|
|
if (!VectorType::isValidElementType(T))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// Reduction instructions are allowed to have exit users.
|
|
|
|
// All other instructions must not have external users.
|
2013-05-31 19:53:50 +00:00
|
|
|
if (hasOutsideLoopUser(TheLoop, it, AllowedExit))
|
|
|
|
return false;
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
} // next instr.
|
|
|
|
|
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
|
2012-10-19 23:05:40 +00:00
|
|
|
if (!Induction) {
|
2012-11-25 08:41:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Did not find one integer induction var.\n");
|
2013-05-09 00:32:18 +00:00
|
|
|
if (Inductions.empty())
|
|
|
|
return false;
|
2012-10-18 05:29:12 +00:00
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
2012-10-26 23:49:28 +00:00
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
void LoopVectorizationLegality::collectLoopUniforms() {
|
2012-10-26 23:49:28 +00:00
|
|
|
// We now know that the loop is vectorizable!
|
|
|
|
// Collect variables that will remain uniform after vectorization.
|
|
|
|
std::vector<Value*> Worklist;
|
2012-12-03 21:06:35 +00:00
|
|
|
BasicBlock *Latch = TheLoop->getLoopLatch();
|
|
|
|
|
2012-10-26 23:49:28 +00:00
|
|
|
// Start with the conditional branch and walk up the block.
|
2012-12-03 21:06:35 +00:00
|
|
|
Worklist.push_back(Latch->getTerminator()->getOperand(0));
|
2012-10-26 23:49:28 +00:00
|
|
|
|
|
|
|
while (Worklist.size()) {
|
|
|
|
Instruction *I = dyn_cast<Instruction>(Worklist.back());
|
|
|
|
Worklist.pop_back();
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// Look at instructions inside this loop.
|
|
|
|
// Stop when reaching PHI nodes.
|
2012-12-03 22:57:09 +00:00
|
|
|
// TODO: we need to follow values all over the loop, not only in this block.
|
2012-12-03 21:06:35 +00:00
|
|
|
if (!I || !TheLoop->contains(I) || isa<PHINode>(I))
|
2012-11-25 08:41:35 +00:00
|
|
|
continue;
|
2012-10-26 23:49:28 +00:00
|
|
|
|
|
|
|
// This is a known uniform.
|
|
|
|
Uniforms.insert(I);
|
|
|
|
|
|
|
|
// Insert all operands.
|
2013-05-17 14:48:17 +00:00
|
|
|
Worklist.insert(Worklist.end(), I->op_begin(), I->op_end());
|
2012-10-26 23:49:28 +00:00
|
|
|
}
|
2012-10-20 04:59:06 +00:00
|
|
|
}
|
|
|
|
|
2013-06-29 17:02:06 +00:00
|
|
|
namespace {
|
2013-06-24 03:55:44 +00:00
|
|
|
/// \brief Analyses memory accesses in a loop.
|
|
|
|
///
|
|
|
|
/// Checks whether run time pointer checks are needed and builds sets for data
|
|
|
|
/// dependence checking.
|
|
|
|
class AccessAnalysis {
|
|
|
|
public:
|
|
|
|
/// \brief Read or write access location.
|
2013-06-29 17:52:08 +00:00
|
|
|
typedef PointerIntPair<Value *, 1, bool> MemAccessInfo;
|
|
|
|
typedef SmallPtrSet<MemAccessInfo, 8> MemAccessInfoSet;
|
2013-06-24 03:55:44 +00:00
|
|
|
|
|
|
|
/// \brief Set of potential dependent memory accesses.
|
|
|
|
typedef EquivalenceClasses<MemAccessInfo> DepCandidates;
|
|
|
|
|
|
|
|
AccessAnalysis(DataLayout *Dl, DepCandidates &DA) :
|
|
|
|
DL(Dl), DepCands(DA), AreAllWritesIdentified(true),
|
|
|
|
AreAllReadsIdentified(true), IsRTCheckNeeded(false) {}
|
|
|
|
|
|
|
|
/// \brief Register a load and whether it is only read from.
|
|
|
|
void addLoad(Value *Ptr, bool IsReadOnly) {
|
2013-06-29 17:52:08 +00:00
|
|
|
Accesses.insert(MemAccessInfo(Ptr, false));
|
2013-06-24 03:55:44 +00:00
|
|
|
if (IsReadOnly)
|
|
|
|
ReadOnlyPtr.insert(Ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Register a store.
|
|
|
|
void addStore(Value *Ptr) {
|
2013-06-29 17:52:08 +00:00
|
|
|
Accesses.insert(MemAccessInfo(Ptr, true));
|
2013-06-24 03:55:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Check whether we can check the pointers at runtime for
|
|
|
|
/// non-intersection.
|
|
|
|
bool canCheckPtrAtRT(LoopVectorizationLegality::RuntimePointerCheck &RtCheck,
|
|
|
|
unsigned &NumComparisons, ScalarEvolution *SE,
|
|
|
|
Loop *TheLoop);
|
|
|
|
|
|
|
|
/// \brief Goes over all memory accesses, checks whether a RT check is needed
|
|
|
|
/// and builds sets of dependent accesses.
|
|
|
|
void buildDependenceSets() {
|
|
|
|
// Process read-write pointers first.
|
|
|
|
processMemAccesses(false);
|
|
|
|
// Next, process read pointers.
|
|
|
|
processMemAccesses(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isRTCheckNeeded() { return IsRTCheckNeeded; }
|
|
|
|
|
|
|
|
bool isDependencyCheckNeeded() { return !CheckDeps.empty(); }
|
|
|
|
|
2013-06-29 17:52:08 +00:00
|
|
|
MemAccessInfoSet &getDependenciesToCheck() { return CheckDeps; }
|
2013-06-24 03:55:44 +00:00
|
|
|
|
|
|
|
private:
|
2013-06-24 12:09:12 +00:00
|
|
|
typedef SetVector<MemAccessInfo> PtrAccessSet;
|
2013-06-24 03:55:44 +00:00
|
|
|
typedef DenseMap<Value*, MemAccessInfo> UnderlyingObjToAccessMap;
|
|
|
|
|
|
|
|
/// \brief Go over all memory access or only the deferred ones if
|
|
|
|
/// \p UseDeferred is true and check whether runtime pointer checks are needed
|
|
|
|
/// and build sets of dependency check candidates.
|
|
|
|
void processMemAccesses(bool UseDeferred);
|
|
|
|
|
|
|
|
/// Set of all accesses.
|
|
|
|
PtrAccessSet Accesses;
|
|
|
|
|
|
|
|
/// Set of access to check after all writes have been processed.
|
|
|
|
PtrAccessSet DeferredAccesses;
|
|
|
|
|
|
|
|
/// Map of pointers to last access encountered.
|
|
|
|
UnderlyingObjToAccessMap ObjToLastAccess;
|
|
|
|
|
|
|
|
/// Set of accesses that need a further dependence check.
|
2013-06-29 17:52:08 +00:00
|
|
|
MemAccessInfoSet CheckDeps;
|
2013-06-24 03:55:44 +00:00
|
|
|
|
|
|
|
/// Set of pointers that are read only.
|
|
|
|
SmallPtrSet<Value*, 16> ReadOnlyPtr;
|
|
|
|
|
|
|
|
/// Set of underlying objects already written to.
|
|
|
|
SmallPtrSet<Value*, 16> WriteObjects;
|
|
|
|
|
|
|
|
DataLayout *DL;
|
|
|
|
|
|
|
|
/// Sets of potentially dependent accesses - members of one set share an
|
|
|
|
/// underlying pointer. The set "CheckDeps" identfies which sets really need a
|
|
|
|
/// dependence check.
|
|
|
|
DepCandidates &DepCands;
|
|
|
|
|
|
|
|
bool AreAllWritesIdentified;
|
|
|
|
bool AreAllReadsIdentified;
|
|
|
|
bool IsRTCheckNeeded;
|
|
|
|
};
|
|
|
|
|
2013-06-29 17:02:06 +00:00
|
|
|
} // end anonymous namespace
|
|
|
|
|
2013-06-24 03:55:44 +00:00
|
|
|
/// \brief Check whether a pointer can participate in a runtime bounds check.
|
|
|
|
static bool hasComputableBounds(ScalarEvolution *SE, Value *Ptr) {
|
|
|
|
const SCEV *PtrScev = SE->getSCEV(Ptr);
|
|
|
|
const SCEVAddRecExpr *AR = dyn_cast<SCEVAddRecExpr>(PtrScev);
|
|
|
|
if (!AR)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return AR->isAffine();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AccessAnalysis::canCheckPtrAtRT(
|
|
|
|
LoopVectorizationLegality::RuntimePointerCheck &RtCheck,
|
|
|
|
unsigned &NumComparisons, ScalarEvolution *SE,
|
|
|
|
Loop *TheLoop) {
|
|
|
|
// Find pointers with computable bounds. We are going to use this information
|
|
|
|
// to place a runtime bound check.
|
|
|
|
unsigned NumReadPtrChecks = 0;
|
|
|
|
unsigned NumWritePtrChecks = 0;
|
|
|
|
bool CanDoRT = true;
|
|
|
|
|
|
|
|
bool IsDepCheckNeeded = isDependencyCheckNeeded();
|
|
|
|
// We assign consecutive id to access from different dependence sets.
|
|
|
|
// Accesses within the same set don't need a runtime check.
|
|
|
|
unsigned RunningDepId = 1;
|
|
|
|
DenseMap<Value *, unsigned> DepSetId;
|
|
|
|
|
|
|
|
for (PtrAccessSet::iterator AI = Accesses.begin(), AE = Accesses.end();
|
|
|
|
AI != AE; ++AI) {
|
|
|
|
const MemAccessInfo &Access = *AI;
|
2013-06-29 17:52:08 +00:00
|
|
|
Value *Ptr = Access.getPointer();
|
|
|
|
bool IsWrite = Access.getInt();
|
2013-06-24 03:55:44 +00:00
|
|
|
|
|
|
|
// Just add write checks if we have both.
|
2013-06-29 17:52:08 +00:00
|
|
|
if (!IsWrite && Accesses.count(MemAccessInfo(Ptr, true)))
|
2013-06-24 03:55:44 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (IsWrite)
|
|
|
|
++NumWritePtrChecks;
|
|
|
|
else
|
|
|
|
++NumReadPtrChecks;
|
|
|
|
|
|
|
|
if (hasComputableBounds(SE, Ptr)) {
|
|
|
|
// The id of the dependence set.
|
|
|
|
unsigned DepId;
|
|
|
|
|
|
|
|
if (IsDepCheckNeeded) {
|
2013-06-29 17:52:08 +00:00
|
|
|
Value *Leader = DepCands.getLeaderValue(Access).getPointer();
|
2013-06-24 03:55:44 +00:00
|
|
|
unsigned &LeaderId = DepSetId[Leader];
|
|
|
|
if (!LeaderId)
|
|
|
|
LeaderId = RunningDepId++;
|
|
|
|
DepId = LeaderId;
|
|
|
|
} else
|
|
|
|
// Each access has its own dependence set.
|
|
|
|
DepId = RunningDepId++;
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
RtCheck.insert(SE, TheLoop, Ptr, IsWrite, DepId);
|
2013-06-24 03:55:44 +00:00
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV: Found a runtime check ptr:" << *Ptr <<"\n");
|
|
|
|
} else {
|
|
|
|
CanDoRT = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IsDepCheckNeeded && CanDoRT && RunningDepId == 2)
|
|
|
|
NumComparisons = 0; // Only one dependence set.
|
|
|
|
else
|
|
|
|
NumComparisons = (NumWritePtrChecks * (NumReadPtrChecks +
|
|
|
|
NumWritePtrChecks - 1));
|
|
|
|
return CanDoRT;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool isFunctionScopeIdentifiedObject(Value *Ptr) {
|
|
|
|
return isNoAliasArgument(Ptr) || isNoAliasCall(Ptr) || isa<AllocaInst>(Ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AccessAnalysis::processMemAccesses(bool UseDeferred) {
|
|
|
|
// We process the set twice: first we process read-write pointers, last we
|
|
|
|
// process read-only pointers. This allows us to skip dependence tests for
|
|
|
|
// read-only pointers.
|
|
|
|
|
|
|
|
PtrAccessSet &S = UseDeferred ? DeferredAccesses : Accesses;
|
|
|
|
for (PtrAccessSet::iterator AI = S.begin(), AE = S.end(); AI != AE; ++AI) {
|
|
|
|
const MemAccessInfo &Access = *AI;
|
2013-06-29 17:52:08 +00:00
|
|
|
Value *Ptr = Access.getPointer();
|
|
|
|
bool IsWrite = Access.getInt();
|
2013-06-24 03:55:44 +00:00
|
|
|
|
|
|
|
DepCands.insert(Access);
|
|
|
|
|
|
|
|
// Memorize read-only pointers for later processing and skip them in the
|
|
|
|
// first round (they need to be checked after we have seen all write
|
|
|
|
// pointers). Note: we also mark pointer that are not consecutive as
|
|
|
|
// "read-only" pointers (so that we check "a[b[i]] +="). Hence, we need the
|
|
|
|
// second check for "!IsWrite".
|
|
|
|
bool IsReadOnlyPtr = ReadOnlyPtr.count(Ptr) && !IsWrite;
|
|
|
|
if (!UseDeferred && IsReadOnlyPtr) {
|
|
|
|
DeferredAccesses.insert(Access);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool NeedDepCheck = false;
|
|
|
|
// Check whether there is the possiblity of dependency because of underlying
|
|
|
|
// objects being the same.
|
|
|
|
typedef SmallVector<Value*, 16> ValueVector;
|
|
|
|
ValueVector TempObjects;
|
|
|
|
GetUnderlyingObjects(Ptr, TempObjects, DL);
|
|
|
|
for (ValueVector::iterator UI = TempObjects.begin(), UE = TempObjects.end();
|
|
|
|
UI != UE; ++UI) {
|
|
|
|
Value *UnderlyingObj = *UI;
|
|
|
|
|
|
|
|
// If this is a write then it needs to be an identified object. If this a
|
|
|
|
// read and all writes (so far) are identified function scope objects we
|
|
|
|
// don't need an identified underlying object but only an Argument (the
|
|
|
|
// next write is going to invalidate this assumption if it is
|
|
|
|
// unidentified).
|
|
|
|
// This is a micro-optimization for the case where all writes are
|
|
|
|
// identified and we have one argument pointer.
|
|
|
|
// Otherwise, we do need a runtime check.
|
|
|
|
if ((IsWrite && !isFunctionScopeIdentifiedObject(UnderlyingObj)) ||
|
|
|
|
(!IsWrite && (!AreAllWritesIdentified ||
|
|
|
|
!isa<Argument>(UnderlyingObj)) &&
|
|
|
|
!isIdentifiedObject(UnderlyingObj))) {
|
|
|
|
DEBUG(dbgs() << "LV: Found an unidentified " <<
|
|
|
|
(IsWrite ? "write" : "read" ) << " ptr:" << *UnderlyingObj <<
|
|
|
|
"\n");
|
|
|
|
IsRTCheckNeeded = (IsRTCheckNeeded ||
|
|
|
|
!isIdentifiedObject(UnderlyingObj) ||
|
|
|
|
!AreAllReadsIdentified);
|
|
|
|
|
|
|
|
if (IsWrite)
|
|
|
|
AreAllWritesIdentified = false;
|
|
|
|
if (!IsWrite)
|
|
|
|
AreAllReadsIdentified = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this is a write - check other reads and writes for conflicts. If
|
|
|
|
// this is a read only check other writes for conflicts (but only if there
|
|
|
|
// is no other write to the ptr - this is an optimization to catch "a[i] =
|
|
|
|
// a[i] + " without having to do a dependence check).
|
|
|
|
if ((IsWrite || IsReadOnlyPtr) && WriteObjects.count(UnderlyingObj))
|
|
|
|
NeedDepCheck = true;
|
|
|
|
|
|
|
|
if (IsWrite)
|
|
|
|
WriteObjects.insert(UnderlyingObj);
|
|
|
|
|
|
|
|
// Create sets of pointers connected by shared underlying objects.
|
|
|
|
UnderlyingObjToAccessMap::iterator Prev =
|
|
|
|
ObjToLastAccess.find(UnderlyingObj);
|
|
|
|
if (Prev != ObjToLastAccess.end())
|
|
|
|
DepCands.unionSets(Access, Prev->second);
|
|
|
|
|
|
|
|
ObjToLastAccess[UnderlyingObj] = Access;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (NeedDepCheck)
|
|
|
|
CheckDeps.insert(Access);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-29 17:02:06 +00:00
|
|
|
namespace {
|
2013-06-24 03:55:45 +00:00
|
|
|
/// \brief Checks memory dependences among accesses to the same underlying
|
|
|
|
/// object to determine whether there vectorization is legal or not (and at
|
|
|
|
/// which vectorization factor).
|
|
|
|
///
|
|
|
|
/// This class works under the assumption that we already checked that memory
|
|
|
|
/// locations with different underlying pointers are "must-not alias".
|
|
|
|
/// We use the ScalarEvolution framework to symbolically evalutate access
|
|
|
|
/// functions pairs. Since we currently don't restructure the loop we can rely
|
|
|
|
/// on the program order of memory accesses to determine their safety.
|
|
|
|
/// At the moment we will only deem accesses as safe for:
|
|
|
|
/// * A negative constant distance assuming program order.
|
|
|
|
///
|
|
|
|
/// Safe: tmp = a[i + 1]; OR a[i + 1] = x;
|
|
|
|
/// a[i] = tmp; y = a[i];
|
|
|
|
///
|
|
|
|
/// The latter case is safe because later checks guarantuee that there can't
|
|
|
|
/// be a cycle through a phi node (that is, we check that "x" and "y" is not
|
|
|
|
/// the same variable: a header phi can only be an induction or a reduction, a
|
|
|
|
/// reduction can't have a memory sink, an induction can't have a memory
|
|
|
|
/// source). This is important and must not be violated (or we have to
|
|
|
|
/// resort to checking for cycles through memory).
|
|
|
|
///
|
|
|
|
/// * A positive constant distance assuming program order that is bigger
|
|
|
|
/// than the biggest memory access.
|
|
|
|
///
|
|
|
|
/// tmp = a[i] OR b[i] = x
|
|
|
|
/// a[i+2] = tmp y = b[i+2];
|
|
|
|
///
|
|
|
|
/// Safe distance: 2 x sizeof(a[0]), and 2 x sizeof(b[0]), respectively.
|
|
|
|
///
|
|
|
|
/// * Zero distances and all accesses have the same size.
|
|
|
|
///
|
|
|
|
class MemoryDepChecker {
|
|
|
|
public:
|
2013-06-29 17:52:08 +00:00
|
|
|
typedef PointerIntPair<Value *, 1, bool> MemAccessInfo;
|
|
|
|
typedef SmallPtrSet<MemAccessInfo, 8> MemAccessInfoSet;
|
2013-06-24 03:55:45 +00:00
|
|
|
|
|
|
|
MemoryDepChecker(ScalarEvolution *Se, DataLayout *Dl, const Loop *L) :
|
|
|
|
SE(Se), DL(Dl), InnermostLoop(L), AccessIdx(0) {}
|
|
|
|
|
|
|
|
/// \brief Register the location (instructions are given increasing numbers)
|
|
|
|
/// of a write access.
|
|
|
|
void addAccess(StoreInst *SI) {
|
|
|
|
Value *Ptr = SI->getPointerOperand();
|
2013-06-29 17:52:08 +00:00
|
|
|
Accesses[MemAccessInfo(Ptr, true)].push_back(AccessIdx);
|
2013-06-24 03:55:45 +00:00
|
|
|
InstMap.push_back(SI);
|
|
|
|
++AccessIdx;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Register the location (instructions are given increasing numbers)
|
|
|
|
/// of a write access.
|
|
|
|
void addAccess(LoadInst *LI) {
|
|
|
|
Value *Ptr = LI->getPointerOperand();
|
2013-06-29 17:52:08 +00:00
|
|
|
Accesses[MemAccessInfo(Ptr, false)].push_back(AccessIdx);
|
2013-06-24 03:55:45 +00:00
|
|
|
InstMap.push_back(LI);
|
|
|
|
++AccessIdx;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Check whether the dependencies between the accesses are safe.
|
|
|
|
///
|
|
|
|
/// Only checks sets with elements in \p CheckDeps.
|
|
|
|
bool areDepsSafe(AccessAnalysis::DepCandidates &AccessSets,
|
2013-06-29 17:52:08 +00:00
|
|
|
MemAccessInfoSet &CheckDeps);
|
2013-06-24 03:55:45 +00:00
|
|
|
|
|
|
|
/// \brief The maximum number of bytes of a vector register we can vectorize
|
|
|
|
/// the accesses safely with.
|
|
|
|
unsigned getMaxSafeDepDistBytes() { return MaxSafeDepDistBytes; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
ScalarEvolution *SE;
|
|
|
|
DataLayout *DL;
|
|
|
|
const Loop *InnermostLoop;
|
|
|
|
|
|
|
|
/// \brief Maps access locations (ptr, read/write) to program order.
|
|
|
|
DenseMap<MemAccessInfo, std::vector<unsigned> > Accesses;
|
|
|
|
|
|
|
|
/// \brief Memory access instructions in program order.
|
|
|
|
SmallVector<Instruction *, 16> InstMap;
|
|
|
|
|
|
|
|
/// \brief The program order index to be used for the next instruction.
|
|
|
|
unsigned AccessIdx;
|
|
|
|
|
|
|
|
// We can access this many bytes in parallel safely.
|
|
|
|
unsigned MaxSafeDepDistBytes;
|
|
|
|
|
|
|
|
/// \brief Check whether there is a plausible dependence between the two
|
|
|
|
/// accesses.
|
|
|
|
///
|
|
|
|
/// Access \p A must happen before \p B in program order. The two indices
|
|
|
|
/// identify the index into the program order map.
|
|
|
|
///
|
|
|
|
/// This function checks whether there is a plausible dependence (or the
|
|
|
|
/// absence of such can't be proved) between the two accesses. If there is a
|
|
|
|
/// plausible dependence but the dependence distance is bigger than one
|
|
|
|
/// element access it records this distance in \p MaxSafeDepDistBytes (if this
|
|
|
|
/// distance is smaller than any other distance encountered so far).
|
|
|
|
/// Otherwise, this function returns true signaling a possible dependence.
|
|
|
|
bool isDependent(const MemAccessInfo &A, unsigned AIdx,
|
|
|
|
const MemAccessInfo &B, unsigned BIdx);
|
|
|
|
|
|
|
|
/// \brief Check whether the data dependence could prevent store-load
|
|
|
|
/// forwarding.
|
|
|
|
bool couldPreventStoreLoadForward(unsigned Distance, unsigned TypeByteSize);
|
|
|
|
};
|
|
|
|
|
2013-06-29 17:02:06 +00:00
|
|
|
} // end anonymous namespace
|
|
|
|
|
2013-06-24 03:55:45 +00:00
|
|
|
static bool isInBoundsGep(Value *Ptr) {
|
|
|
|
if (GetElementPtrInst *GEP = dyn_cast<GetElementPtrInst>(Ptr))
|
|
|
|
return GEP->isInBounds();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Check whether the access through \p Ptr has a constant stride.
|
|
|
|
static int isStridedPtr(ScalarEvolution *SE, DataLayout *DL, Value *Ptr,
|
|
|
|
const Loop *Lp) {
|
2013-07-11 15:21:55 +00:00
|
|
|
const Type *Ty = Ptr->getType();
|
|
|
|
assert(Ty->isPointerTy() && "Unexpected non ptr");
|
2013-06-24 03:55:45 +00:00
|
|
|
|
|
|
|
// Make sure that the pointer does not point to aggregate types.
|
2013-07-11 15:21:55 +00:00
|
|
|
const PointerType *PtrTy = cast<PointerType>(Ty);
|
|
|
|
if (PtrTy->getElementType()->isAggregateType()) {
|
2013-07-31 22:17:45 +00:00
|
|
|
DEBUG(dbgs() << "LV: Bad stride - Not a pointer to a scalar type" << *Ptr <<
|
|
|
|
"\n");
|
2013-06-24 03:55:45 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const SCEV *PtrScev = SE->getSCEV(Ptr);
|
|
|
|
const SCEVAddRecExpr *AR = dyn_cast<SCEVAddRecExpr>(PtrScev);
|
|
|
|
if (!AR) {
|
|
|
|
DEBUG(dbgs() << "LV: Bad stride - Not an AddRecExpr pointer "
|
|
|
|
<< *Ptr << " SCEV: " << *PtrScev << "\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The accesss function must stride over the innermost loop.
|
|
|
|
if (Lp != AR->getLoop()) {
|
2013-07-31 22:17:45 +00:00
|
|
|
DEBUG(dbgs() << "LV: Bad stride - Not striding over innermost loop " <<
|
|
|
|
*Ptr << " SCEV: " << *PtrScev << "\n");
|
2013-06-24 03:55:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// The address calculation must not wrap. Otherwise, a dependence could be
|
2013-07-11 15:21:55 +00:00
|
|
|
// inverted.
|
|
|
|
// An inbounds getelementptr that is a AddRec with a unit stride
|
2013-06-24 03:55:45 +00:00
|
|
|
// cannot wrap per definition. The unit stride requirement is checked later.
|
2013-07-11 15:21:55 +00:00
|
|
|
// An getelementptr without an inbounds attribute and unit stride would have
|
|
|
|
// to access the pointer value "0" which is undefined behavior in address
|
|
|
|
// space 0, therefore we can also vectorize this case.
|
2013-06-24 03:55:45 +00:00
|
|
|
bool IsInBoundsGEP = isInBoundsGep(Ptr);
|
|
|
|
bool IsNoWrapAddRec = AR->getNoWrapFlags(SCEV::NoWrapMask);
|
2013-07-11 15:21:55 +00:00
|
|
|
bool IsInAddressSpaceZero = PtrTy->getAddressSpace() == 0;
|
|
|
|
if (!IsNoWrapAddRec && !IsInBoundsGEP && !IsInAddressSpaceZero) {
|
2013-06-24 03:55:45 +00:00
|
|
|
DEBUG(dbgs() << "LV: Bad stride - Pointer may wrap in the address space "
|
|
|
|
<< *Ptr << " SCEV: " << *PtrScev << "\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the step is constant.
|
|
|
|
const SCEV *Step = AR->getStepRecurrence(*SE);
|
|
|
|
|
|
|
|
// Calculate the pointer stride and check if it is consecutive.
|
|
|
|
const SCEVConstant *C = dyn_cast<SCEVConstant>(Step);
|
|
|
|
if (!C) {
|
|
|
|
DEBUG(dbgs() << "LV: Bad stride - Not a constant strided " << *Ptr <<
|
|
|
|
" SCEV: " << *PtrScev << "\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-07-11 15:21:55 +00:00
|
|
|
int64_t Size = DL->getTypeAllocSize(PtrTy->getElementType());
|
2013-06-24 03:55:45 +00:00
|
|
|
const APInt &APStepVal = C->getValue()->getValue();
|
|
|
|
|
|
|
|
// Huge step value - give up.
|
|
|
|
if (APStepVal.getBitWidth() > 64)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
int64_t StepVal = APStepVal.getSExtValue();
|
|
|
|
|
|
|
|
// Strided access.
|
|
|
|
int64_t Stride = StepVal / Size;
|
|
|
|
int64_t Rem = StepVal % Size;
|
|
|
|
if (Rem)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// If the SCEV could wrap but we have an inbounds gep with a unit stride we
|
2013-07-11 15:21:55 +00:00
|
|
|
// know we can't "wrap around the address space". In case of address space
|
|
|
|
// zero we know that this won't happen without triggering undefined behavior.
|
|
|
|
if (!IsNoWrapAddRec && (IsInBoundsGEP || IsInAddressSpaceZero) &&
|
|
|
|
Stride != 1 && Stride != -1)
|
2013-06-24 03:55:45 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
return Stride;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MemoryDepChecker::couldPreventStoreLoadForward(unsigned Distance,
|
|
|
|
unsigned TypeByteSize) {
|
|
|
|
// If loads occur at a distance that is not a multiple of a feasible vector
|
|
|
|
// factor store-load forwarding does not take place.
|
|
|
|
// Positive dependences might cause troubles because vectorizing them might
|
|
|
|
// prevent store-load forwarding making vectorized code run a lot slower.
|
|
|
|
// a[i] = a[i-3] ^ a[i-8];
|
|
|
|
// The stores to a[i:i+1] don't align with the stores to a[i-3:i-2] and
|
|
|
|
// hence on your typical architecture store-load forwarding does not take
|
|
|
|
// place. Vectorizing in such cases does not make sense.
|
|
|
|
// Store-load forwarding distance.
|
|
|
|
const unsigned NumCyclesForStoreLoadThroughMemory = 8*TypeByteSize;
|
|
|
|
// Maximum vector factor.
|
|
|
|
unsigned MaxVFWithoutSLForwardIssues = MaxVectorWidth*TypeByteSize;
|
|
|
|
if(MaxSafeDepDistBytes < MaxVFWithoutSLForwardIssues)
|
|
|
|
MaxVFWithoutSLForwardIssues = MaxSafeDepDistBytes;
|
|
|
|
|
|
|
|
for (unsigned vf = 2*TypeByteSize; vf <= MaxVFWithoutSLForwardIssues;
|
|
|
|
vf *= 2) {
|
|
|
|
if (Distance % vf && Distance / vf < NumCyclesForStoreLoadThroughMemory) {
|
|
|
|
MaxVFWithoutSLForwardIssues = (vf >>=1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MaxVFWithoutSLForwardIssues< 2*TypeByteSize) {
|
|
|
|
DEBUG(dbgs() << "LV: Distance " << Distance <<
|
|
|
|
" that could cause a store-load forwarding conflict\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MaxVFWithoutSLForwardIssues < MaxSafeDepDistBytes &&
|
|
|
|
MaxVFWithoutSLForwardIssues != MaxVectorWidth*TypeByteSize)
|
|
|
|
MaxSafeDepDistBytes = MaxVFWithoutSLForwardIssues;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MemoryDepChecker::isDependent(const MemAccessInfo &A, unsigned AIdx,
|
|
|
|
const MemAccessInfo &B, unsigned BIdx) {
|
|
|
|
assert (AIdx < BIdx && "Must pass arguments in program order");
|
|
|
|
|
2013-06-29 17:52:08 +00:00
|
|
|
Value *APtr = A.getPointer();
|
|
|
|
Value *BPtr = B.getPointer();
|
|
|
|
bool AIsWrite = A.getInt();
|
|
|
|
bool BIsWrite = B.getInt();
|
2013-06-24 03:55:45 +00:00
|
|
|
|
|
|
|
// Two reads are independent.
|
|
|
|
if (!AIsWrite && !BIsWrite)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const SCEV *AScev = SE->getSCEV(APtr);
|
|
|
|
const SCEV *BScev = SE->getSCEV(BPtr);
|
|
|
|
|
|
|
|
int StrideAPtr = isStridedPtr(SE, DL, APtr, InnermostLoop);
|
|
|
|
int StrideBPtr = isStridedPtr(SE, DL, BPtr, InnermostLoop);
|
|
|
|
|
|
|
|
const SCEV *Src = AScev;
|
|
|
|
const SCEV *Sink = BScev;
|
|
|
|
|
|
|
|
// If the induction step is negative we have to invert source and sink of the
|
|
|
|
// dependence.
|
|
|
|
if (StrideAPtr < 0) {
|
|
|
|
//Src = BScev;
|
|
|
|
//Sink = AScev;
|
|
|
|
std::swap(APtr, BPtr);
|
|
|
|
std::swap(Src, Sink);
|
|
|
|
std::swap(AIsWrite, BIsWrite);
|
|
|
|
std::swap(AIdx, BIdx);
|
|
|
|
std::swap(StrideAPtr, StrideBPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
const SCEV *Dist = SE->getMinusSCEV(Sink, Src);
|
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV: Src Scev: " << *Src << "Sink Scev: " << *Sink
|
|
|
|
<< "(Induction step: " << StrideAPtr << ")\n");
|
|
|
|
DEBUG(dbgs() << "LV: Distance for " << *InstMap[AIdx] << " to "
|
|
|
|
<< *InstMap[BIdx] << ": " << *Dist << "\n");
|
|
|
|
|
|
|
|
// Need consecutive accesses. We don't want to vectorize
|
|
|
|
// "A[B[i]] += ..." and similar code or pointer arithmetic that could wrap in
|
|
|
|
// the address space.
|
|
|
|
if (!StrideAPtr || !StrideBPtr || StrideAPtr != StrideBPtr){
|
|
|
|
DEBUG(dbgs() << "Non-consecutive pointer access\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const SCEVConstant *C = dyn_cast<SCEVConstant>(Dist);
|
|
|
|
if (!C) {
|
|
|
|
DEBUG(dbgs() << "LV: Dependence because of non constant distance\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Type *ATy = APtr->getType()->getPointerElementType();
|
|
|
|
Type *BTy = BPtr->getType()->getPointerElementType();
|
|
|
|
unsigned TypeByteSize = DL->getTypeAllocSize(ATy);
|
|
|
|
|
|
|
|
// Negative distances are not plausible dependencies.
|
|
|
|
const APInt &Val = C->getValue()->getValue();
|
|
|
|
if (Val.isNegative()) {
|
|
|
|
bool IsTrueDataDependence = (AIsWrite && !BIsWrite);
|
|
|
|
if (IsTrueDataDependence &&
|
|
|
|
(couldPreventStoreLoadForward(Val.abs().getZExtValue(), TypeByteSize) ||
|
|
|
|
ATy != BTy))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV: Dependence is negative: NoDep\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write to the same location with the same size.
|
|
|
|
// Could be improved to assert type sizes are the same (i32 == float, etc).
|
|
|
|
if (Val == 0) {
|
|
|
|
if (ATy == BTy)
|
|
|
|
return false;
|
|
|
|
DEBUG(dbgs() << "LV: Zero dependence difference but different types");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(Val.isStrictlyPositive() && "Expect a positive value");
|
|
|
|
|
|
|
|
// Positive distance bigger than max vectorization factor.
|
|
|
|
if (ATy != BTy) {
|
|
|
|
DEBUG(dbgs() <<
|
|
|
|
"LV: ReadWrite-Write positive dependency with different types");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned Distance = (unsigned) Val.getZExtValue();
|
|
|
|
|
|
|
|
// Bail out early if passed-in parameters make vectorization not feasible.
|
|
|
|
unsigned ForcedFactor = VectorizationFactor ? VectorizationFactor : 1;
|
|
|
|
unsigned ForcedUnroll = VectorizationUnroll ? VectorizationUnroll : 1;
|
|
|
|
|
|
|
|
// The distance must be bigger than the size needed for a vectorized version
|
|
|
|
// of the operation and the size of the vectorized operation must not be
|
|
|
|
// bigger than the currrent maximum size.
|
|
|
|
if (Distance < 2*TypeByteSize ||
|
|
|
|
2*TypeByteSize > MaxSafeDepDistBytes ||
|
|
|
|
Distance < TypeByteSize * ForcedUnroll * ForcedFactor) {
|
|
|
|
DEBUG(dbgs() << "LV: Failure because of Positive distance "
|
|
|
|
<< Val.getSExtValue() << "\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
MaxSafeDepDistBytes = Distance < MaxSafeDepDistBytes ?
|
|
|
|
Distance : MaxSafeDepDistBytes;
|
|
|
|
|
|
|
|
bool IsTrueDataDependence = (!AIsWrite && BIsWrite);
|
|
|
|
if (IsTrueDataDependence &&
|
|
|
|
couldPreventStoreLoadForward(Distance, TypeByteSize))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV: Positive distance " << Val.getSExtValue() <<
|
|
|
|
" with max VF=" << MaxSafeDepDistBytes/TypeByteSize << "\n");
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
MemoryDepChecker::areDepsSafe(AccessAnalysis::DepCandidates &AccessSets,
|
2013-06-29 17:52:08 +00:00
|
|
|
MemAccessInfoSet &CheckDeps) {
|
2013-06-24 03:55:45 +00:00
|
|
|
|
|
|
|
MaxSafeDepDistBytes = -1U;
|
|
|
|
while (!CheckDeps.empty()) {
|
|
|
|
MemAccessInfo CurAccess = *CheckDeps.begin();
|
|
|
|
|
|
|
|
// Get the relevant memory access set.
|
|
|
|
EquivalenceClasses<MemAccessInfo>::iterator I =
|
|
|
|
AccessSets.findValue(AccessSets.getLeaderValue(CurAccess));
|
|
|
|
|
|
|
|
// Check accesses within this set.
|
|
|
|
EquivalenceClasses<MemAccessInfo>::member_iterator AI, AE;
|
|
|
|
AI = AccessSets.member_begin(I), AE = AccessSets.member_end();
|
|
|
|
|
|
|
|
// Check every access pair.
|
|
|
|
while (AI != AE) {
|
|
|
|
CheckDeps.erase(*AI);
|
|
|
|
EquivalenceClasses<MemAccessInfo>::member_iterator OI = llvm::next(AI);
|
|
|
|
while (OI != AE) {
|
|
|
|
// Check every accessing instruction pair in program order.
|
|
|
|
for (std::vector<unsigned>::iterator I1 = Accesses[*AI].begin(),
|
|
|
|
I1E = Accesses[*AI].end(); I1 != I1E; ++I1)
|
|
|
|
for (std::vector<unsigned>::iterator I2 = Accesses[*OI].begin(),
|
|
|
|
I2E = Accesses[*OI].end(); I2 != I2E; ++I2) {
|
|
|
|
if (*I1 < *I2 && isDependent(*AI, *I1, *OI, *I2))
|
|
|
|
return false;
|
|
|
|
if (*I2 < *I1 && isDependent(*OI, *I2, *AI, *I1))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
++OI;
|
|
|
|
}
|
|
|
|
AI++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
bool LoopVectorizationLegality::canVectorizeMemory() {
|
2013-02-13 18:08:57 +00:00
|
|
|
|
2012-10-20 08:26:33 +00:00
|
|
|
typedef SmallVector<Value*, 16> ValueVector;
|
|
|
|
typedef SmallPtrSet<Value*, 16> ValueSet;
|
2013-06-24 12:09:15 +00:00
|
|
|
|
2012-10-20 08:26:33 +00:00
|
|
|
// Holds the Load and Store *instructions*.
|
|
|
|
ValueVector Loads;
|
|
|
|
ValueVector Stores;
|
2013-06-24 12:09:15 +00:00
|
|
|
|
|
|
|
// Holds all the different accesses in the loop.
|
|
|
|
unsigned NumReads = 0;
|
|
|
|
unsigned NumReadWrites = 0;
|
|
|
|
|
2012-11-09 07:09:44 +00:00
|
|
|
PtrRtCheck.Pointers.clear();
|
|
|
|
PtrRtCheck.Need = false;
|
2012-10-20 04:59:06 +00:00
|
|
|
|
2013-04-23 16:44:43 +00:00
|
|
|
const bool IsAnnotatedParallel = TheLoop->isAnnotatedParallel();
|
2013-06-24 12:09:15 +00:00
|
|
|
MemoryDepChecker DepChecker(SE, DL, TheLoop);
|
2013-04-23 16:44:43 +00:00
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// For each block.
|
|
|
|
for (Loop::block_iterator bb = TheLoop->block_begin(),
|
|
|
|
be = TheLoop->block_end(); bb != be; ++bb) {
|
|
|
|
|
|
|
|
// Scan the BB and collect legal loads and stores.
|
|
|
|
for (BasicBlock::iterator it = (*bb)->begin(), e = (*bb)->end(); it != e;
|
|
|
|
++it) {
|
|
|
|
|
|
|
|
// If this is a load, save it. If this instruction can read from memory
|
|
|
|
// but is not a load, then we quit. Notice that we don't handle function
|
|
|
|
// calls that read or write.
|
|
|
|
if (it->mayReadFromMemory()) {
|
2013-07-01 00:54:44 +00:00
|
|
|
// Many math library functions read the rounding mode. We will only
|
|
|
|
// vectorize a loop if it contains known function calls that don't set
|
|
|
|
// the flag. Therefore, it is safe to ignore this read from memory.
|
|
|
|
CallInst *Call = dyn_cast<CallInst>(it);
|
|
|
|
if (Call && getIntrinsicIDForCall(Call, TLI))
|
|
|
|
continue;
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
LoadInst *Ld = dyn_cast<LoadInst>(it);
|
|
|
|
if (!Ld) return false;
|
2013-04-23 16:44:43 +00:00
|
|
|
if (!Ld->isSimple() && !IsAnnotatedParallel) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a non-simple load.\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Loads.push_back(Ld);
|
2013-06-24 12:09:15 +00:00
|
|
|
DepChecker.addAccess(Ld);
|
2012-12-03 21:06:35 +00:00
|
|
|
continue;
|
2012-10-20 04:59:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
// Save 'store' instructions. Abort if other instructions write to memory.
|
|
|
|
if (it->mayWriteToMemory()) {
|
|
|
|
StoreInst *St = dyn_cast<StoreInst>(it);
|
|
|
|
if (!St) return false;
|
2013-04-23 16:44:43 +00:00
|
|
|
if (!St->isSimple() && !IsAnnotatedParallel) {
|
2012-12-03 21:06:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a non-simple store.\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Stores.push_back(St);
|
2013-06-24 12:09:15 +00:00
|
|
|
DepChecker.addAccess(St);
|
2012-10-20 04:59:06 +00:00
|
|
|
}
|
2012-12-03 21:06:35 +00:00
|
|
|
} // next instr.
|
|
|
|
} // next block.
|
2012-10-20 04:59:06 +00:00
|
|
|
|
2012-10-20 08:26:33 +00:00
|
|
|
// Now we have two lists that hold the loads and the stores.
|
|
|
|
// Next, we find the pointers that they use.
|
2012-10-20 04:59:06 +00:00
|
|
|
|
2012-10-20 08:26:33 +00:00
|
|
|
// Check if we see any stores. If there are no stores, then we don't
|
|
|
|
// care if the pointers are *restrict*.
|
|
|
|
if (!Stores.size()) {
|
2012-12-10 21:39:02 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a read-only loop!\n");
|
|
|
|
return true;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
AccessAnalysis::DepCandidates DependentAccesses;
|
|
|
|
AccessAnalysis Accesses(DL, DependentAccesses);
|
2012-10-20 08:26:33 +00:00
|
|
|
|
|
|
|
// Holds the analyzed pointers. We don't want to call GetUnderlyingObjects
|
|
|
|
// multiple times on the same object. If the ptr is accessed twice, once
|
|
|
|
// for read and once for write, it will only appear once (on the write
|
|
|
|
// list). This is okay, since we are going to check for conflicts between
|
|
|
|
// writes and between reads and writes, but not between reads and reads.
|
|
|
|
ValueSet Seen;
|
|
|
|
|
|
|
|
ValueVector::iterator I, IE;
|
|
|
|
for (I = Stores.begin(), IE = Stores.end(); I != IE; ++I) {
|
2012-12-21 00:07:35 +00:00
|
|
|
StoreInst *ST = cast<StoreInst>(*I);
|
2012-10-20 08:26:33 +00:00
|
|
|
Value* Ptr = ST->getPointerOperand();
|
2012-11-09 07:09:44 +00:00
|
|
|
|
|
|
|
if (isUniform(Ptr)) {
|
|
|
|
DEBUG(dbgs() << "LV: We don't allow storing to uniform addresses\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
// If we did *not* see this pointer before, insert it to the read-write
|
|
|
|
// list. At this phase it is only a 'write' list.
|
|
|
|
if (Seen.insert(Ptr)) {
|
|
|
|
++NumReadWrites;
|
|
|
|
Accesses.addStore(Ptr);
|
|
|
|
}
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2013-04-23 16:44:43 +00:00
|
|
|
if (IsAnnotatedParallel) {
|
2013-04-23 08:08:51 +00:00
|
|
|
DEBUG(dbgs()
|
|
|
|
<< "LV: A loop annotated parallel, ignore memory dependency "
|
|
|
|
<< "checks.\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
SmallPtrSet<Value *, 16> ReadOnlyPtr;
|
2012-10-20 08:26:33 +00:00
|
|
|
for (I = Loads.begin(), IE = Loads.end(); I != IE; ++I) {
|
2012-12-21 00:07:35 +00:00
|
|
|
LoadInst *LD = cast<LoadInst>(*I);
|
2012-10-20 08:26:33 +00:00
|
|
|
Value* Ptr = LD->getPointerOperand();
|
|
|
|
// If we did *not* see this pointer before, insert it to the
|
|
|
|
// read list. If we *did* see it before, then it is already in
|
|
|
|
// the read-write list. This allows us to vectorize expressions
|
|
|
|
// such as A[i] += x; Because the address of A[i] is a read-write
|
|
|
|
// pointer. This only works if the index of A[i] is consecutive.
|
|
|
|
// If the address of i is unknown (for example A[B[i]]) then we may
|
|
|
|
// read a few words, modify, and write a few words, and some of the
|
|
|
|
// words may be written to the same address.
|
2013-06-24 12:09:15 +00:00
|
|
|
bool IsReadOnlyPtr = false;
|
|
|
|
if (Seen.insert(Ptr) || !isStridedPtr(SE, DL, Ptr, TheLoop)) {
|
|
|
|
++NumReads;
|
|
|
|
IsReadOnlyPtr = true;
|
|
|
|
}
|
|
|
|
Accesses.addLoad(Ptr, IsReadOnlyPtr);
|
2012-10-20 08:26:33 +00:00
|
|
|
}
|
|
|
|
|
2012-10-30 18:36:45 +00:00
|
|
|
// If we write (or read-write) to a single destination and there are no
|
|
|
|
// other reads in this loop then is it safe to vectorize.
|
2013-06-24 12:09:15 +00:00
|
|
|
if (NumReadWrites == 1 && NumReads == 0) {
|
2012-10-30 18:36:45 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found a write-only loop!\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
// Build dependence sets and check whether we need a runtime pointer bounds
|
|
|
|
// check.
|
|
|
|
Accesses.buildDependenceSets();
|
|
|
|
bool NeedRTCheck = Accesses.isRTCheckNeeded();
|
2013-04-26 05:08:59 +00:00
|
|
|
|
2012-11-09 07:09:44 +00:00
|
|
|
// Find pointers with computable bounds. We are going to use this information
|
|
|
|
// to place a runtime bound check.
|
2013-06-24 12:09:15 +00:00
|
|
|
unsigned NumComparisons = 0;
|
|
|
|
bool CanDoRT = false;
|
|
|
|
if (NeedRTCheck)
|
|
|
|
CanDoRT = Accesses.canCheckPtrAtRT(PtrRtCheck, NumComparisons, SE, TheLoop);
|
2013-06-24 03:55:48 +00:00
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV: We need to do " << NumComparisons <<
|
|
|
|
" pointer comparisons.\n");
|
|
|
|
|
|
|
|
// If we only have one set of dependences to check pointers among we don't
|
|
|
|
// need a runtime check.
|
|
|
|
if (NumComparisons == 0 && NeedRTCheck)
|
|
|
|
NeedRTCheck = false;
|
|
|
|
|
|
|
|
// Check that we did not collect too many pointers or found a unsizeable
|
|
|
|
// pointer.
|
2013-04-26 05:08:59 +00:00
|
|
|
if (!CanDoRT || NumComparisons > RuntimeMemoryCheckThreshold) {
|
2012-11-25 08:41:35 +00:00
|
|
|
PtrRtCheck.reset();
|
2012-12-21 00:07:35 +00:00
|
|
|
CanDoRT = false;
|
2012-11-09 07:09:44 +00:00
|
|
|
}
|
|
|
|
|
2012-12-21 00:07:35 +00:00
|
|
|
if (CanDoRT) {
|
2012-11-09 07:09:44 +00:00
|
|
|
DEBUG(dbgs() << "LV: We can perform a memory runtime check if needed.\n");
|
|
|
|
}
|
|
|
|
|
2012-12-21 00:07:35 +00:00
|
|
|
if (NeedRTCheck && !CanDoRT) {
|
|
|
|
DEBUG(dbgs() << "LV: We can't vectorize because we can't find " <<
|
|
|
|
"the array bounds.\n");
|
|
|
|
PtrRtCheck.reset();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
PtrRtCheck.Need = NeedRTCheck;
|
|
|
|
|
|
|
|
bool CanVecMem = true;
|
|
|
|
if (Accesses.isDependencyCheckNeeded()) {
|
|
|
|
DEBUG(dbgs() << "LV: Checking memory dependencies\n");
|
|
|
|
CanVecMem = DepChecker.areDepsSafe(DependentAccesses,
|
|
|
|
Accesses.getDependenciesToCheck());
|
|
|
|
MaxSafeDepDistBytes = DepChecker.getMaxSafeDepDistBytes();
|
|
|
|
}
|
|
|
|
|
2012-12-21 00:07:35 +00:00
|
|
|
DEBUG(dbgs() << "LV: We "<< (NeedRTCheck ? "" : "don't") <<
|
|
|
|
" need a runtime memory check.\n");
|
2013-06-24 12:09:15 +00:00
|
|
|
|
|
|
|
return CanVecMem;
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
static bool hasMultipleUsesOf(Instruction *I,
|
|
|
|
SmallPtrSet<Instruction *, 8> &Insts) {
|
|
|
|
unsigned NumUses = 0;
|
|
|
|
for(User::op_iterator Use = I->op_begin(), E = I->op_end(); Use != E; ++Use) {
|
|
|
|
if (Insts.count(dyn_cast<Instruction>(*Use)))
|
|
|
|
++NumUses;
|
|
|
|
if (NumUses > 1)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool areAllUsesIn(Instruction *I, SmallPtrSet<Instruction *, 8> &Set) {
|
|
|
|
for(User::op_iterator Use = I->op_begin(), E = I->op_end(); Use != E; ++Use)
|
|
|
|
if (!Set.count(dyn_cast<Instruction>(*Use)))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-10-19 23:05:40 +00:00
|
|
|
bool LoopVectorizationLegality::AddReductionVar(PHINode *Phi,
|
2012-10-21 05:52:51 +00:00
|
|
|
ReductionKind Kind) {
|
2012-10-19 23:05:40 +00:00
|
|
|
if (Phi->getNumIncomingValues() != 2)
|
|
|
|
return false;
|
|
|
|
|
2012-12-04 18:17:33 +00:00
|
|
|
// Reduction variables are only found in the loop header block.
|
|
|
|
if (Phi->getParent() != TheLoop->getHeader())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Obtain the reduction start value from the value that comes from the loop
|
|
|
|
// preheader.
|
|
|
|
Value *RdxStart = Phi->getIncomingValueForBlock(TheLoop->getLoopPreheader());
|
2012-10-19 23:05:40 +00:00
|
|
|
|
|
|
|
// ExitInstruction is the single value which is used outside the loop.
|
|
|
|
// We only allow for a single reduction value to be used outside the loop.
|
|
|
|
// This includes users of the reduction, variables (which form a cycle
|
|
|
|
// which ends in the phi node).
|
|
|
|
Instruction *ExitInstruction = 0;
|
2013-05-07 21:55:37 +00:00
|
|
|
// Indicates that we found a reduction operation in our scan.
|
|
|
|
bool FoundReduxOp = false;
|
2012-10-19 23:05:40 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
// We start with the PHI node and scan for all of the users of this
|
|
|
|
// instruction. All users must be instructions that can be used as reduction
|
|
|
|
// variables (such as ADD). We must have a single out-of-block user. The cycle
|
|
|
|
// must include the original PHI.
|
|
|
|
bool FoundStartPHI = false;
|
2013-04-18 14:29:13 +00:00
|
|
|
|
2013-04-18 17:22:34 +00:00
|
|
|
// To recognize min/max patterns formed by a icmp select sequence, we store
|
|
|
|
// the number of instruction we saw from the recognized min/max pattern,
|
2013-05-07 21:55:37 +00:00
|
|
|
// to make sure we only see exactly the two instructions.
|
2013-05-05 01:54:48 +00:00
|
|
|
unsigned NumCmpSelectPatternInst = 0;
|
2013-04-18 17:22:34 +00:00
|
|
|
ReductionInstDesc ReduxDesc(false, 0);
|
|
|
|
|
2013-04-18 14:29:13 +00:00
|
|
|
SmallPtrSet<Instruction *, 8> VisitedInsts;
|
2013-05-07 21:55:37 +00:00
|
|
|
SmallVector<Instruction *, 8> Worklist;
|
|
|
|
Worklist.push_back(Phi);
|
|
|
|
VisitedInsts.insert(Phi);
|
|
|
|
|
|
|
|
// A value in the reduction can be used:
|
|
|
|
// - By the reduction:
|
|
|
|
// - Reduction operation:
|
|
|
|
// - One use of reduction value (safe).
|
|
|
|
// - Multiple use of reduction value (not safe).
|
|
|
|
// - PHI:
|
|
|
|
// - All uses of the PHI must be the reduction (safe).
|
|
|
|
// - Otherwise, not safe.
|
|
|
|
// - By one instruction outside of the loop (safe).
|
|
|
|
// - By further instructions outside of the loop (not safe).
|
|
|
|
// - By an instruction that is not part of the reduction (not safe).
|
|
|
|
// This is either:
|
|
|
|
// * An instruction type other than PHI or the reduction operation.
|
|
|
|
// * A PHI in the header other than the initial PHI.
|
|
|
|
while (!Worklist.empty()) {
|
|
|
|
Instruction *Cur = Worklist.back();
|
|
|
|
Worklist.pop_back();
|
|
|
|
|
|
|
|
// No Users.
|
|
|
|
// If the instruction has no users then this is a broken chain and can't be
|
|
|
|
// a reduction variable.
|
|
|
|
if (Cur->use_empty())
|
2012-12-04 22:40:22 +00:00
|
|
|
return false;
|
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
bool IsAPhi = isa<PHINode>(Cur);
|
2012-10-20 20:45:01 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
// A header PHI use other than the original PHI.
|
|
|
|
if (Cur != Phi && IsAPhi && Cur->getParent() == Phi->getParent())
|
|
|
|
return false;
|
2013-01-07 23:13:00 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
// Reductions of instructions such as Div, and Sub is only possible if the
|
|
|
|
// LHS is the reduction variable.
|
|
|
|
if (!Cur->isCommutative() && !IsAPhi && !isa<SelectInst>(Cur) &&
|
|
|
|
!isa<ICmpInst>(Cur) && !isa<FCmpInst>(Cur) &&
|
|
|
|
!VisitedInsts.count(dyn_cast<Instruction>(Cur->getOperand(0))))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Any reduction instruction must be of one of the allowed kinds.
|
|
|
|
ReduxDesc = isReductionInstr(Cur, Kind, ReduxDesc);
|
|
|
|
if (!ReduxDesc.IsReduction)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// A reduction operation must only have one use of the reduction value.
|
|
|
|
if (!IsAPhi && Kind != RK_IntegerMinMax && Kind != RK_FloatMinMax &&
|
|
|
|
hasMultipleUsesOf(Cur, VisitedInsts))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// All inputs to a PHI node must be a reduction value.
|
|
|
|
if(IsAPhi && Cur != Phi && !areAllUsesIn(Cur, VisitedInsts))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (Kind == RK_IntegerMinMax && (isa<ICmpInst>(Cur) ||
|
|
|
|
isa<SelectInst>(Cur)))
|
|
|
|
++NumCmpSelectPatternInst;
|
|
|
|
if (Kind == RK_FloatMinMax && (isa<FCmpInst>(Cur) ||
|
|
|
|
isa<SelectInst>(Cur)))
|
|
|
|
++NumCmpSelectPatternInst;
|
|
|
|
|
|
|
|
// Check whether we found a reduction operator.
|
|
|
|
FoundReduxOp |= !IsAPhi;
|
|
|
|
|
|
|
|
// Process users of current instruction. Push non PHI nodes after PHI nodes
|
|
|
|
// onto the stack. This way we are going to have seen all inputs to PHI
|
|
|
|
// nodes once we get to them.
|
|
|
|
SmallVector<Instruction *, 8> NonPHIs;
|
|
|
|
SmallVector<Instruction *, 8> PHIs;
|
|
|
|
for (Value::use_iterator UI = Cur->use_begin(), E = Cur->use_end(); UI != E;
|
|
|
|
++UI) {
|
|
|
|
Instruction *Usr = cast<Instruction>(*UI);
|
2012-12-04 18:17:33 +00:00
|
|
|
|
2012-10-19 23:05:40 +00:00
|
|
|
// Check if we found the exit user.
|
2013-05-07 21:55:37 +00:00
|
|
|
BasicBlock *Parent = Usr->getParent();
|
2012-12-04 18:17:33 +00:00
|
|
|
if (!TheLoop->contains(Parent)) {
|
LoopVectorizer: Disallow reductions whose header phi is used outside the loop
If an outside loop user of the reduction value uses the header phi node we
cannot just reduce the vectorized phi value in the vector code epilog because
we would loose VF-1 reductions.
lp:
p = phi (0, lv)
lv = lv + 1
...
brcond , lp, outside
outside:
usr = add 0, p
(Say the loop iterates two times, the value of p coming out of the loop is one).
We cannot just transform this to:
vlp:
p = phi (<0,0>, lv)
lv = lv + <1,1>
..
brcond , lp, outside
outside:
p_reduced = p[0] + [1];
usr = add 0, p_reduced
(Because the original loop iterated two times the vectorized loop would iterate
one time, but p_reduced ends up being zero instead of one).
We would have to execute VF-1 iterations in the scalar remainder loop in such
cases. For now, just disable vectorization.
PR16522
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@186256 91177308-0d34-0410-b5e6-96231b3b80d8
2013-07-13 19:09:29 +00:00
|
|
|
// Exit if you find multiple outside users or if the header phi node is
|
|
|
|
// being used. In this case the user uses the value of the previous
|
|
|
|
// iteration, in which case we would loose "VF-1" iterations of the
|
|
|
|
// reduction operation if we vectorize.
|
|
|
|
if (ExitInstruction != 0 || Cur == Phi)
|
2012-10-19 23:05:40 +00:00
|
|
|
return false;
|
LoopVectorizer: Disallow reductions whose header phi is used outside the loop
If an outside loop user of the reduction value uses the header phi node we
cannot just reduce the vectorized phi value in the vector code epilog because
we would loose VF-1 reductions.
lp:
p = phi (0, lv)
lv = lv + 1
...
brcond , lp, outside
outside:
usr = add 0, p
(Say the loop iterates two times, the value of p coming out of the loop is one).
We cannot just transform this to:
vlp:
p = phi (<0,0>, lv)
lv = lv + <1,1>
..
brcond , lp, outside
outside:
p_reduced = p[0] + [1];
usr = add 0, p_reduced
(Because the original loop iterated two times the vectorized loop would iterate
one time, but p_reduced ends up being zero instead of one).
We would have to execute VF-1 iterations in the scalar remainder loop in such
cases. For now, just disable vectorization.
PR16522
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@186256 91177308-0d34-0410-b5e6-96231b3b80d8
2013-07-13 19:09:29 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
ExitInstruction = Cur;
|
2012-12-04 22:40:22 +00:00
|
|
|
continue;
|
2013-05-07 21:55:37 +00:00
|
|
|
}
|
2012-12-04 22:40:22 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
// Process instructions only once (termination).
|
|
|
|
if (VisitedInsts.insert(Usr)) {
|
|
|
|
if (isa<PHINode>(Usr))
|
|
|
|
PHIs.push_back(Usr);
|
|
|
|
else
|
|
|
|
NonPHIs.push_back(Usr);
|
|
|
|
}
|
|
|
|
// Remember that we completed the cycle.
|
|
|
|
if (Usr == Phi)
|
|
|
|
FoundStartPHI = true;
|
|
|
|
}
|
|
|
|
Worklist.append(PHIs.begin(), PHIs.end());
|
|
|
|
Worklist.append(NonPHIs.begin(), NonPHIs.end());
|
|
|
|
}
|
2013-01-05 01:15:47 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
// This means we have seen one but not the other instruction of the
|
|
|
|
// pattern or more than just a select and cmp.
|
|
|
|
if ((Kind == RK_IntegerMinMax || Kind == RK_FloatMinMax) &&
|
|
|
|
NumCmpSelectPatternInst != 2)
|
|
|
|
return false;
|
2013-04-18 17:22:34 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
if (!FoundStartPHI || !FoundReduxOp || !ExitInstruction)
|
|
|
|
return false;
|
2013-01-05 01:15:47 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
// We found a reduction var if we have reached the original phi node and we
|
|
|
|
// only have a single instruction with out-of-loop users.
|
2012-10-19 23:05:40 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
// This instruction is allowed to have out-of-loop users.
|
|
|
|
AllowedExit.insert(ExitInstruction);
|
2013-04-18 17:22:34 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
// Save the description of this reduction variable.
|
|
|
|
ReductionDescriptor RD(RdxStart, ExitInstruction, Kind,
|
|
|
|
ReduxDesc.MinMaxKind);
|
|
|
|
Reductions[Phi] = RD;
|
|
|
|
// We've ended the cycle. This is a reduction variable if we have an
|
|
|
|
// outside user and it has a binary op.
|
2013-04-18 14:29:13 +00:00
|
|
|
|
2013-05-07 21:55:37 +00:00
|
|
|
return true;
|
2012-10-19 23:05:40 +00:00
|
|
|
}
|
|
|
|
|
2013-04-18 17:22:34 +00:00
|
|
|
/// Returns true if the instruction is a Select(ICmp(X, Y), X, Y) instruction
|
|
|
|
/// pattern corresponding to a min(X, Y) or max(X, Y).
|
2013-04-19 21:03:36 +00:00
|
|
|
LoopVectorizationLegality::ReductionInstDesc
|
2013-05-05 01:54:48 +00:00
|
|
|
LoopVectorizationLegality::isMinMaxSelectCmpPattern(Instruction *I,
|
|
|
|
ReductionInstDesc &Prev) {
|
2013-04-18 17:22:34 +00:00
|
|
|
|
2013-05-05 01:54:48 +00:00
|
|
|
assert((isa<ICmpInst>(I) || isa<FCmpInst>(I) || isa<SelectInst>(I)) &&
|
2013-04-18 17:22:34 +00:00
|
|
|
"Expect a select instruction");
|
2013-05-05 01:54:48 +00:00
|
|
|
Instruction *Cmp = 0;
|
2013-04-18 17:22:34 +00:00
|
|
|
SelectInst *Select = 0;
|
|
|
|
|
2013-04-19 21:03:36 +00:00
|
|
|
// We must handle the select(cmp()) as a single instruction. Advance to the
|
|
|
|
// select.
|
2013-05-05 01:54:48 +00:00
|
|
|
if ((Cmp = dyn_cast<ICmpInst>(I)) || (Cmp = dyn_cast<FCmpInst>(I))) {
|
2013-04-19 21:03:36 +00:00
|
|
|
if (!Cmp->hasOneUse() || !(Select = dyn_cast<SelectInst>(*I->use_begin())))
|
|
|
|
return ReductionInstDesc(false, I);
|
|
|
|
return ReductionInstDesc(Select, Prev.MinMaxKind);
|
2013-04-18 17:22:34 +00:00
|
|
|
}
|
|
|
|
|
2013-04-19 21:03:36 +00:00
|
|
|
// Only handle single use cases for now.
|
|
|
|
if (!(Select = dyn_cast<SelectInst>(I)))
|
|
|
|
return ReductionInstDesc(false, I);
|
2013-05-05 01:54:48 +00:00
|
|
|
if (!(Cmp = dyn_cast<ICmpInst>(I->getOperand(0))) &&
|
|
|
|
!(Cmp = dyn_cast<FCmpInst>(I->getOperand(0))))
|
2013-04-19 21:03:36 +00:00
|
|
|
return ReductionInstDesc(false, I);
|
|
|
|
if (!Cmp->hasOneUse())
|
|
|
|
return ReductionInstDesc(false, I);
|
2013-04-18 17:22:34 +00:00
|
|
|
|
2013-05-05 01:54:44 +00:00
|
|
|
Value *CmpLeft;
|
|
|
|
Value *CmpRight;
|
2013-04-18 17:22:34 +00:00
|
|
|
|
2013-04-19 21:03:36 +00:00
|
|
|
// Look for a min/max pattern.
|
|
|
|
if (m_UMin(m_Value(CmpLeft), m_Value(CmpRight)).match(Select))
|
|
|
|
return ReductionInstDesc(Select, MRK_UIntMin);
|
|
|
|
else if (m_UMax(m_Value(CmpLeft), m_Value(CmpRight)).match(Select))
|
|
|
|
return ReductionInstDesc(Select, MRK_UIntMax);
|
|
|
|
else if (m_SMax(m_Value(CmpLeft), m_Value(CmpRight)).match(Select))
|
|
|
|
return ReductionInstDesc(Select, MRK_SIntMax);
|
|
|
|
else if (m_SMin(m_Value(CmpLeft), m_Value(CmpRight)).match(Select))
|
|
|
|
return ReductionInstDesc(Select, MRK_SIntMin);
|
2013-05-05 01:54:48 +00:00
|
|
|
else if (m_OrdFMin(m_Value(CmpLeft), m_Value(CmpRight)).match(Select))
|
|
|
|
return ReductionInstDesc(Select, MRK_FloatMin);
|
|
|
|
else if (m_OrdFMax(m_Value(CmpLeft), m_Value(CmpRight)).match(Select))
|
|
|
|
return ReductionInstDesc(Select, MRK_FloatMax);
|
|
|
|
else if (m_UnordFMin(m_Value(CmpLeft), m_Value(CmpRight)).match(Select))
|
|
|
|
return ReductionInstDesc(Select, MRK_FloatMin);
|
|
|
|
else if (m_UnordFMax(m_Value(CmpLeft), m_Value(CmpRight)).match(Select))
|
|
|
|
return ReductionInstDesc(Select, MRK_FloatMax);
|
2013-04-19 21:03:36 +00:00
|
|
|
|
|
|
|
return ReductionInstDesc(false, I);
|
2013-04-18 17:22:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LoopVectorizationLegality::ReductionInstDesc
|
2012-10-19 23:05:40 +00:00
|
|
|
LoopVectorizationLegality::isReductionInstr(Instruction *I,
|
2013-04-18 17:22:34 +00:00
|
|
|
ReductionKind Kind,
|
2013-04-19 21:03:36 +00:00
|
|
|
ReductionInstDesc &Prev) {
|
2013-01-07 23:13:00 +00:00
|
|
|
bool FP = I->getType()->isFloatingPointTy();
|
|
|
|
bool FastMath = (FP && I->isCommutative() && I->isAssociative());
|
2012-12-10 21:39:02 +00:00
|
|
|
switch (I->getOpcode()) {
|
|
|
|
default:
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(false, I);
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::PHI:
|
2013-05-05 01:54:48 +00:00
|
|
|
if (FP && (Kind != RK_FloatMult && Kind != RK_FloatAdd &&
|
|
|
|
Kind != RK_FloatMinMax))
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(false, I);
|
2013-04-19 21:03:36 +00:00
|
|
|
return ReductionInstDesc(I, Prev.MinMaxKind);
|
2013-01-05 01:15:47 +00:00
|
|
|
case Instruction::Sub:
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::Add:
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(Kind == RK_IntegerAdd, I);
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::Mul:
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(Kind == RK_IntegerMult, I);
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::And:
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(Kind == RK_IntegerAnd, I);
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::Or:
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(Kind == RK_IntegerOr, I);
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::Xor:
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(Kind == RK_IntegerXor, I);
|
2013-01-07 23:13:00 +00:00
|
|
|
case Instruction::FMul:
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(Kind == RK_FloatMult && FastMath, I);
|
2013-01-07 23:13:00 +00:00
|
|
|
case Instruction::FAdd:
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(Kind == RK_FloatAdd && FastMath, I);
|
2013-05-05 01:54:48 +00:00
|
|
|
case Instruction::FCmp:
|
2013-04-18 17:22:34 +00:00
|
|
|
case Instruction::ICmp:
|
|
|
|
case Instruction::Select:
|
2013-05-05 01:54:48 +00:00
|
|
|
if (Kind != RK_IntegerMinMax &&
|
|
|
|
(!HasFunNoNaNAttr || Kind != RK_FloatMinMax))
|
2013-04-18 17:22:34 +00:00
|
|
|
return ReductionInstDesc(false, I);
|
2013-04-19 21:03:36 +00:00
|
|
|
return isMinMaxSelectCmpPattern(I, Prev);
|
2013-04-18 17:22:34 +00:00
|
|
|
}
|
2012-10-19 23:05:40 +00:00
|
|
|
}
|
|
|
|
|
2012-12-10 19:25:06 +00:00
|
|
|
LoopVectorizationLegality::InductionKind
|
|
|
|
LoopVectorizationLegality::isInductionVariable(PHINode *Phi) {
|
2012-11-17 00:27:03 +00:00
|
|
|
Type *PhiTy = Phi->getType();
|
|
|
|
// We only handle integer and pointer inductions variables.
|
|
|
|
if (!PhiTy->isIntegerTy() && !PhiTy->isPointerTy())
|
2013-01-08 17:23:17 +00:00
|
|
|
return IK_NoInduction;
|
2012-11-17 00:27:03 +00:00
|
|
|
|
2013-01-21 17:59:18 +00:00
|
|
|
// Check that the PHI is consecutive.
|
2012-10-20 08:26:33 +00:00
|
|
|
const SCEV *PhiScev = SE->getSCEV(Phi);
|
|
|
|
const SCEVAddRecExpr *AR = dyn_cast<SCEVAddRecExpr>(PhiScev);
|
|
|
|
if (!AR) {
|
|
|
|
DEBUG(dbgs() << "LV: PHI is not a poly recurrence.\n");
|
2013-01-08 17:23:17 +00:00
|
|
|
return IK_NoInduction;
|
2012-10-20 08:26:33 +00:00
|
|
|
}
|
|
|
|
const SCEV *Step = AR->getStepRecurrence(*SE);
|
|
|
|
|
2012-11-17 00:27:03 +00:00
|
|
|
// Integer inductions need to have a stride of one.
|
2012-12-10 19:25:06 +00:00
|
|
|
if (PhiTy->isIntegerTy()) {
|
|
|
|
if (Step->isOne())
|
2013-01-08 17:23:17 +00:00
|
|
|
return IK_IntInduction;
|
2012-12-10 19:25:06 +00:00
|
|
|
if (Step->isAllOnesValue())
|
2013-01-08 17:23:17 +00:00
|
|
|
return IK_ReverseIntInduction;
|
|
|
|
return IK_NoInduction;
|
2012-12-10 19:25:06 +00:00
|
|
|
}
|
2012-11-17 00:27:03 +00:00
|
|
|
|
|
|
|
// Calculate the pointer stride and check if it is consecutive.
|
|
|
|
const SCEVConstant *C = dyn_cast<SCEVConstant>(Step);
|
2012-12-10 19:25:06 +00:00
|
|
|
if (!C)
|
2013-01-08 17:23:17 +00:00
|
|
|
return IK_NoInduction;
|
2012-11-17 00:27:03 +00:00
|
|
|
|
|
|
|
assert(PhiTy->isPointerTy() && "The PHI must be a pointer");
|
|
|
|
uint64_t Size = DL->getTypeAllocSize(PhiTy->getPointerElementType());
|
2012-12-10 19:25:06 +00:00
|
|
|
if (C->getValue()->equalsInt(Size))
|
2013-01-08 17:23:17 +00:00
|
|
|
return IK_PtrInduction;
|
2013-01-23 01:35:00 +00:00
|
|
|
else if (C->getValue()->equalsInt(0 - Size))
|
|
|
|
return IK_ReversePtrInduction;
|
2012-12-10 19:25:06 +00:00
|
|
|
|
2013-01-08 17:23:17 +00:00
|
|
|
return IK_NoInduction;
|
2012-10-20 08:26:33 +00:00
|
|
|
}
|
|
|
|
|
2012-12-13 00:21:03 +00:00
|
|
|
bool LoopVectorizationLegality::isInductionVariable(const Value *V) {
|
|
|
|
Value *In0 = const_cast<Value*>(V);
|
|
|
|
PHINode *PN = dyn_cast_or_null<PHINode>(In0);
|
|
|
|
if (!PN)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return Inductions.count(PN);
|
|
|
|
}
|
|
|
|
|
2012-12-03 21:06:35 +00:00
|
|
|
bool LoopVectorizationLegality::blockNeedsPredication(BasicBlock *BB) {
|
|
|
|
assert(TheLoop->contains(BB) && "Unknown block used");
|
|
|
|
|
|
|
|
// Blocks that do not dominate the latch need predication.
|
|
|
|
BasicBlock* Latch = TheLoop->getLoopLatch();
|
|
|
|
return !DT->dominates(BB, Latch);
|
|
|
|
}
|
|
|
|
|
2013-06-28 20:46:27 +00:00
|
|
|
bool LoopVectorizationLegality::blockCanBePredicated(BasicBlock *BB,
|
|
|
|
SmallPtrSet<Value *, 8>& SafePtrs) {
|
2012-12-03 21:06:35 +00:00
|
|
|
for (BasicBlock::iterator it = BB->begin(), e = BB->end(); it != e; ++it) {
|
2013-05-15 01:44:30 +00:00
|
|
|
// We might be able to hoist the load.
|
2013-06-28 20:46:27 +00:00
|
|
|
if (it->mayReadFromMemory()) {
|
|
|
|
LoadInst *LI = dyn_cast<LoadInst>(it);
|
|
|
|
if (!LI || !SafePtrs.count(LI->getPointerOperand()))
|
|
|
|
return false;
|
|
|
|
}
|
2013-05-15 01:44:30 +00:00
|
|
|
|
2013-05-15 02:02:45 +00:00
|
|
|
// We don't predicate stores at the moment.
|
2013-05-15 01:44:30 +00:00
|
|
|
if (it->mayWriteToMemory() || it->mayThrow())
|
2012-12-03 21:06:35 +00:00
|
|
|
return false;
|
|
|
|
|
2012-12-12 01:31:10 +00:00
|
|
|
// The instructions below can trap.
|
2012-12-03 21:06:35 +00:00
|
|
|
switch (it->getOpcode()) {
|
2012-12-10 21:39:02 +00:00
|
|
|
default: continue;
|
|
|
|
case Instruction::UDiv:
|
|
|
|
case Instruction::SDiv:
|
|
|
|
case Instruction::URem:
|
|
|
|
case Instruction::SRem:
|
|
|
|
return false;
|
2012-12-03 21:06:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-01-28 16:02:45 +00:00
|
|
|
LoopVectorizationCostModel::VectorizationFactor
|
2012-12-12 01:11:46 +00:00
|
|
|
LoopVectorizationCostModel::selectVectorizationFactor(bool OptForSize,
|
2013-01-04 17:48:25 +00:00
|
|
|
unsigned UserVF) {
|
2013-01-28 16:02:45 +00:00
|
|
|
// Width 1 means no vectorize
|
|
|
|
VectorizationFactor Factor = { 1U, 0U };
|
2012-12-12 01:11:46 +00:00
|
|
|
if (OptForSize && Legal->getRuntimePointerCheck()->Need) {
|
|
|
|
DEBUG(dbgs() << "LV: Aborting. Runtime ptr check is required in Os.\n");
|
2013-01-28 16:02:45 +00:00
|
|
|
return Factor;
|
2012-12-12 01:11:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Find the trip count.
|
|
|
|
unsigned TC = SE->getSmallConstantTripCount(TheLoop, TheLoop->getLoopLatch());
|
|
|
|
DEBUG(dbgs() << "LV: Found trip count:"<<TC<<"\n");
|
|
|
|
|
2013-01-09 22:29:00 +00:00
|
|
|
unsigned WidestType = getWidestType();
|
|
|
|
unsigned WidestRegister = TTI.getRegisterBitWidth(true);
|
2013-06-24 12:09:15 +00:00
|
|
|
unsigned MaxSafeDepDist = -1U;
|
|
|
|
if (Legal->getMaxSafeDepDistBytes() != -1U)
|
|
|
|
MaxSafeDepDist = Legal->getMaxSafeDepDistBytes() * 8;
|
2013-07-31 22:17:45 +00:00
|
|
|
WidestRegister = ((WidestRegister < MaxSafeDepDist) ?
|
|
|
|
WidestRegister : MaxSafeDepDist);
|
2013-01-09 22:29:00 +00:00
|
|
|
unsigned MaxVectorSize = WidestRegister / WidestType;
|
|
|
|
DEBUG(dbgs() << "LV: The Widest type: " << WidestType << " bits.\n");
|
|
|
|
DEBUG(dbgs() << "LV: The Widest register is:" << WidestRegister << "bits.\n");
|
|
|
|
|
|
|
|
if (MaxVectorSize == 0) {
|
|
|
|
DEBUG(dbgs() << "LV: The target has no vector registers.\n");
|
2013-01-15 18:25:16 +00:00
|
|
|
MaxVectorSize = 1;
|
2013-01-09 22:29:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assert(MaxVectorSize <= 32 && "Did not expect to pack so many elements"
|
2013-01-20 05:24:29 +00:00
|
|
|
" into one vector!");
|
2013-01-11 07:11:59 +00:00
|
|
|
|
2012-12-12 01:11:46 +00:00
|
|
|
unsigned VF = MaxVectorSize;
|
|
|
|
|
|
|
|
// If we optimize the program for size, avoid creating the tail loop.
|
|
|
|
if (OptForSize) {
|
|
|
|
// If we are unable to calculate the trip count then don't try to vectorize.
|
|
|
|
if (TC < 2) {
|
|
|
|
DEBUG(dbgs() << "LV: Aborting. A tail loop is required in Os.\n");
|
2013-01-28 16:02:45 +00:00
|
|
|
return Factor;
|
2012-12-12 01:11:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Find the maximum SIMD width that can fit within the trip count.
|
|
|
|
VF = TC % MaxVectorSize;
|
|
|
|
|
|
|
|
if (VF == 0)
|
|
|
|
VF = MaxVectorSize;
|
|
|
|
|
|
|
|
// If the trip count that we found modulo the vectorization factor is not
|
|
|
|
// zero then we require a tail.
|
|
|
|
if (VF < 2) {
|
|
|
|
DEBUG(dbgs() << "LV: Aborting. A tail loop is required in Os.\n");
|
2013-01-28 16:02:45 +00:00
|
|
|
return Factor;
|
2012-12-12 01:11:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (UserVF != 0) {
|
|
|
|
assert(isPowerOf2_32(UserVF) && "VF needs to be a power of two");
|
|
|
|
DEBUG(dbgs() << "LV: Using user VF "<<UserVF<<".\n");
|
|
|
|
|
2013-01-28 16:02:45 +00:00
|
|
|
Factor.Width = UserVF;
|
|
|
|
return Factor;
|
2012-12-12 01:11:46 +00:00
|
|
|
}
|
|
|
|
|
2012-10-24 20:36:32 +00:00
|
|
|
float Cost = expectedCost(1);
|
|
|
|
unsigned Width = 1;
|
|
|
|
DEBUG(dbgs() << "LV: Scalar loop costs: "<< (int)Cost << ".\n");
|
|
|
|
for (unsigned i=2; i <= VF; i*=2) {
|
|
|
|
// Notice that the vector loop needs to be executed less times, so
|
|
|
|
// we need to divide the cost of the vector loops by the width of
|
|
|
|
// the vector elements.
|
|
|
|
float VectorCost = expectedCost(i) / (float)i;
|
|
|
|
DEBUG(dbgs() << "LV: Vector loop of width "<< i << " costs: " <<
|
|
|
|
(int)VectorCost << ".\n");
|
|
|
|
if (VectorCost < Cost) {
|
|
|
|
Cost = VectorCost;
|
|
|
|
Width = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV: Selecting VF = : "<< Width << ".\n");
|
2013-01-28 16:02:45 +00:00
|
|
|
Factor.Width = Width;
|
|
|
|
Factor.Cost = Width * Cost;
|
|
|
|
return Factor;
|
2012-10-24 20:36:32 +00:00
|
|
|
}
|
|
|
|
|
2013-01-09 22:29:00 +00:00
|
|
|
unsigned LoopVectorizationCostModel::getWidestType() {
|
|
|
|
unsigned MaxWidth = 8;
|
|
|
|
|
|
|
|
// For each block.
|
|
|
|
for (Loop::block_iterator bb = TheLoop->block_begin(),
|
|
|
|
be = TheLoop->block_end(); bb != be; ++bb) {
|
|
|
|
BasicBlock *BB = *bb;
|
|
|
|
|
|
|
|
// For each instruction in the loop.
|
|
|
|
for (BasicBlock::iterator it = BB->begin(), e = BB->end(); it != e; ++it) {
|
2013-01-11 07:11:59 +00:00
|
|
|
Type *T = it->getType();
|
|
|
|
|
|
|
|
// Only examine Loads, Stores and PHINodes.
|
|
|
|
if (!isa<LoadInst>(it) && !isa<StoreInst>(it) && !isa<PHINode>(it))
|
2013-01-09 22:29:00 +00:00
|
|
|
continue;
|
|
|
|
|
2013-01-11 07:11:59 +00:00
|
|
|
// Examine PHI nodes that are reduction variables.
|
|
|
|
if (PHINode *PN = dyn_cast<PHINode>(it))
|
|
|
|
if (!Legal->getReductionVars()->count(PN))
|
|
|
|
continue;
|
2013-01-09 22:29:00 +00:00
|
|
|
|
2013-01-11 07:11:59 +00:00
|
|
|
// Examine the stored values.
|
2013-02-13 21:12:29 +00:00
|
|
|
if (StoreInst *ST = dyn_cast<StoreInst>(it))
|
2013-01-09 22:29:00 +00:00
|
|
|
T = ST->getValueOperand()->getType();
|
|
|
|
|
2013-02-05 15:08:02 +00:00
|
|
|
// Ignore loaded pointer types and stored pointer types that are not
|
|
|
|
// consecutive. However, we do want to take consecutive stores/loads of
|
|
|
|
// pointer vectors into account.
|
2013-02-13 21:12:29 +00:00
|
|
|
if (T->isPointerTy() && !isConsecutiveLoadOrStore(it))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
MaxWidth = std::max(MaxWidth,
|
|
|
|
(unsigned)DL->getTypeSizeInBits(T->getScalarType()));
|
2013-01-09 22:29:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return MaxWidth;
|
|
|
|
}
|
|
|
|
|
2013-01-04 17:48:25 +00:00
|
|
|
unsigned
|
|
|
|
LoopVectorizationCostModel::selectUnrollFactor(bool OptForSize,
|
2013-01-20 05:24:29 +00:00
|
|
|
unsigned UserUF,
|
|
|
|
unsigned VF,
|
|
|
|
unsigned LoopCost) {
|
|
|
|
|
|
|
|
// -- The unroll heuristics --
|
|
|
|
// We unroll the loop in order to expose ILP and reduce the loop overhead.
|
|
|
|
// There are many micro-architectural considerations that we can't predict
|
|
|
|
// at this level. For example frontend pressure (on decode or fetch) due to
|
|
|
|
// code size, or the number and capabilities of the execution ports.
|
|
|
|
//
|
|
|
|
// We use the following heuristics to select the unroll factor:
|
|
|
|
// 1. If the code has reductions the we unroll in order to break the cross
|
|
|
|
// iteration dependency.
|
|
|
|
// 2. If the loop is really small then we unroll in order to reduce the loop
|
|
|
|
// overhead.
|
|
|
|
// 3. We don't unroll if we think that we will spill registers to memory due
|
|
|
|
// to the increased register pressure.
|
|
|
|
|
2013-01-04 17:48:25 +00:00
|
|
|
// Use the user preference, unless 'auto' is selected.
|
|
|
|
if (UserUF != 0)
|
|
|
|
return UserUF;
|
|
|
|
|
|
|
|
// When we optimize for size we don't unroll.
|
|
|
|
if (OptForSize)
|
|
|
|
return 1;
|
|
|
|
|
2013-06-24 12:09:15 +00:00
|
|
|
// We used the distance for the unroll factor.
|
|
|
|
if (Legal->getMaxSafeDepDistBytes() != -1U)
|
|
|
|
return 1;
|
|
|
|
|
2013-01-07 21:54:51 +00:00
|
|
|
// Do not unroll loops with a relatively small trip count.
|
|
|
|
unsigned TC = SE->getSmallConstantTripCount(TheLoop,
|
|
|
|
TheLoop->getLoopLatch());
|
|
|
|
if (TC > 1 && TC < TinyTripCountUnrollThreshold)
|
|
|
|
return 1;
|
|
|
|
|
2013-01-07 11:12:29 +00:00
|
|
|
unsigned TargetVectorRegisters = TTI.getNumberOfRegisters(true);
|
2013-01-04 17:48:25 +00:00
|
|
|
DEBUG(dbgs() << "LV: The target has " << TargetVectorRegisters <<
|
|
|
|
" vector registers\n");
|
|
|
|
|
|
|
|
LoopVectorizationCostModel::RegisterUsage R = calculateRegisterUsage();
|
|
|
|
// We divide by these constants so assume that we have at least one
|
|
|
|
// instruction that uses at least one register.
|
|
|
|
R.MaxLocalUsers = std::max(R.MaxLocalUsers, 1U);
|
|
|
|
R.NumInstructions = std::max(R.NumInstructions, 1U);
|
|
|
|
|
|
|
|
// We calculate the unroll factor using the following formula.
|
|
|
|
// Subtract the number of loop invariants from the number of available
|
|
|
|
// registers. These registers are used by all of the unrolled instances.
|
|
|
|
// Next, divide the remaining registers by the number of registers that is
|
|
|
|
// required by the loop, in order to estimate how many parallel instances
|
|
|
|
// fit without causing spills.
|
|
|
|
unsigned UF = (TargetVectorRegisters - R.LoopInvariantRegs) / R.MaxLocalUsers;
|
|
|
|
|
|
|
|
// Clamp the unroll factor ranges to reasonable factors.
|
2013-01-09 01:15:42 +00:00
|
|
|
unsigned MaxUnrollSize = TTI.getMaximumUnrollFactor();
|
2013-01-20 05:24:29 +00:00
|
|
|
|
|
|
|
// If we did not calculate the cost for VF (because the user selected the VF)
|
|
|
|
// then we calculate the cost of VF here.
|
|
|
|
if (LoopCost == 0)
|
|
|
|
LoopCost = expectedCost(VF);
|
|
|
|
|
|
|
|
// Clamp the calculated UF to be between the 1 and the max unroll factor
|
|
|
|
// that the target allows.
|
2013-01-04 17:48:25 +00:00
|
|
|
if (UF > MaxUnrollSize)
|
|
|
|
UF = MaxUnrollSize;
|
|
|
|
else if (UF < 1)
|
|
|
|
UF = 1;
|
|
|
|
|
2013-08-26 22:33:26 +00:00
|
|
|
bool HasReductions = Legal->getReductionVars()->size();
|
|
|
|
|
|
|
|
// Decide if we want to unroll if we decided that it is legal to vectorize
|
|
|
|
// but not profitable.
|
|
|
|
if (VF == 1) {
|
|
|
|
if (TheLoop->getNumBlocks() > 1 || !HasReductions ||
|
|
|
|
LoopCost > SmallLoopCost)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return UF;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (HasReductions) {
|
2013-01-20 05:24:29 +00:00
|
|
|
DEBUG(dbgs() << "LV: Unrolling because of reductions. \n");
|
|
|
|
return UF;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We want to unroll tiny loops in order to reduce the loop overhead.
|
|
|
|
// We assume that the cost overhead is 1 and we use the cost model
|
|
|
|
// to estimate the cost of the loop and unroll until the cost of the
|
|
|
|
// loop overhead is about 5% of the cost of the loop.
|
|
|
|
DEBUG(dbgs() << "LV: Loop cost is "<< LoopCost <<" \n");
|
2013-08-26 22:33:26 +00:00
|
|
|
if (LoopCost < SmallLoopCost) {
|
2013-01-20 05:24:29 +00:00
|
|
|
DEBUG(dbgs() << "LV: Unrolling to reduce branch cost. \n");
|
2013-08-26 22:33:26 +00:00
|
|
|
unsigned NewUF = SmallLoopCost / (LoopCost + 1);
|
2013-01-20 05:24:29 +00:00
|
|
|
return std::min(NewUF, UF);
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV: Not Unrolling. \n");
|
|
|
|
return 1;
|
2013-01-04 17:48:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LoopVectorizationCostModel::RegisterUsage
|
|
|
|
LoopVectorizationCostModel::calculateRegisterUsage() {
|
|
|
|
// This function calculates the register usage by measuring the highest number
|
|
|
|
// of values that are alive at a single location. Obviously, this is a very
|
|
|
|
// rough estimation. We scan the loop in a topological order in order and
|
|
|
|
// assign a number to each instruction. We use RPO to ensure that defs are
|
|
|
|
// met before their users. We assume that each instruction that has in-loop
|
|
|
|
// users starts an interval. We record every time that an in-loop value is
|
|
|
|
// used, so we have a list of the first and last occurrences of each
|
|
|
|
// instruction. Next, we transpose this data structure into a multi map that
|
|
|
|
// holds the list of intervals that *end* at a specific location. This multi
|
|
|
|
// map allows us to perform a linear search. We scan the instructions linearly
|
|
|
|
// and record each time that a new interval starts, by placing it in a set.
|
|
|
|
// If we find this value in the multi-map then we remove it from the set.
|
|
|
|
// The max register usage is the maximum size of the set.
|
|
|
|
// We also search for instructions that are defined outside the loop, but are
|
|
|
|
// used inside the loop. We need this number separately from the max-interval
|
|
|
|
// usage number because when we unroll, loop-invariant values do not take
|
|
|
|
// more register.
|
|
|
|
LoopBlocksDFS DFS(TheLoop);
|
|
|
|
DFS.perform(LI);
|
|
|
|
|
|
|
|
RegisterUsage R;
|
|
|
|
R.NumInstructions = 0;
|
|
|
|
|
|
|
|
// Each 'key' in the map opens a new interval. The values
|
|
|
|
// of the map are the index of the 'last seen' usage of the
|
|
|
|
// instruction that is the key.
|
|
|
|
typedef DenseMap<Instruction*, unsigned> IntervalMap;
|
|
|
|
// Maps instruction to its index.
|
|
|
|
DenseMap<unsigned, Instruction*> IdxToInstr;
|
|
|
|
// Marks the end of each interval.
|
|
|
|
IntervalMap EndPoint;
|
|
|
|
// Saves the list of instruction indices that are used in the loop.
|
|
|
|
SmallSet<Instruction*, 8> Ends;
|
|
|
|
// Saves the list of values that are used in the loop but are
|
|
|
|
// defined outside the loop, such as arguments and constants.
|
|
|
|
SmallPtrSet<Value*, 8> LoopInvariants;
|
|
|
|
|
|
|
|
unsigned Index = 0;
|
|
|
|
for (LoopBlocksDFS::RPOIterator bb = DFS.beginRPO(),
|
|
|
|
be = DFS.endRPO(); bb != be; ++bb) {
|
|
|
|
R.NumInstructions += (*bb)->size();
|
|
|
|
for (BasicBlock::iterator it = (*bb)->begin(), e = (*bb)->end(); it != e;
|
|
|
|
++it) {
|
|
|
|
Instruction *I = it;
|
|
|
|
IdxToInstr[Index++] = I;
|
|
|
|
|
|
|
|
// Save the end location of each USE.
|
|
|
|
for (unsigned i = 0; i < I->getNumOperands(); ++i) {
|
|
|
|
Value *U = I->getOperand(i);
|
|
|
|
Instruction *Instr = dyn_cast<Instruction>(U);
|
|
|
|
|
|
|
|
// Ignore non-instruction values such as arguments, constants, etc.
|
|
|
|
if (!Instr) continue;
|
|
|
|
|
|
|
|
// If this instruction is outside the loop then record it and continue.
|
|
|
|
if (!TheLoop->contains(Instr)) {
|
|
|
|
LoopInvariants.insert(Instr);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Overwrite previous end points.
|
|
|
|
EndPoint[Instr] = Index;
|
|
|
|
Ends.insert(Instr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Saves the list of intervals that end with the index in 'key'.
|
|
|
|
typedef SmallVector<Instruction*, 2> InstrList;
|
|
|
|
DenseMap<unsigned, InstrList> TransposeEnds;
|
|
|
|
|
|
|
|
// Transpose the EndPoints to a list of values that end at each index.
|
|
|
|
for (IntervalMap::iterator it = EndPoint.begin(), e = EndPoint.end();
|
|
|
|
it != e; ++it)
|
|
|
|
TransposeEnds[it->second].push_back(it->first);
|
|
|
|
|
|
|
|
SmallSet<Instruction*, 8> OpenIntervals;
|
|
|
|
unsigned MaxUsage = 0;
|
|
|
|
|
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV(REG): Calculating max register usage:\n");
|
|
|
|
for (unsigned int i = 0; i < Index; ++i) {
|
|
|
|
Instruction *I = IdxToInstr[i];
|
|
|
|
// Ignore instructions that are never used within the loop.
|
|
|
|
if (!Ends.count(I)) continue;
|
|
|
|
|
|
|
|
// Remove all of the instructions that end at this location.
|
|
|
|
InstrList &List = TransposeEnds[i];
|
2013-01-04 21:08:44 +00:00
|
|
|
for (unsigned int j=0, e = List.size(); j < e; ++j)
|
|
|
|
OpenIntervals.erase(List[j]);
|
2013-01-04 17:48:25 +00:00
|
|
|
|
|
|
|
// Count the number of live interals.
|
|
|
|
MaxUsage = std::max(MaxUsage, OpenIntervals.size());
|
|
|
|
|
|
|
|
DEBUG(dbgs() << "LV(REG): At #" << i << " Interval # " <<
|
|
|
|
OpenIntervals.size() <<"\n");
|
|
|
|
|
|
|
|
// Add the current instruction to the list of open intervals.
|
|
|
|
OpenIntervals.insert(I);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned Invariant = LoopInvariants.size();
|
|
|
|
DEBUG(dbgs() << "LV(REG): Found max usage: " << MaxUsage << " \n");
|
|
|
|
DEBUG(dbgs() << "LV(REG): Found invariant usage: " << Invariant << " \n");
|
|
|
|
DEBUG(dbgs() << "LV(REG): LoopSize: " << R.NumInstructions << " \n");
|
|
|
|
|
|
|
|
R.LoopInvariantRegs = Invariant;
|
|
|
|
R.MaxLocalUsers = MaxUsage;
|
|
|
|
return R;
|
|
|
|
}
|
|
|
|
|
2012-10-24 20:36:32 +00:00
|
|
|
unsigned LoopVectorizationCostModel::expectedCost(unsigned VF) {
|
|
|
|
unsigned Cost = 0;
|
|
|
|
|
2012-12-03 22:46:31 +00:00
|
|
|
// For each block.
|
|
|
|
for (Loop::block_iterator bb = TheLoop->block_begin(),
|
|
|
|
be = TheLoop->block_end(); bb != be; ++bb) {
|
|
|
|
unsigned BlockCost = 0;
|
|
|
|
BasicBlock *BB = *bb;
|
2012-12-04 00:49:28 +00:00
|
|
|
|
2012-12-03 22:46:31 +00:00
|
|
|
// For each instruction in the old loop.
|
|
|
|
for (BasicBlock::iterator it = BB->begin(), e = BB->end(); it != e; ++it) {
|
2013-03-09 16:27:27 +00:00
|
|
|
// Skip dbg intrinsics.
|
|
|
|
if (isa<DbgInfoIntrinsic>(it))
|
2013-03-09 15:56:34 +00:00
|
|
|
continue;
|
|
|
|
|
2012-12-03 22:46:31 +00:00
|
|
|
unsigned C = getInstructionCost(it, VF);
|
2013-07-22 17:10:48 +00:00
|
|
|
BlockCost += C;
|
2012-12-03 22:46:31 +00:00
|
|
|
DEBUG(dbgs() << "LV: Found an estimated cost of "<< C <<" for VF " <<
|
|
|
|
VF << " For instruction: "<< *it << "\n");
|
|
|
|
}
|
|
|
|
|
2012-12-04 07:11:52 +00:00
|
|
|
// We assume that if-converted blocks have a 50% chance of being executed.
|
|
|
|
// When the code is scalar then some of the blocks are avoided due to CF.
|
|
|
|
// When the code is vectorized we execute all code paths.
|
2013-07-22 17:10:48 +00:00
|
|
|
if (VF == 1 && Legal->blockNeedsPredication(*bb))
|
2012-12-04 07:11:52 +00:00
|
|
|
BlockCost /= 2;
|
2012-12-04 00:49:28 +00:00
|
|
|
|
2012-12-03 22:46:31 +00:00
|
|
|
Cost += BlockCost;
|
2012-10-24 20:36:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return Cost;
|
|
|
|
}
|
|
|
|
|
2013-07-12 19:16:02 +00:00
|
|
|
/// \brief Check whether the address computation for a non-consecutive memory
|
|
|
|
/// access looks like an unlikely candidate for being merged into the indexing
|
|
|
|
/// mode.
|
|
|
|
///
|
|
|
|
/// We look for a GEP which has one index that is an induction variable and all
|
|
|
|
/// other indices are loop invariant. If the stride of this access is also
|
|
|
|
/// within a small bound we decide that this address computation can likely be
|
|
|
|
/// merged into the addressing mode.
|
|
|
|
/// In all other cases, we identify the address computation as complex.
|
|
|
|
static bool isLikelyComplexAddressComputation(Value *Ptr,
|
|
|
|
LoopVectorizationLegality *Legal,
|
|
|
|
ScalarEvolution *SE,
|
|
|
|
const Loop *TheLoop) {
|
|
|
|
GetElementPtrInst *Gep = dyn_cast<GetElementPtrInst>(Ptr);
|
|
|
|
if (!Gep)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// We are looking for a gep with all loop invariant indices except for one
|
|
|
|
// which should be an induction variable.
|
|
|
|
unsigned NumOperands = Gep->getNumOperands();
|
|
|
|
for (unsigned i = 1; i < NumOperands; ++i) {
|
|
|
|
Value *Opd = Gep->getOperand(i);
|
|
|
|
if (!SE->isLoopInvariant(SE->getSCEV(Opd), TheLoop) &&
|
|
|
|
!Legal->isInductionVariable(Opd))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now we know we have a GEP ptr, %inv, %ind, %inv. Make sure that the step
|
|
|
|
// can likely be merged into the address computation.
|
|
|
|
unsigned MaxMergeDistance = 64;
|
|
|
|
|
|
|
|
const SCEVAddRecExpr *AddRec = dyn_cast<SCEVAddRecExpr>(SE->getSCEV(Ptr));
|
|
|
|
if (!AddRec)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Check the step is constant.
|
|
|
|
const SCEV *Step = AddRec->getStepRecurrence(*SE);
|
|
|
|
// Calculate the pointer stride and check if it is consecutive.
|
|
|
|
const SCEVConstant *C = dyn_cast<SCEVConstant>(Step);
|
|
|
|
if (!C)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
const APInt &APStepVal = C->getValue()->getValue();
|
|
|
|
|
|
|
|
// Huge step value - give up.
|
|
|
|
if (APStepVal.getBitWidth() > 64)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
int64_t StepVal = APStepVal.getSExtValue();
|
|
|
|
|
|
|
|
return StepVal > MaxMergeDistance;
|
|
|
|
}
|
|
|
|
|
2012-10-24 20:36:32 +00:00
|
|
|
unsigned
|
|
|
|
LoopVectorizationCostModel::getInstructionCost(Instruction *I, unsigned VF) {
|
2012-10-26 23:49:28 +00:00
|
|
|
// If we know that this instruction will remain uniform, check the cost of
|
|
|
|
// the scalar version.
|
|
|
|
if (Legal->isUniformAfterVectorization(I))
|
|
|
|
VF = 1;
|
|
|
|
|
2012-10-25 21:03:48 +00:00
|
|
|
Type *RetTy = I->getType();
|
|
|
|
Type *VectorTy = ToVectorTy(RetTy, VF);
|
|
|
|
|
|
|
|
// TODO: We need to estimate the cost of intrinsic calls.
|
2012-10-24 20:36:32 +00:00
|
|
|
switch (I->getOpcode()) {
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::GetElementPtr:
|
2013-02-08 14:50:48 +00:00
|
|
|
// We mark this instruction as zero-cost because the cost of GEPs in
|
|
|
|
// vectorized code depends on whether the corresponding memory instruction
|
|
|
|
// is scalarized or not. Therefore, we handle GEPs with the memory
|
|
|
|
// instruction cost.
|
2012-12-10 21:39:02 +00:00
|
|
|
return 0;
|
|
|
|
case Instruction::Br: {
|
2013-01-07 11:12:29 +00:00
|
|
|
return TTI.getCFInstrCost(I->getOpcode());
|
2012-12-10 21:39:02 +00:00
|
|
|
}
|
|
|
|
case Instruction::PHI:
|
|
|
|
//TODO: IF-converted IFs become selects.
|
|
|
|
return 0;
|
|
|
|
case Instruction::Add:
|
|
|
|
case Instruction::FAdd:
|
|
|
|
case Instruction::Sub:
|
|
|
|
case Instruction::FSub:
|
|
|
|
case Instruction::Mul:
|
|
|
|
case Instruction::FMul:
|
|
|
|
case Instruction::UDiv:
|
|
|
|
case Instruction::SDiv:
|
|
|
|
case Instruction::FDiv:
|
|
|
|
case Instruction::URem:
|
|
|
|
case Instruction::SRem:
|
|
|
|
case Instruction::FRem:
|
|
|
|
case Instruction::Shl:
|
|
|
|
case Instruction::LShr:
|
|
|
|
case Instruction::AShr:
|
|
|
|
case Instruction::And:
|
|
|
|
case Instruction::Or:
|
2013-04-04 23:26:27 +00:00
|
|
|
case Instruction::Xor: {
|
|
|
|
// Certain instructions can be cheaper to vectorize if they have a constant
|
|
|
|
// second vector operand. One example of this are shifts on x86.
|
|
|
|
TargetTransformInfo::OperandValueKind Op1VK =
|
|
|
|
TargetTransformInfo::OK_AnyValue;
|
|
|
|
TargetTransformInfo::OperandValueKind Op2VK =
|
|
|
|
TargetTransformInfo::OK_AnyValue;
|
|
|
|
|
|
|
|
if (isa<ConstantInt>(I->getOperand(1)))
|
|
|
|
Op2VK = TargetTransformInfo::OK_UniformConstantValue;
|
|
|
|
|
|
|
|
return TTI.getArithmeticInstrCost(I->getOpcode(), VectorTy, Op1VK, Op2VK);
|
|
|
|
}
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::Select: {
|
|
|
|
SelectInst *SI = cast<SelectInst>(I);
|
|
|
|
const SCEV *CondSCEV = SE->getSCEV(SI->getCondition());
|
|
|
|
bool ScalarCond = (SE->isLoopInvariant(CondSCEV, TheLoop));
|
|
|
|
Type *CondTy = SI->getCondition()->getType();
|
2013-03-14 18:54:36 +00:00
|
|
|
if (!ScalarCond)
|
2012-12-10 21:39:02 +00:00
|
|
|
CondTy = VectorType::get(CondTy, VF);
|
|
|
|
|
2013-01-07 11:12:29 +00:00
|
|
|
return TTI.getCmpSelInstrCost(I->getOpcode(), VectorTy, CondTy);
|
2012-12-10 21:39:02 +00:00
|
|
|
}
|
|
|
|
case Instruction::ICmp:
|
|
|
|
case Instruction::FCmp: {
|
|
|
|
Type *ValTy = I->getOperand(0)->getType();
|
|
|
|
VectorTy = ToVectorTy(ValTy, VF);
|
2013-01-07 11:12:29 +00:00
|
|
|
return TTI.getCmpSelInstrCost(I->getOpcode(), VectorTy);
|
2012-12-10 21:39:02 +00:00
|
|
|
}
|
2013-02-07 19:05:21 +00:00
|
|
|
case Instruction::Store:
|
|
|
|
case Instruction::Load: {
|
|
|
|
StoreInst *SI = dyn_cast<StoreInst>(I);
|
|
|
|
LoadInst *LI = dyn_cast<LoadInst>(I);
|
|
|
|
Type *ValTy = (SI ? SI->getValueOperand()->getType() :
|
|
|
|
LI->getType());
|
|
|
|
VectorTy = ToVectorTy(ValTy, VF);
|
|
|
|
|
|
|
|
unsigned Alignment = SI ? SI->getAlignment() : LI->getAlignment();
|
|
|
|
unsigned AS = SI ? SI->getPointerAddressSpace() :
|
|
|
|
LI->getPointerAddressSpace();
|
|
|
|
Value *Ptr = SI ? SI->getPointerOperand() : LI->getPointerOperand();
|
2013-02-08 14:50:48 +00:00
|
|
|
// We add the cost of address computation here instead of with the gep
|
|
|
|
// instruction because only here we know whether the operation is
|
|
|
|
// scalarized.
|
2013-02-07 19:05:21 +00:00
|
|
|
if (VF == 1)
|
2013-02-08 14:50:48 +00:00
|
|
|
return TTI.getAddressComputationCost(VectorTy) +
|
|
|
|
TTI.getMemoryOpCost(I->getOpcode(), VectorTy, Alignment, AS);
|
2013-02-05 18:46:41 +00:00
|
|
|
|
2013-02-07 19:05:21 +00:00
|
|
|
// Scalarized loads/stores.
|
2013-04-24 16:16:03 +00:00
|
|
|
int ConsecutiveStride = Legal->isConsecutivePtr(Ptr);
|
|
|
|
bool Reverse = ConsecutiveStride < 0;
|
2013-04-24 16:16:01 +00:00
|
|
|
unsigned ScalarAllocatedSize = DL->getTypeAllocSize(ValTy);
|
|
|
|
unsigned VectorElementSize = DL->getTypeStoreSize(VectorTy)/VF;
|
2013-04-24 16:16:03 +00:00
|
|
|
if (!ConsecutiveStride || ScalarAllocatedSize != VectorElementSize) {
|
2013-07-12 19:16:02 +00:00
|
|
|
bool IsComplexComputation =
|
|
|
|
isLikelyComplexAddressComputation(Ptr, Legal, SE, TheLoop);
|
2013-02-07 19:05:21 +00:00
|
|
|
unsigned Cost = 0;
|
|
|
|
// The cost of extracting from the value vector and pointer vector.
|
|
|
|
Type *PtrTy = ToVectorTy(Ptr->getType(), VF);
|
|
|
|
for (unsigned i = 0; i < VF; ++i) {
|
|
|
|
// The cost of extracting the pointer operand.
|
|
|
|
Cost += TTI.getVectorInstrCost(Instruction::ExtractElement, PtrTy, i);
|
|
|
|
// In case of STORE, the cost of ExtractElement from the vector.
|
|
|
|
// In case of LOAD, the cost of InsertElement into the returned
|
|
|
|
// vector.
|
|
|
|
Cost += TTI.getVectorInstrCost(SI ? Instruction::ExtractElement :
|
|
|
|
Instruction::InsertElement,
|
|
|
|
VectorTy, i);
|
|
|
|
}
|
|
|
|
|
2013-02-08 14:50:48 +00:00
|
|
|
// The cost of the scalar loads/stores.
|
2013-07-12 19:16:02 +00:00
|
|
|
Cost += VF * TTI.getAddressComputationCost(PtrTy, IsComplexComputation);
|
2013-02-07 19:05:21 +00:00
|
|
|
Cost += VF * TTI.getMemoryOpCost(I->getOpcode(), ValTy->getScalarType(),
|
|
|
|
Alignment, AS);
|
|
|
|
return Cost;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wide load/stores.
|
2013-02-08 14:50:48 +00:00
|
|
|
unsigned Cost = TTI.getAddressComputationCost(VectorTy);
|
|
|
|
Cost += TTI.getMemoryOpCost(I->getOpcode(), VectorTy, Alignment, AS);
|
|
|
|
|
2013-02-07 19:05:21 +00:00
|
|
|
if (Reverse)
|
|
|
|
Cost += TTI.getShuffleCost(TargetTransformInfo::SK_Reverse,
|
|
|
|
VectorTy, 0);
|
|
|
|
return Cost;
|
|
|
|
}
|
2012-12-10 21:39:02 +00:00
|
|
|
case Instruction::ZExt:
|
|
|
|
case Instruction::SExt:
|
|
|
|
case Instruction::FPToUI:
|
|
|
|
case Instruction::FPToSI:
|
|
|
|
case Instruction::FPExt:
|
|
|
|
case Instruction::PtrToInt:
|
|
|
|
case Instruction::IntToPtr:
|
|
|
|
case Instruction::SIToFP:
|
|
|
|
case Instruction::UIToFP:
|
|
|
|
case Instruction::Trunc:
|
|
|
|
case Instruction::FPTrunc:
|
|
|
|
case Instruction::BitCast: {
|
2012-12-13 00:21:03 +00:00
|
|
|
// We optimize the truncation of induction variable.
|
|
|
|
// The cost of these is the same as the scalar operation.
|
|
|
|
if (I->getOpcode() == Instruction::Trunc &&
|
|
|
|
Legal->isInductionVariable(I->getOperand(0)))
|
2013-01-07 11:12:29 +00:00
|
|
|
return TTI.getCastInstrCost(I->getOpcode(), I->getType(),
|
|
|
|
I->getOperand(0)->getType());
|
2012-12-13 00:21:03 +00:00
|
|
|
|
2012-12-10 21:39:02 +00:00
|
|
|
Type *SrcVecTy = ToVectorTy(I->getOperand(0)->getType(), VF);
|
2013-01-07 11:12:29 +00:00
|
|
|
return TTI.getCastInstrCost(I->getOpcode(), VectorTy, SrcVecTy);
|
2012-12-10 21:39:02 +00:00
|
|
|
}
|
|
|
|
case Instruction::Call: {
|
2013-02-27 15:24:19 +00:00
|
|
|
CallInst *CI = cast<CallInst>(I);
|
|
|
|
Intrinsic::ID ID = getIntrinsicIDForCall(CI, TLI);
|
|
|
|
assert(ID && "Not an intrinsic call!");
|
|
|
|
Type *RetTy = ToVectorTy(CI->getType(), VF);
|
2012-12-10 21:39:02 +00:00
|
|
|
SmallVector<Type*, 4> Tys;
|
2013-02-27 15:24:19 +00:00
|
|
|
for (unsigned i = 0, ie = CI->getNumArgOperands(); i != ie; ++i)
|
|
|
|
Tys.push_back(ToVectorTy(CI->getArgOperand(i)->getType(), VF));
|
|
|
|
return TTI.getIntrinsicInstrCost(ID, RetTy, Tys);
|
2012-12-10 21:39:02 +00:00
|
|
|
}
|
|
|
|
default: {
|
|
|
|
// We are scalarizing the instruction. Return the cost of the scalar
|
|
|
|
// instruction, plus the cost of insert and extract into vector
|
|
|
|
// elements, times the vector width.
|
|
|
|
unsigned Cost = 0;
|
2012-10-24 20:36:32 +00:00
|
|
|
|
2012-12-23 13:21:41 +00:00
|
|
|
if (!RetTy->isVoidTy() && VF != 1) {
|
2013-01-07 11:12:29 +00:00
|
|
|
unsigned InsCost = TTI.getVectorInstrCost(Instruction::InsertElement,
|
|
|
|
VectorTy);
|
|
|
|
unsigned ExtCost = TTI.getVectorInstrCost(Instruction::ExtractElement,
|
|
|
|
VectorTy);
|
2012-12-23 13:19:18 +00:00
|
|
|
|
|
|
|
// The cost of inserting the results plus extracting each one of the
|
|
|
|
// operands.
|
|
|
|
Cost += VF * (InsCost + ExtCost * I->getNumOperands());
|
|
|
|
}
|
2012-12-10 21:39:02 +00:00
|
|
|
|
2012-12-23 07:23:55 +00:00
|
|
|
// The cost of executing VF copies of the scalar instruction. This opcode
|
|
|
|
// is unknown. Assume that it is the same as 'mul'.
|
2013-01-07 11:12:29 +00:00
|
|
|
Cost += VF * TTI.getArithmeticInstrCost(Instruction::Mul, VectorTy);
|
2012-12-10 21:39:02 +00:00
|
|
|
return Cost;
|
|
|
|
}
|
2012-10-24 20:36:32 +00:00
|
|
|
}// end of switch.
|
|
|
|
}
|
|
|
|
|
2012-10-25 21:03:48 +00:00
|
|
|
Type* LoopVectorizationCostModel::ToVectorTy(Type *Scalar, unsigned VF) {
|
|
|
|
if (Scalar->isVoidTy() || VF == 1)
|
|
|
|
return Scalar;
|
|
|
|
return VectorType::get(Scalar, VF);
|
|
|
|
}
|
2012-10-24 20:36:32 +00:00
|
|
|
|
2012-10-17 18:25:06 +00:00
|
|
|
char LoopVectorize::ID = 0;
|
|
|
|
static const char lv_name[] = "Loop Vectorization";
|
|
|
|
INITIALIZE_PASS_BEGIN(LoopVectorize, LV_NAME, lv_name, false, false)
|
2013-01-07 11:12:29 +00:00
|
|
|
INITIALIZE_AG_DEPENDENCY(TargetTransformInfo)
|
2012-10-17 18:25:06 +00:00
|
|
|
INITIALIZE_PASS_DEPENDENCY(ScalarEvolution)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(LoopSimplify)
|
|
|
|
INITIALIZE_PASS_END(LoopVectorize, LV_NAME, lv_name, false, false)
|
|
|
|
|
|
|
|
namespace llvm {
|
2013-08-28 18:33:10 +00:00
|
|
|
Pass *createLoopVectorizePass(bool NoUnrolling) {
|
|
|
|
return new LoopVectorize(NoUnrolling);
|
2012-10-17 18:25:06 +00:00
|
|
|
}
|
|
|
|
}
|
2012-12-10 21:39:02 +00:00
|
|
|
|
2013-02-05 15:08:02 +00:00
|
|
|
bool LoopVectorizationCostModel::isConsecutiveLoadOrStore(Instruction *Inst) {
|
|
|
|
// Check for a store.
|
2013-02-13 21:12:29 +00:00
|
|
|
if (StoreInst *ST = dyn_cast<StoreInst>(Inst))
|
2013-02-05 15:08:02 +00:00
|
|
|
return Legal->isConsecutivePtr(ST->getPointerOperand()) != 0;
|
2012-12-10 21:39:02 +00:00
|
|
|
|
2013-02-05 15:08:02 +00:00
|
|
|
// Check for a load.
|
2013-02-13 21:12:29 +00:00
|
|
|
if (LoadInst *LI = dyn_cast<LoadInst>(Inst))
|
2013-02-05 15:08:02 +00:00
|
|
|
return Legal->isConsecutivePtr(LI->getPointerOperand()) != 0;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2013-08-26 22:33:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
void InnerLoopUnroller::scalarizeInstruction(Instruction *Instr) {
|
|
|
|
assert(!Instr->getType()->isAggregateType() && "Can't handle vectors");
|
|
|
|
// Holds vector parameters or scalars, in case of uniform vals.
|
|
|
|
SmallVector<VectorParts, 4> Params;
|
|
|
|
|
|
|
|
setDebugLocFromInst(Builder, Instr);
|
|
|
|
|
|
|
|
// Find all of the vectorized parameters.
|
|
|
|
for (unsigned op = 0, e = Instr->getNumOperands(); op != e; ++op) {
|
|
|
|
Value *SrcOp = Instr->getOperand(op);
|
|
|
|
|
|
|
|
// If we are accessing the old induction variable, use the new one.
|
|
|
|
if (SrcOp == OldInduction) {
|
|
|
|
Params.push_back(getVectorValue(SrcOp));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try using previously calculated values.
|
|
|
|
Instruction *SrcInst = dyn_cast<Instruction>(SrcOp);
|
|
|
|
|
|
|
|
// If the src is an instruction that appeared earlier in the basic block
|
|
|
|
// then it should already be vectorized.
|
|
|
|
if (SrcInst && OrigLoop->contains(SrcInst)) {
|
|
|
|
assert(WidenMap.has(SrcInst) && "Source operand is unavailable");
|
|
|
|
// The parameter is a vector value from earlier.
|
|
|
|
Params.push_back(WidenMap.get(SrcInst));
|
|
|
|
} else {
|
|
|
|
// The parameter is a scalar from outside the loop. Maybe even a constant.
|
|
|
|
VectorParts Scalars;
|
|
|
|
Scalars.append(UF, SrcOp);
|
|
|
|
Params.push_back(Scalars);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(Params.size() == Instr->getNumOperands() &&
|
|
|
|
"Invalid number of operands");
|
|
|
|
|
|
|
|
// Does this instruction return a value ?
|
|
|
|
bool IsVoidRetTy = Instr->getType()->isVoidTy();
|
|
|
|
|
|
|
|
Value *UndefVec = IsVoidRetTy ? 0 :
|
|
|
|
UndefValue::get(Instr->getType());
|
|
|
|
// Create a new entry in the WidenMap and initialize it to Undef or Null.
|
|
|
|
VectorParts &VecResults = WidenMap.splat(Instr, UndefVec);
|
|
|
|
|
|
|
|
// For each vector unroll 'part':
|
|
|
|
for (unsigned Part = 0; Part < UF; ++Part) {
|
|
|
|
// For each scalar that we create:
|
|
|
|
|
|
|
|
Instruction *Cloned = Instr->clone();
|
|
|
|
if (!IsVoidRetTy)
|
|
|
|
Cloned->setName(Instr->getName() + ".cloned");
|
|
|
|
// Replace the operands of the cloned instrucions with extracted scalars.
|
|
|
|
for (unsigned op = 0, e = Instr->getNumOperands(); op != e; ++op) {
|
|
|
|
Value *Op = Params[op][Part];
|
|
|
|
Cloned->setOperand(op, Op);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Place the cloned scalar in the new loop.
|
|
|
|
Builder.Insert(Cloned);
|
|
|
|
|
|
|
|
// If the original scalar returns a value we need to place it in a vector
|
|
|
|
// so that future users will be able to use it.
|
|
|
|
if (!IsVoidRetTy)
|
|
|
|
VecResults[Part] = Cloned;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
InnerLoopUnroller::vectorizeMemoryInstruction(Instruction *Instr,
|
|
|
|
LoopVectorizationLegality*) {
|
|
|
|
return scalarizeInstruction(Instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
Value *InnerLoopUnroller::reverseVector(Value *Vec) {
|
|
|
|
return Vec;
|
|
|
|
}
|
|
|
|
|
|
|
|
Value *InnerLoopUnroller::getBroadcastInstrs(Value *V) {
|
|
|
|
return V;
|
|
|
|
}
|
|
|
|
|
|
|
|
Value *InnerLoopUnroller::getConsecutiveVector(Value* Val, int StartIdx,
|
|
|
|
bool Negate) {
|
|
|
|
// When unrolling and the VF is 1, we only need to add a simple scalar.
|
|
|
|
Type *ITy = Val->getType();
|
|
|
|
assert(!ITy->isVectorTy() && "Val must be a scalar");
|
|
|
|
Constant *C = ConstantInt::get(ITy, StartIdx, Negate);
|
|
|
|
return Builder.CreateAdd(Val, C, "induction");
|
|
|
|
}
|
|
|
|
|