2005-04-22 06:32:48 +00:00
|
|
|
//===-- DependenceAnalyzer.cpp - DependenceAnalyzer ------------*- C++ -*-===//
|
2005-03-29 20:33:42 +00:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file was developed by the LLVM research group and is distributed under
|
|
|
|
// the University of Illinois Open Source License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
2005-04-21 23:30:14 +00:00
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
2005-03-29 20:33:42 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#define DEBUG_TYPE "ModuloSched"
|
|
|
|
|
|
|
|
#include "DependenceAnalyzer.h"
|
2005-04-01 15:41:30 +00:00
|
|
|
#include "llvm/Type.h"
|
2005-03-29 20:33:42 +00:00
|
|
|
#include "llvm/Support/Debug.h"
|
2005-04-22 06:32:48 +00:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
|
|
#include "llvm/Constants.h"
|
2005-03-29 20:33:42 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
using namespace llvm;
|
2005-03-29 20:33:42 +00:00
|
|
|
|
2005-04-06 19:45:39 +00:00
|
|
|
namespace llvm {
|
2005-04-22 06:32:48 +00:00
|
|
|
|
|
|
|
/// Create ModuloSchedulingPass
|
|
|
|
FunctionPass *createDependenceAnalyzer() {
|
2005-07-27 06:12:32 +00:00
|
|
|
return new DependenceAnalyzer();
|
2005-04-22 06:32:48 +00:00
|
|
|
}
|
2005-04-06 19:45:39 +00:00
|
|
|
}
|
2005-04-22 06:32:48 +00:00
|
|
|
|
|
|
|
Statistic<> NoDeps("depanalyzer-nodeps", "Number of dependences eliminated");
|
2005-07-27 06:12:32 +00:00
|
|
|
Statistic<> NumDeps("depanalyzer-deps",
|
2005-07-27 05:53:44 +00:00
|
|
|
"Number of dependences could not eliminate");
|
2005-07-27 06:12:32 +00:00
|
|
|
Statistic<> AdvDeps("depanalyzer-advdeps",
|
2005-07-27 05:53:44 +00:00
|
|
|
"Number of dependences using advanced techniques");
|
2005-04-22 06:32:48 +00:00
|
|
|
|
|
|
|
bool DependenceAnalyzer::runOnFunction(Function &F) {
|
|
|
|
AA = &getAnalysis<AliasAnalysis>();
|
|
|
|
TD = &getAnalysis<TargetData>();
|
|
|
|
SE = &getAnalysis<ScalarEvolution>();
|
|
|
|
|
|
|
|
return false;
|
2005-03-29 20:33:42 +00:00
|
|
|
}
|
|
|
|
|
2005-07-27 06:12:32 +00:00
|
|
|
static RegisterAnalysis<DependenceAnalyzer>X("depanalyzer",
|
2005-07-27 05:53:44 +00:00
|
|
|
"Dependence Analyzer");
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
// - Get inter and intra dependences between loads and stores
|
|
|
|
//
|
2005-07-27 06:12:32 +00:00
|
|
|
// Overview of Method:
|
|
|
|
// Step 1: Use alias analysis to determine dependencies if values are loop
|
|
|
|
// invariant
|
|
|
|
// Step 2: If pointers are not GEP, then there is a dependence.
|
|
|
|
// Step 3: Compare GEP base pointers with AA. If no alias, no dependence.
|
|
|
|
// If may alias, then add a dependence. If must alias, then analyze
|
|
|
|
// further (Step 4)
|
2005-04-22 06:32:48 +00:00
|
|
|
// Step 4: do advanced analysis
|
2005-07-27 06:12:32 +00:00
|
|
|
void DependenceAnalyzer::AnalyzeDeps(Value *val, Value *val2, bool valLoad,
|
|
|
|
bool val2Load,
|
|
|
|
std::vector<Dependence> &deps,
|
|
|
|
BasicBlock *BB,
|
2005-07-27 05:53:44 +00:00
|
|
|
bool srcBeforeDest) {
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
bool loopInvariant = true;
|
2005-04-21 23:30:14 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//Check if both are instructions and prove not loop invariant if possible
|
|
|
|
if(Instruction *valInst = dyn_cast<Instruction>(val))
|
|
|
|
if(valInst->getParent() == BB)
|
|
|
|
loopInvariant = false;
|
|
|
|
if(Instruction *val2Inst = dyn_cast<Instruction>(val2))
|
|
|
|
if(val2Inst->getParent() == BB)
|
|
|
|
loopInvariant = false;
|
2005-07-27 06:12:32 +00:00
|
|
|
|
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//If Loop invariant, let AA decide
|
|
|
|
if(loopInvariant) {
|
|
|
|
if(AA->alias(val, (unsigned)TD->getTypeSize(val->getType()),
|
2005-07-27 05:53:44 +00:00
|
|
|
val2,(unsigned)TD->getTypeSize(val2->getType()))
|
2005-04-22 06:32:48 +00:00
|
|
|
!= AliasAnalysis::NoAlias) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++NoDeps;
|
|
|
|
return;
|
2005-03-29 20:33:42 +00:00
|
|
|
}
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//Otherwise, continue with step 2
|
2005-03-29 20:33:42 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
GetElementPtrInst *GP = dyn_cast<GetElementPtrInst>(val);
|
|
|
|
GetElementPtrInst *GP2 = dyn_cast<GetElementPtrInst>(val2);
|
2005-04-21 23:30:14 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//If both are not GP instructions, we can not do further analysis
|
|
|
|
if(!GP || !GP2) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
|
|
|
}
|
2005-03-29 20:33:42 +00:00
|
|
|
|
2005-04-21 23:30:14 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//Otherwise, compare GEP bases (op #0) with Alias Analysis
|
2005-03-29 20:33:42 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
Value *GPop = GP->getOperand(0);
|
|
|
|
Value *GP2op = GP2->getOperand(0);
|
|
|
|
int alias = AA->alias(GPop, (unsigned)TD->getTypeSize(GPop->getType()),
|
2005-07-27 05:53:44 +00:00
|
|
|
GP2op,(unsigned)TD->getTypeSize(GP2op->getType()));
|
2005-03-29 20:33:42 +00:00
|
|
|
|
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
if(alias == AliasAnalysis::MustAlias) {
|
|
|
|
//Further dep analysis to do
|
|
|
|
advancedDepAnalysis(GP, GP2, valLoad, val2Load, deps, srcBeforeDest);
|
|
|
|
++AdvDeps;
|
|
|
|
}
|
|
|
|
else if(alias == AliasAnalysis::MayAlias) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
}
|
|
|
|
//Otherwise no dependence since there is no alias
|
|
|
|
else
|
|
|
|
++NoDeps;
|
|
|
|
}
|
|
|
|
|
2005-03-29 20:33:42 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
// advancedDepAnalysis - Do advanced data dependence tests
|
2005-07-27 06:12:32 +00:00
|
|
|
void DependenceAnalyzer::advancedDepAnalysis(GetElementPtrInst *gp1,
|
2005-07-27 05:53:44 +00:00
|
|
|
GetElementPtrInst *gp2,
|
|
|
|
bool valLoad,
|
|
|
|
bool val2Load,
|
|
|
|
std::vector<Dependence> &deps,
|
|
|
|
bool srcBeforeDest) {
|
2005-04-21 23:30:14 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//Check if both GEPs are in a simple form: 3 ops, constant 0 as second arg
|
|
|
|
if(gp1->getNumOperands() != 3 || gp2->getNumOperands() != 3) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//Check second arg is constant 0
|
|
|
|
bool GPok = false;
|
|
|
|
if(Constant *c1 = dyn_cast<Constant>(gp1->getOperand(1)))
|
|
|
|
if(Constant *c2 = dyn_cast<Constant>(gp2->getOperand(1)))
|
|
|
|
if(c1->isNullValue() && c2->isNullValue())
|
2005-07-27 05:53:44 +00:00
|
|
|
GPok = true;
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
if(!GPok) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Value *Gep1Idx = gp1->getOperand(2);
|
|
|
|
Value *Gep2Idx = gp2->getOperand(2);
|
|
|
|
|
|
|
|
if(CastInst *c1 = dyn_cast<CastInst>(Gep1Idx))
|
|
|
|
Gep1Idx = c1->getOperand(0);
|
|
|
|
if(CastInst *c2 = dyn_cast<CastInst>(Gep2Idx))
|
|
|
|
Gep2Idx = c2->getOperand(0);
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//Get SCEV for each index into the area
|
|
|
|
SCEVHandle SV1 = SE->getSCEV(Gep1Idx);
|
|
|
|
SCEVHandle SV2 = SE->getSCEV(Gep2Idx);
|
|
|
|
|
|
|
|
//Now handle special cases of dependence analysis
|
2005-06-17 04:00:57 +00:00
|
|
|
//SV1->print(std::cerr);
|
|
|
|
//std::cerr << "\n";
|
|
|
|
//SV2->print(std::cerr);
|
|
|
|
//std::cerr << "\n";
|
2005-04-22 06:32:48 +00:00
|
|
|
|
|
|
|
//Check if we have an SCEVAddExpr, cause we can only handle those
|
|
|
|
SCEVAddRecExpr *SVAdd1 = dyn_cast<SCEVAddRecExpr>(SV1);
|
|
|
|
SCEVAddRecExpr *SVAdd2 = dyn_cast<SCEVAddRecExpr>(SV2);
|
|
|
|
|
|
|
|
//Default to having a dependence since we can't analyze further
|
|
|
|
if(!SVAdd1 || !SVAdd2) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
|
|
|
}
|
2005-03-29 20:33:42 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//Check if not Affine, we can't handle those
|
|
|
|
if(!SVAdd1->isAffine( ) || !SVAdd2->isAffine()) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
2005-03-29 20:33:42 +00:00
|
|
|
}
|
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//We know the SCEV is in the form A + B*x, check that B is the same for both
|
|
|
|
SCEVConstant *B1 = dyn_cast<SCEVConstant>(SVAdd1->getOperand(1));
|
|
|
|
SCEVConstant *B2 = dyn_cast<SCEVConstant>(SVAdd2->getOperand(1));
|
|
|
|
|
|
|
|
if(B1->getValue() != B2->getValue()) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
|
|
|
}
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
if(B1->getValue()->getRawValue() != 1 || B2->getValue()->getRawValue() != 1) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SCEVConstant *A1 = dyn_cast<SCEVConstant>(SVAdd1->getOperand(0));
|
|
|
|
SCEVConstant *A2 = dyn_cast<SCEVConstant>(SVAdd2->getOperand(0));
|
|
|
|
|
|
|
|
//Come back and deal with nested SCEV!
|
|
|
|
if(!A1 || !A2) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//If equal, create dep as normal
|
|
|
|
if(A1->getValue() == A2->getValue()) {
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
//Eliminate a dep if this is a intra dep
|
|
|
|
else if(srcBeforeDest) {
|
|
|
|
++NoDeps;
|
|
|
|
return;
|
|
|
|
}
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//Find constant index difference
|
|
|
|
int diff = A1->getValue()->getRawValue() - A2->getValue()->getRawValue();
|
2005-06-17 04:00:57 +00:00
|
|
|
//std::cerr << diff << "\n";
|
2005-04-30 23:07:59 +00:00
|
|
|
if(diff > 5)
|
|
|
|
diff = 2;
|
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
if(diff > 0)
|
|
|
|
createDep(deps, valLoad, val2Load, srcBeforeDest, diff);
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//assert(diff > 0 && "Expected diff to be greater then 0");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create dependences once its determined these two instructions
|
|
|
|
// references the same memory
|
2005-07-27 06:12:32 +00:00
|
|
|
void DependenceAnalyzer::createDep(std::vector<Dependence> &deps,
|
|
|
|
bool valLoad, bool val2Load,
|
2005-07-27 05:53:44 +00:00
|
|
|
bool srcBeforeDest, int diff) {
|
2005-04-22 06:32:48 +00:00
|
|
|
|
|
|
|
//If the source instruction occurs after the destination instruction
|
|
|
|
//(execution order), then this dependence is across iterations
|
|
|
|
if(!srcBeforeDest && (diff==0))
|
|
|
|
diff = 1;
|
|
|
|
|
|
|
|
//If load/store pair
|
|
|
|
if(valLoad && !val2Load) {
|
2005-07-27 06:12:32 +00:00
|
|
|
if(srcBeforeDest)
|
2005-06-17 04:00:57 +00:00
|
|
|
//Anti Dep
|
|
|
|
deps.push_back(Dependence(diff, Dependence::AntiDep));
|
|
|
|
else
|
|
|
|
deps.push_back(Dependence(diff, Dependence::TrueDep));
|
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
++NumDeps;
|
|
|
|
}
|
|
|
|
//If store/load pair
|
|
|
|
else if(!valLoad && val2Load) {
|
2005-07-27 06:12:32 +00:00
|
|
|
if(srcBeforeDest)
|
2005-06-17 04:00:57 +00:00
|
|
|
//True Dep
|
|
|
|
deps.push_back(Dependence(diff, Dependence::TrueDep));
|
|
|
|
else
|
|
|
|
deps.push_back(Dependence(diff, Dependence::AntiDep));
|
2005-04-22 06:32:48 +00:00
|
|
|
++NumDeps;
|
|
|
|
}
|
|
|
|
//If store/store pair
|
|
|
|
else if(!valLoad && !val2Load) {
|
|
|
|
//True Dep
|
|
|
|
deps.push_back(Dependence(diff, Dependence::OutputDep));
|
|
|
|
++NumDeps;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
//Get Dependence Info for a pair of Instructions
|
2005-07-27 06:12:32 +00:00
|
|
|
DependenceResult DependenceAnalyzer::getDependenceInfo(Instruction *inst1,
|
|
|
|
Instruction *inst2,
|
2005-07-27 05:53:44 +00:00
|
|
|
bool srcBeforeDest) {
|
2005-04-22 06:32:48 +00:00
|
|
|
std::vector<Dependence> deps;
|
|
|
|
|
|
|
|
DEBUG(std::cerr << "Inst1: " << *inst1 << "\n");
|
|
|
|
DEBUG(std::cerr << "Inst2: " << *inst2 << "\n");
|
|
|
|
|
|
|
|
//No self deps
|
|
|
|
if(inst1 == inst2)
|
|
|
|
return DependenceResult(deps);
|
|
|
|
|
|
|
|
if(LoadInst *ldInst = dyn_cast<LoadInst>(inst1)) {
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
if(StoreInst *stInst = dyn_cast<StoreInst>(inst2))
|
2005-07-27 06:12:32 +00:00
|
|
|
AnalyzeDeps(ldInst->getOperand(0), stInst->getOperand(1),
|
2005-07-27 05:53:44 +00:00
|
|
|
true, false, deps, ldInst->getParent(), srcBeforeDest);
|
2005-04-22 06:32:48 +00:00
|
|
|
}
|
|
|
|
else if(StoreInst *stInst = dyn_cast<StoreInst>(inst1)) {
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
if(LoadInst *ldInst = dyn_cast<LoadInst>(inst2))
|
2005-07-27 06:12:32 +00:00
|
|
|
AnalyzeDeps(stInst->getOperand(1), ldInst->getOperand(0), false, true,
|
2005-07-27 05:53:44 +00:00
|
|
|
deps, ldInst->getParent(), srcBeforeDest);
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
else if(StoreInst *stInst2 = dyn_cast<StoreInst>(inst2))
|
2005-07-27 06:12:32 +00:00
|
|
|
AnalyzeDeps(stInst->getOperand(1), stInst2->getOperand(1), false, false,
|
2005-07-27 05:53:44 +00:00
|
|
|
deps, stInst->getParent(), srcBeforeDest);
|
2005-04-22 06:32:48 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
assert(0 && "Expected a load or a store\n");
|
2005-07-27 06:12:32 +00:00
|
|
|
|
2005-04-22 06:32:48 +00:00
|
|
|
DependenceResult dr = DependenceResult(deps);
|
|
|
|
return dr;
|
|
|
|
}
|
|
|
|
|