Index: include/polly/ScopBuilder.h =================================================================== --- include/polly/ScopBuilder.h +++ include/polly/ScopBuilder.h @@ -59,26 +59,36 @@ /// Try to pattern match and find the array descriptor structure in case of a /// fortran array accesss. succeeds on load/store into a fortran array that - /// has been allocated. + /// is a global and has been allocated. /// - /// @see polly::FortranArrayDescriptor + /// @see polly::MemoryAccess, polly::ScopArrayInfo /// /// @param Inst The load/store instruction that access the memory. /// /// @note assumes -polly-canonicalize has been run. - GlobalValue *findFortranArrayDescriptorForAllocArrayAccess(MemAccInst Inst); + Value *findFADGlobalAlloc(MemAccInst Inst); /// Try to pattern match and find the array descriptor structure in case of a /// fortran array accesss. succeeds on load/store into a fortran array that - /// has been allocated. + /// is a global has been allocated. /// - /// @see polly::FortranArrayDescriptor + /// @see polly::MemoryAccess, polly::ScopArrayInfo /// /// @param Inst The load/store instruction that access the memory. /// /// @note assumes -polly-canonicalize has been run. - GlobalValue * - findFortranArrayDescriptorForNonAllocArrayAccess(MemAccInst Inst); + Value *findFADGlobalNonAlloc(MemAccInst Inst); + + /// Try to pattern match and find the array descriptor structure in case of a + /// fortran array accesss. succeeds on load/store into a fortran array that + /// is a local and has been allocation + /// + /// @see polly::MemoryAccess, polly::ScopArrayInfo + /// + /// @param Inst The load/store instruction that access the memory. + /// + /// @note assumes -polly-canonicalize has been run. + Value *findFADLocalNonAlloc(MemAccInst Inst); // @} // Build the SCoP for Region @p R. Index: include/polly/ScopInfo.h =================================================================== --- include/polly/ScopInfo.h +++ include/polly/ScopInfo.h @@ -620,7 +620,7 @@ /// along with auxiliary fields with information such as dimensions. /// We hold a reference to the descriptor corresponding to a MemoryAccess /// into a Fortran array. FAD for "Fortran Array Descriptor" - AssertingVH FAD; + AssertingVH FAD; // @} __isl_give isl_basic_map *createBasicAccessMap(ScopStmt *Statement); @@ -1019,7 +1019,7 @@ /// Set the array descriptor corresponding to the Array on which the /// memory access is performed. - void setFortranArrayDescriptor(GlobalValue *FAD); + void setFortranArrayDescriptor(Value *FAD); /// Update the original access relation. /// Index: lib/Analysis/ScopBuilder.cpp =================================================================== --- lib/Analysis/ScopBuilder.cpp +++ lib/Analysis/ScopBuilder.cpp @@ -143,8 +143,15 @@ /// generates for Fortran arrays. /// /// @param Global the global variable believed to be a Fortran array -bool isGlobalFortranArray(GlobalValue *Global) { - auto StructArrTy = dyn_cast(Global->getValueType()); +bool isValueFortranArray(Value *V) { + PointerType *PTy = dyn_cast(V->getType()); + + if (!PTy) + return false; + + Type *Ty = PTy->getElementType(); + assert(Ty != nullptr && "Ty expected to be initialized"); + auto StructArrTy = dyn_cast(Ty); if (!(StructArrTy && StructArrTy->hasName())) return false; @@ -158,7 +165,7 @@ const ArrayRef ArrMemberTys = StructArrTy->elements(); // i8* match - if (ArrMemberTys[0] != Type::getInt8PtrTy(Global->getContext())) + if (ArrMemberTys[0] != Type::getInt8PtrTy(V->getContext())) return false; // Get a reference to the int type and check that all the members @@ -216,8 +223,7 @@ /// /// 4.1 store/load , * %typedmem, align 8 /// 4.2 store/load , * %slot, align 8 -GlobalValue * -ScopBuilder::findFortranArrayDescriptorForAllocArrayAccess(MemAccInst Inst) { +Value *ScopBuilder::findFADGlobalNonAlloc(MemAccInst Inst) { // match: 4.1 & 4.2 store/load if (!isa(Inst) && !isa(Inst)) return nullptr; @@ -274,13 +280,12 @@ if (!(DescriptorType && DescriptorType->hasName())) continue; - GlobalValue *Descriptor = - dyn_cast(DescriptorGEP->getPointerOperand()); + Value *Descriptor = dyn_cast(DescriptorGEP->getPointerOperand()); if (!Descriptor) continue; - if (!isGlobalFortranArray(Descriptor)) + if (!isValueFortranArray(Descriptor)) continue; return Descriptor; @@ -307,8 +312,7 @@ /// /// 3.1 store/load , * %slot, align 8 /// 3.2 store/load , * %mem, align 8 -GlobalValue * -ScopBuilder::findFortranArrayDescriptorForNonAllocArrayAccess(MemAccInst Inst) { +Value *ScopBuilder::findFADGlobalAlloc(MemAccInst Inst) { // match: 3 if (!isa(Inst) && !isa(Inst)) return nullptr; @@ -337,12 +341,62 @@ if (!BitcastOperator) return nullptr; - GlobalValue *Descriptor = - dyn_cast(BitcastOperator->getOperand(0)); + Value *Descriptor = dyn_cast(BitcastOperator->getOperand(0)); + if (!Descriptor) + return nullptr; + + if (!isValueFortranArray(Descriptor)) + return nullptr; + + return Descriptor; +} + +/// This is matching against code generated by dragonegg after simplifier +/// passes have been run. +/// +/// This is trying to match against %param, the descriptor +/// of the Fortran array that is being accessed at load/store. This style +/// of code is generated for arrays that are local, generated as parameters +/// to functions. +/// +/// Pattern Match: +/// 1. %mem = bitcast %"struct.array1_integer(kind=4)"* %param to i32** +/// +/// 2. [%slot = getelementptr inbounds i8, i8* %mem, i64 ] +/// 2 is optional because if you are writing to the 0th index, you don't +/// need a GEP. +/// +/// 3.1 store/load , * %slot, align 8 +/// 3.2 store/load , * %mem, align 8 +Value *ScopBuilder::findFADLocalNonAlloc(MemAccInst Inst) { + // match: 3 + if (!isa(Inst) && !isa(Inst)) + return nullptr; + + // match: 3 + if (Inst.getAlignment() != 8) + return nullptr; + + Value *Slot = Inst.getPointerOperand(); + + BitCastOperator *MemBitcast = nullptr; + // [match: 2] + if (auto *SlotGEP = dyn_cast(Slot)) { + // match: 1 + MemBitcast = dyn_cast(SlotGEP->getPointerOperand()); + } else { + // match: 1 + MemBitcast = dyn_cast(Slot); + } + + if (!MemBitcast) + return nullptr; + + Value *Descriptor = dyn_cast(MemBitcast->getOperand(0)); if (!Descriptor) return nullptr; - if (!isGlobalFortranArray(Descriptor)) + if (!isValueFortranArray(Descriptor)) return nullptr; return Descriptor; @@ -774,11 +828,11 @@ if (!DetectFortranArrays) return; - if (GlobalValue *FAD = - findFortranArrayDescriptorForAllocArrayAccess(MemAccInst)) + if (Value *FAD = findFADGlobalNonAlloc(MemAccInst)) + MemAccess->setFortranArrayDescriptor(FAD); + else if (Value *FAD = findFADGlobalAlloc(MemAccInst)) MemAccess->setFortranArrayDescriptor(FAD); - else if (GlobalValue *FAD = - findFortranArrayDescriptorForNonAllocArrayAccess(MemAccInst)) + else if (Value *FAD = findFADLocalNonAlloc(MemAccInst)) MemAccess->setFortranArrayDescriptor(FAD); } Index: lib/Analysis/ScopInfo.cpp =================================================================== --- lib/Analysis/ScopInfo.cpp +++ lib/Analysis/ScopInfo.cpp @@ -1034,21 +1034,7 @@ return OS; } -void MemoryAccess::setFortranArrayDescriptor(GlobalValue *FAD) { - this->FAD = FAD; - -// TODO: write checks to make sure it looks _exactly_ like a Fortran array -// descriptor -#ifndef NDEBUG - StructType *ty = dyn_cast(FAD->getValueType()); - assert(ty && "expected value of type Fortran array descriptor"); - assert(ty->hasName() && ty->getName().startswith("struct.array") && - "expected global to follow Fortran array descriptor type naming " - "convention"); - assert(ty->getNumElements() == 4 && - "expected layout to be like Fortran array descriptor type"); -#endif -} +void MemoryAccess::setFortranArrayDescriptor(Value *FAD) { this->FAD = FAD; } void MemoryAccess::print(raw_ostream &OS) const { switch (AccType) { Index: test/ScopDetect/fortran_array_param_nonmalloc_nonvectored.ll =================================================================== --- /dev/null +++ test/ScopDetect/fortran_array_param_nonmalloc_nonvectored.ll @@ -0,0 +1,75 @@ +; RUN: opt -S -polly-detect-fortran-arrays -analyze -polly-process-unprofitable \ +; RUN: -polly-remarks-minimal -polly-canonicalize -polly-scops \ +; RUN: -polly-dependences -polly-canonicalize \ +; RUN: -polly-allow-nonaffine -polly-ignore-aliasing \ +; RUN: -polly-invariant-load-hoisting < %s| FileCheck %s + +; ModuleID = 'simple-nonaffine-read.bc' +; ModuleID = 'simple-nonaffine-read.bc' + +; This testcase is the corresponding LLVM for func: +; PROGRAM main +; INTEGER, DIMENSION(1) :: xs + +; CALL testfunc(xs, 10) +; CONTAINS +; SUBROUTINE func(xs, n) +; IMPLICIT NONE +; INTEGER, DIMENSION(:), INTENT(INOUT) :: xs +; INTEGER, INTENT(IN) :: n +; INTEGER :: i + +; DO i = 1, n +; xs(i) = 1 +; END DO + +; END SUBROUTINE func +; END PROGRAM + +target datalayout = "e-p:64:64:64-S128-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f16:16:16-f32:32:32-f64:64:64-f128:128:128-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64" +target triple = "x86_64-unknown-linux-gnu" + +module asm "\09.ident\09\22GCC: (GNU) 4.6.4 LLVM: 3.3.1\22" + +%"struct.array1_integer(kind=4)" = type { i8*, i64, i64, [1 x %struct.descriptor_dimension] } +%struct.descriptor_dimension = type { i64, i64, i64 } +%"struct.array1_integer(kind=4).0" = type { i8*, i64, i64, [1 x %struct.descriptor_dimension] } + +; Function Attrs: nounwind uwtable +define internal void @testfunc(%"struct.array1_integer(kind=4)"* noalias %xs, i32* noalias %n) #0 { +entry: + br label %entry.split + +entry.split: ; preds = %entry + %0 = getelementptr inbounds %"struct.array1_integer(kind=4)", %"struct.array1_integer(kind=4)"* %xs, i64 0, i32 3, i64 0, i32 0 + %1 = load i64, i64* %0, align 8 + %2 = icmp eq i64 %1, 0 + %3 = select i1 %2, i64 1, i64 %1 + %4 = bitcast %"struct.array1_integer(kind=4)"* %xs to i32** + %5 = load i32*, i32** %4, align 8 + %6 = load i32, i32* %n, align 4 + %7 = icmp sgt i32 %6, 0 + br i1 %7, label %"6.preheader", label %return + +"6.preheader": ; preds = %entry.split + br label %"6" + +"6": ; preds = %"6", %"6.preheader" + %8 = phi i32 [ %14, %"6" ], [ 1, %"6.preheader" ] + %9 = sext i32 %8 to i64 + %10 = mul i64 %3, %9 + %11 = sub i64 %10, %3 + %12 = getelementptr i32, i32* %5, i64 %11 + store i32 1, i32* %12, align 4 + %13 = icmp eq i32 %8, %6 + %14 = add i32 %8, 1 + br i1 %13, label %return.loopexit, label %"6" + +return.loopexit: ; preds = %"6" + br label %return + +return: ; preds = %return.loopexit, %entry.split + ret void +} + +; CHECK: ReadAccess := [Reduction Type: NONE] [Fortran array descriptor: xs] [Scalar: 0]