You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			257 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			257 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //===------ PollyIRBuilder.cpp --------------------------------------------===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // The Polly IRBuilder file contains Polly specific extensions for the IRBuilder
 | |
| // that are used e.g. to emit the llvm.loop.parallel metadata.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "polly/CodeGen/IRBuilder.h"
 | |
| #include "polly/ScopInfo.h"
 | |
| #include "polly/Support/ScopHelper.h"
 | |
| #include "llvm/ADT/SmallVector.h"
 | |
| #include "llvm/IR/Metadata.h"
 | |
| #include "llvm/Support/Debug.h"
 | |
| 
 | |
| using namespace llvm;
 | |
| using namespace polly;
 | |
| 
 | |
| static const int MaxArraysInAliasScops = 10;
 | |
| 
 | |
| /// Get a self referencing id metadata node.
 | |
| ///
 | |
| /// The MDNode looks like this (if arg0/arg1 are not null):
 | |
| ///
 | |
| ///    '!n = metadata !{metadata !n, arg0, arg1}'
 | |
| ///
 | |
| /// @return The self referencing id metadata node.
 | |
| static MDNode *getID(LLVMContext &Ctx, Metadata *arg0 = nullptr,
 | |
|                      Metadata *arg1 = nullptr) {
 | |
|   MDNode *ID;
 | |
|   SmallVector<Metadata *, 3> Args;
 | |
|   // Use a temporary node to safely create a unique pointer for the first arg.
 | |
|   auto TempNode = MDNode::getTemporary(Ctx, None);
 | |
|   // Reserve operand 0 for loop id self reference.
 | |
|   Args.push_back(TempNode.get());
 | |
| 
 | |
|   if (arg0)
 | |
|     Args.push_back(arg0);
 | |
|   if (arg1)
 | |
|     Args.push_back(arg1);
 | |
| 
 | |
|   ID = MDNode::get(Ctx, Args);
 | |
|   ID->replaceOperandWith(0, ID);
 | |
|   return ID;
 | |
| }
 | |
| 
 | |
| ScopAnnotator::ScopAnnotator() : SE(nullptr), AliasScopeDomain(nullptr) {}
 | |
| 
 | |
| void ScopAnnotator::buildAliasScopes(Scop &S) {
 | |
|   SE = S.getSE();
 | |
| 
 | |
|   LLVMContext &Ctx = SE->getContext();
 | |
|   AliasScopeDomain = getID(Ctx, MDString::get(Ctx, "polly.alias.scope.domain"));
 | |
| 
 | |
|   AliasScopeMap.clear();
 | |
|   OtherAliasScopeListMap.clear();
 | |
| 
 | |
|   // We are only interested in arrays, but no scalar references. Scalars should
 | |
|   // be handled easily by basicaa.
 | |
|   SmallVector<ScopArrayInfo *, 10> Arrays;
 | |
|   for (ScopArrayInfo *Array : S.arrays())
 | |
|     if (Array->isArrayKind())
 | |
|       Arrays.push_back(Array);
 | |
| 
 | |
|   // The construction of alias scopes is quadratic in the number of arrays
 | |
|   // involved. In case of too many arrays, skip the construction of alias
 | |
|   // information to avoid quadratic increases in compile time and code size.
 | |
|   if (Arrays.size() > MaxArraysInAliasScops)
 | |
|     return;
 | |
| 
 | |
|   std::string AliasScopeStr = "polly.alias.scope.";
 | |
|   for (const ScopArrayInfo *Array : Arrays) {
 | |
|     assert(Array->getBasePtr() && "Base pointer must be present");
 | |
|     AliasScopeMap[Array->getBasePtr()] =
 | |
|         getID(Ctx, AliasScopeDomain,
 | |
|               MDString::get(Ctx, (AliasScopeStr + Array->getName()).c_str()));
 | |
|   }
 | |
| 
 | |
|   for (const ScopArrayInfo *Array : Arrays) {
 | |
|     MDNode *AliasScopeList = MDNode::get(Ctx, {});
 | |
|     for (const auto &AliasScopePair : AliasScopeMap) {
 | |
|       if (Array->getBasePtr() == AliasScopePair.first)
 | |
|         continue;
 | |
| 
 | |
|       Metadata *Args = {AliasScopePair.second};
 | |
|       AliasScopeList =
 | |
|           MDNode::concatenate(AliasScopeList, MDNode::get(Ctx, Args));
 | |
|     }
 | |
| 
 | |
|     OtherAliasScopeListMap[Array->getBasePtr()] = AliasScopeList;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void ScopAnnotator::pushLoop(Loop *L, bool IsParallel) {
 | |
| 
 | |
|   ActiveLoops.push_back(L);
 | |
|   if (!IsParallel)
 | |
|     return;
 | |
| 
 | |
|   BasicBlock *Header = L->getHeader();
 | |
|   MDNode *Id = getID(Header->getContext());
 | |
|   assert(Id->getOperand(0) == Id && "Expected Id to be a self-reference");
 | |
|   assert(Id->getNumOperands() == 1 && "Unexpected extra operands in Id");
 | |
|   MDNode *Ids = ParallelLoops.empty()
 | |
|                     ? Id
 | |
|                     : MDNode::concatenate(ParallelLoops.back(), Id);
 | |
|   ParallelLoops.push_back(Ids);
 | |
| }
 | |
| 
 | |
| void ScopAnnotator::popLoop(bool IsParallel) {
 | |
|   ActiveLoops.pop_back();
 | |
|   if (!IsParallel)
 | |
|     return;
 | |
| 
 | |
|   assert(!ParallelLoops.empty() && "Expected a parallel loop to pop");
 | |
|   ParallelLoops.pop_back();
 | |
| }
 | |
| 
 | |
| void ScopAnnotator::annotateLoopLatch(BranchInst *B, Loop *L, bool IsParallel,
 | |
|                                       bool IsLoopVectorizerDisabled) const {
 | |
|   MDNode *MData = nullptr;
 | |
| 
 | |
|   if (IsLoopVectorizerDisabled) {
 | |
|     SmallVector<Metadata *, 3> Args;
 | |
|     LLVMContext &Ctx = SE->getContext();
 | |
|     Args.push_back(MDString::get(Ctx, "llvm.loop.vectorize.enable"));
 | |
|     auto *FalseValue = ConstantInt::get(Type::getInt1Ty(Ctx), 0);
 | |
|     Args.push_back(ValueAsMetadata::get(FalseValue));
 | |
|     MData = MDNode::concatenate(MData, getID(Ctx, MDNode::get(Ctx, Args)));
 | |
|   }
 | |
| 
 | |
|   if (IsParallel) {
 | |
|     assert(!ParallelLoops.empty() && "Expected a parallel loop to annotate");
 | |
|     MDNode *Ids = ParallelLoops.back();
 | |
|     MDNode *Id = cast<MDNode>(Ids->getOperand(Ids->getNumOperands() - 1));
 | |
|     MData = MDNode::concatenate(MData, Id);
 | |
|   }
 | |
| 
 | |
|   B->setMetadata("llvm.loop", MData);
 | |
| }
 | |
| 
 | |
| /// Get the pointer operand
 | |
| ///
 | |
| /// @param Inst The instruction to be analyzed.
 | |
| /// @return the pointer operand in case @p Inst is a memory access
 | |
| ///         instruction and nullptr otherwise.
 | |
| static llvm::Value *getMemAccInstPointerOperand(Instruction *Inst) {
 | |
|   auto MemInst = MemAccInst::dyn_cast(Inst);
 | |
|   if (!MemInst)
 | |
|     return nullptr;
 | |
| 
 | |
|   return MemInst.getPointerOperand();
 | |
| }
 | |
| 
 | |
| void ScopAnnotator::annotateSecondLevel(llvm::Instruction *Inst,
 | |
|                                         llvm::Value *BasePtr) {
 | |
|   auto *PtrSCEV = SE->getSCEV(getMemAccInstPointerOperand(Inst));
 | |
|   auto *BasePtrSCEV = SE->getPointerBase(PtrSCEV);
 | |
| 
 | |
|   if (!PtrSCEV)
 | |
|     return;
 | |
|   auto SecondLevelAliasScope = SecondLevelAliasScopeMap.lookup(PtrSCEV);
 | |
|   auto SecondLevelOtherAliasScopeList =
 | |
|       SecondLevelOtherAliasScopeListMap.lookup(PtrSCEV);
 | |
|   if (!SecondLevelAliasScope) {
 | |
|     auto AliasScope = AliasScopeMap.lookup(BasePtr);
 | |
|     if (!AliasScope)
 | |
|       return;
 | |
|     LLVMContext &Ctx = SE->getContext();
 | |
|     SecondLevelAliasScope = getID(
 | |
|         Ctx, AliasScope, MDString::get(Ctx, "second level alias metadata"));
 | |
|     SecondLevelAliasScopeMap[PtrSCEV] = SecondLevelAliasScope;
 | |
|     Metadata *Args = {SecondLevelAliasScope};
 | |
|     auto SecondLevelBasePtrAliasScopeList =
 | |
|         SecondLevelAliasScopeMap.lookup(BasePtrSCEV);
 | |
|     SecondLevelAliasScopeMap[BasePtrSCEV] = MDNode::concatenate(
 | |
|         SecondLevelBasePtrAliasScopeList, MDNode::get(Ctx, Args));
 | |
|     auto OtherAliasScopeList = OtherAliasScopeListMap.lookup(BasePtr);
 | |
|     SecondLevelOtherAliasScopeList = MDNode::concatenate(
 | |
|         OtherAliasScopeList, SecondLevelBasePtrAliasScopeList);
 | |
|     SecondLevelOtherAliasScopeListMap[PtrSCEV] = SecondLevelOtherAliasScopeList;
 | |
|   }
 | |
|   Inst->setMetadata("alias.scope", SecondLevelAliasScope);
 | |
|   Inst->setMetadata("noalias", SecondLevelOtherAliasScopeList);
 | |
| }
 | |
| 
 | |
| void ScopAnnotator::annotate(Instruction *Inst) {
 | |
|   if (!Inst->mayReadOrWriteMemory())
 | |
|     return;
 | |
| 
 | |
|   if (!ParallelLoops.empty())
 | |
|     Inst->setMetadata("llvm.mem.parallel_loop_access", ParallelLoops.back());
 | |
| 
 | |
|   // TODO: Use the ScopArrayInfo once available here.
 | |
|   if (!AliasScopeDomain)
 | |
|     return;
 | |
| 
 | |
|   // Do not apply annotations on memory operations that take more than one
 | |
|   // pointer. It would be ambiguous to which pointer the annotation applies.
 | |
|   // FIXME: How can we specify annotations for all pointer arguments?
 | |
|   if (isa<CallInst>(Inst) && !isa<MemSetInst>(Inst))
 | |
|     return;
 | |
| 
 | |
|   auto *Ptr = getMemAccInstPointerOperand(Inst);
 | |
|   if (!Ptr)
 | |
|     return;
 | |
| 
 | |
|   auto *PtrSCEV = SE->getSCEV(Ptr);
 | |
|   auto *BaseSCEV = SE->getPointerBase(PtrSCEV);
 | |
|   auto *SU = dyn_cast<SCEVUnknown>(BaseSCEV);
 | |
| 
 | |
|   if (!SU)
 | |
|     return;
 | |
| 
 | |
|   auto *BasePtr = SU->getValue();
 | |
| 
 | |
|   if (!BasePtr)
 | |
|     return;
 | |
| 
 | |
|   auto AliasScope = AliasScopeMap.lookup(BasePtr);
 | |
| 
 | |
|   if (!AliasScope) {
 | |
|     BasePtr = AlternativeAliasBases.lookup(BasePtr);
 | |
|     if (!BasePtr)
 | |
|       return;
 | |
| 
 | |
|     AliasScope = AliasScopeMap.lookup(BasePtr);
 | |
|     if (!AliasScope)
 | |
|       return;
 | |
|   }
 | |
| 
 | |
|   assert(OtherAliasScopeListMap.count(BasePtr) &&
 | |
|          "BasePtr either expected in AliasScopeMap and OtherAlias...Map");
 | |
|   auto *OtherAliasScopeList = OtherAliasScopeListMap[BasePtr];
 | |
| 
 | |
|   if (InterIterationAliasFreeBasePtrs.count(BasePtr)) {
 | |
|     annotateSecondLevel(Inst, BasePtr);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Inst->setMetadata("alias.scope", AliasScope);
 | |
|   Inst->setMetadata("noalias", OtherAliasScopeList);
 | |
| }
 | |
| 
 | |
| void ScopAnnotator::addInterIterationAliasFreeBasePtr(llvm::Value *BasePtr) {
 | |
|   if (!BasePtr)
 | |
|     return;
 | |
| 
 | |
|   InterIterationAliasFreeBasePtrs.insert(BasePtr);
 | |
| }
 |