//===-- Allocatable.cpp -- Allocatable statements lowering ----------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // // Coding style: https://mlir.llvm.org/getting_started/DeveloperGuide/ // //===----------------------------------------------------------------------===// #include "flang/Lower/Allocatable.h" #include "flang/Evaluate/tools.h" #include "flang/Lower/AbstractConverter.h" #include "flang/Lower/IterationSpace.h" #include "flang/Lower/PFTBuilder.h" #include "flang/Lower/Runtime.h" #include "flang/Lower/StatementContext.h" #include "flang/Optimizer/Builder/FIRBuilder.h" #include "flang/Optimizer/Builder/Runtime/RTBuilder.h" #include "flang/Optimizer/Builder/Todo.h" #include "flang/Optimizer/Dialect/FIROps.h" #include "flang/Optimizer/Dialect/FIROpsSupport.h" #include "flang/Optimizer/Support/FatalError.h" #include "flang/Parser/parse-tree.h" #include "flang/Runtime/allocatable.h" #include "flang/Runtime/pointer.h" #include "flang/Semantics/tools.h" #include "flang/Semantics/type.h" #include "llvm/Support/CommandLine.h" /// By default fir memory operation fir::AllocMemOp/fir::FreeMemOp are used. /// This switch allow forcing the use of runtime and descriptors for everything. /// This is mainly intended as a debug switch. static llvm::cl::opt useAllocateRuntime( "use-alloc-runtime", llvm::cl::desc("Lower allocations to fortran runtime calls"), llvm::cl::init(false)); /// Switch to force lowering of allocatable and pointers to descriptors in all /// cases for debug purposes. static llvm::cl::opt useDescForMutableBox( "use-desc-for-alloc", llvm::cl::desc("Always use descriptors for POINTER and ALLOCATABLE"), llvm::cl::init(false)); //===----------------------------------------------------------------------===// // Error management //===----------------------------------------------------------------------===// namespace { // Manage STAT and ERRMSG specifier information across a sequence of runtime // calls for an ALLOCATE/DEALLOCATE stmt. struct ErrorManager { void init(Fortran::lower::AbstractConverter &converter, mlir::Location loc, const Fortran::lower::SomeExpr *statExpr, const Fortran::lower::SomeExpr *errMsgExpr) { Fortran::lower::StatementContext stmtCtx; fir::FirOpBuilder &builder = converter.getFirOpBuilder(); hasStat = builder.createBool(loc, statExpr != nullptr); statAddr = statExpr ? fir::getBase(converter.genExprAddr(loc, statExpr, stmtCtx)) : mlir::Value{}; errMsgAddr = statExpr && errMsgExpr ? builder.createBox(loc, converter.genExprAddr(loc, errMsgExpr, stmtCtx)) : builder.create( loc, fir::BoxType::get(mlir::NoneType::get(builder.getContext()))); sourceFile = fir::factory::locationToFilename(builder, loc); sourceLine = fir::factory::locationToLineNo(builder, loc, builder.getIntegerType(32)); } bool hasStatSpec() const { return static_cast(statAddr); } void genStatCheck(fir::FirOpBuilder &builder, mlir::Location loc) { if (statValue) { mlir::Value zero = builder.createIntegerConstant(loc, statValue.getType(), 0); auto cmp = builder.create( loc, mlir::arith::CmpIPredicate::eq, statValue, zero); auto ifOp = builder.create(loc, cmp, /*withElseRegion=*/false); builder.setInsertionPointToStart(&ifOp.getThenRegion().front()); } } void assignStat(fir::FirOpBuilder &builder, mlir::Location loc, mlir::Value stat) { if (hasStatSpec()) { assert(stat && "missing stat value"); mlir::Value castStat = builder.createConvert( loc, fir::dyn_cast_ptrEleTy(statAddr.getType()), stat); builder.create(loc, castStat, statAddr); statValue = stat; } } mlir::Value hasStat; mlir::Value errMsgAddr; mlir::Value sourceFile; mlir::Value sourceLine; private: mlir::Value statAddr; // STAT variable address mlir::Value statValue; // current runtime STAT value }; //===----------------------------------------------------------------------===// // Allocatables runtime call generators //===----------------------------------------------------------------------===// using namespace Fortran::runtime; /// Generate a runtime call to set the bounds of an allocatable or pointer /// descriptor. static void genRuntimeSetBounds(fir::FirOpBuilder &builder, mlir::Location loc, const fir::MutableBoxValue &box, mlir::Value dimIndex, mlir::Value lowerBound, mlir::Value upperBound) { mlir::func::FuncOp callee = box.isPointer() ? fir::runtime::getRuntimeFunc(loc, builder) : fir::runtime::getRuntimeFunc( loc, builder); llvm::SmallVector args{box.getAddr(), dimIndex, lowerBound, upperBound}; llvm::SmallVector operands; for (auto [fst, snd] : llvm::zip(args, callee.getFunctionType().getInputs())) operands.emplace_back(builder.createConvert(loc, snd, fst)); builder.create(loc, callee, operands); } /// Generate runtime call to set the lengths of a character allocatable or /// pointer descriptor. static void genRuntimeInitCharacter(fir::FirOpBuilder &builder, mlir::Location loc, const fir::MutableBoxValue &box, mlir::Value len) { mlir::func::FuncOp callee = box.isPointer() ? fir::runtime::getRuntimeFunc( loc, builder) : fir::runtime::getRuntimeFunc( loc, builder); llvm::ArrayRef inputTypes = callee.getFunctionType().getInputs(); if (inputTypes.size() != 5) fir::emitFatalError( loc, "AllocatableInitCharacter runtime interface not as expected"); llvm::SmallVector args; args.push_back(builder.createConvert(loc, inputTypes[0], box.getAddr())); args.push_back(builder.createConvert(loc, inputTypes[1], len)); int kind = box.getEleTy().cast().getFKind(); args.push_back(builder.createIntegerConstant(loc, inputTypes[2], kind)); int rank = box.rank(); args.push_back(builder.createIntegerConstant(loc, inputTypes[3], rank)); // TODO: coarrays int corank = 0; args.push_back(builder.createIntegerConstant(loc, inputTypes[4], corank)); builder.create(loc, callee, args); } /// Generate a sequence of runtime calls to allocate memory. static mlir::Value genRuntimeAllocate(fir::FirOpBuilder &builder, mlir::Location loc, const fir::MutableBoxValue &box, ErrorManager &errorManager) { mlir::func::FuncOp callee = box.isPointer() ? fir::runtime::getRuntimeFunc(loc, builder) : fir::runtime::getRuntimeFunc(loc, builder); llvm::SmallVector args{ box.getAddr(), errorManager.hasStat, errorManager.errMsgAddr, errorManager.sourceFile, errorManager.sourceLine}; llvm::SmallVector operands; for (auto [fst, snd] : llvm::zip(args, callee.getFunctionType().getInputs())) operands.emplace_back(builder.createConvert(loc, snd, fst)); return builder.create(loc, callee, operands).getResult(0); } /// Generate a runtime call to deallocate memory. static mlir::Value genRuntimeDeallocate(fir::FirOpBuilder &builder, mlir::Location loc, const fir::MutableBoxValue &box, ErrorManager &errorManager) { // Ensure fir.box is up-to-date before passing it to deallocate runtime. mlir::Value boxAddress = fir::factory::getMutableIRBox(builder, loc, box); mlir::func::FuncOp callee = box.isPointer() ? fir::runtime::getRuntimeFunc(loc, builder) : fir::runtime::getRuntimeFunc( loc, builder); llvm::SmallVector args{ boxAddress, errorManager.hasStat, errorManager.errMsgAddr, errorManager.sourceFile, errorManager.sourceLine}; llvm::SmallVector operands; for (auto [fst, snd] : llvm::zip(args, callee.getFunctionType().getInputs())) operands.emplace_back(builder.createConvert(loc, snd, fst)); return builder.create(loc, callee, operands).getResult(0); } //===----------------------------------------------------------------------===// // Allocate statement implementation //===----------------------------------------------------------------------===// /// Helper to get symbol from AllocateObject. static const Fortran::semantics::Symbol & unwrapSymbol(const Fortran::parser::AllocateObject &allocObj) { const Fortran::parser::Name &lastName = Fortran::parser::GetLastName(allocObj); assert(lastName.symbol); return *lastName.symbol; } static fir::MutableBoxValue genMutableBoxValue(Fortran::lower::AbstractConverter &converter, mlir::Location loc, const Fortran::parser::AllocateObject &allocObj) { const Fortran::lower::SomeExpr *expr = Fortran::semantics::GetExpr(allocObj); assert(expr && "semantic analysis failure"); return converter.genExprMutableBox(loc, *expr); } /// Implement Allocate statement lowering. class AllocateStmtHelper { public: AllocateStmtHelper(Fortran::lower::AbstractConverter &converter, const Fortran::parser::AllocateStmt &stmt, mlir::Location loc) : converter{converter}, builder{converter.getFirOpBuilder()}, stmt{stmt}, loc{loc} {} void lower() { visitAllocateOptions(); lowerAllocateLengthParameters(); errorManager.init(converter, loc, statExpr, errMsgExpr); if (sourceExpr || moldExpr) TODO(loc, "lower MOLD/SOURCE expr in allocate"); mlir::OpBuilder::InsertPoint insertPt = builder.saveInsertionPoint(); for (const auto &allocation : std::get>(stmt.t)) lowerAllocation(unwrapAllocation(allocation)); builder.restoreInsertionPoint(insertPt); } private: struct Allocation { const Fortran::parser::Allocation &alloc; const Fortran::semantics::DeclTypeSpec &type; bool hasCoarraySpec() const { return std::get>( alloc.t) .has_value(); } const Fortran::parser::AllocateObject &getAllocObj() const { return std::get(alloc.t); } const Fortran::semantics::Symbol &getSymbol() const { return unwrapSymbol(getAllocObj()); } const std::list &getShapeSpecs() const { return std::get>(alloc.t); } }; Allocation unwrapAllocation(const Fortran::parser::Allocation &alloc) { const auto &allocObj = std::get(alloc.t); const Fortran::semantics::Symbol &symbol = unwrapSymbol(allocObj); assert(symbol.GetType()); return Allocation{alloc, *symbol.GetType()}; } void visitAllocateOptions() { for (const auto &allocOption : std::get>(stmt.t)) std::visit( Fortran::common::visitors{ [&](const Fortran::parser::StatOrErrmsg &statOrErr) { std::visit( Fortran::common::visitors{ [&](const Fortran::parser::StatVariable &statVar) { statExpr = Fortran::semantics::GetExpr(statVar); }, [&](const Fortran::parser::MsgVariable &errMsgVar) { errMsgExpr = Fortran::semantics::GetExpr(errMsgVar); }, }, statOrErr.u); }, [&](const Fortran::parser::AllocOpt::Source &source) { sourceExpr = Fortran::semantics::GetExpr(source.v.value()); }, [&](const Fortran::parser::AllocOpt::Mold &mold) { moldExpr = Fortran::semantics::GetExpr(mold.v.value()); }, }, allocOption.u); } void lowerAllocation(const Allocation &alloc) { fir::MutableBoxValue boxAddr = genMutableBoxValue(converter, loc, alloc.getAllocObj()); if (sourceExpr) { genSourceAllocation(alloc, boxAddr); } else if (moldExpr) { genMoldAllocation(alloc, boxAddr); } else { genSimpleAllocation(alloc, boxAddr); } } static bool lowerBoundsAreOnes(const Allocation &alloc) { for (const Fortran::parser::AllocateShapeSpec &shapeSpec : alloc.getShapeSpecs()) if (std::get<0>(shapeSpec.t)) return false; return true; } /// Build name for the fir::allocmem generated for alloc. std::string mangleAlloc(const Allocation &alloc) { return converter.mangleName(alloc.getSymbol()) + ".alloc"; } /// Generate allocation without runtime calls. /// Only for intrinsic types. No coarrays, no polymorphism. No error recovery. void genInlinedAllocation(const Allocation &alloc, const fir::MutableBoxValue &box) { llvm::SmallVector lbounds; llvm::SmallVector extents; Fortran::lower::StatementContext stmtCtx; mlir::Type idxTy = builder.getIndexType(); bool lBoundsAreOnes = lowerBoundsAreOnes(alloc); mlir::Value one = builder.createIntegerConstant(loc, idxTy, 1); for (const Fortran::parser::AllocateShapeSpec &shapeSpec : alloc.getShapeSpecs()) { mlir::Value lb; if (!lBoundsAreOnes) { if (const std::optional &lbExpr = std::get<0>(shapeSpec.t)) { lb = fir::getBase(converter.genExprValue( loc, Fortran::semantics::GetExpr(*lbExpr), stmtCtx)); lb = builder.createConvert(loc, idxTy, lb); } else { lb = one; } lbounds.emplace_back(lb); } mlir::Value ub = fir::getBase(converter.genExprValue( loc, Fortran::semantics::GetExpr(std::get<1>(shapeSpec.t)), stmtCtx)); ub = builder.createConvert(loc, idxTy, ub); if (lb) { mlir::Value diff = builder.create(loc, ub, lb); extents.emplace_back( builder.create(loc, diff, one)); } else { extents.emplace_back(ub); } } fir::factory::genInlinedAllocation(builder, loc, box, lbounds, extents, lenParams, mangleAlloc(alloc)); } void genSimpleAllocation(const Allocation &alloc, const fir::MutableBoxValue &box) { if (!box.isDerived() && !errorManager.hasStatSpec() && !alloc.type.IsPolymorphic() && !alloc.hasCoarraySpec() && !useAllocateRuntime) { genInlinedAllocation(alloc, box); return; } // Generate a sequence of runtime calls. errorManager.genStatCheck(builder, loc); if (box.isPointer()) { // For pointers, the descriptor may still be uninitialized (see Fortran // 2018 19.5.2.2). The allocation runtime needs to be given a descriptor // with initialized rank, types and attributes. Initialize the descriptor // here to ensure these constraints are fulfilled. mlir::Value nullPointer = fir::factory::createUnallocatedBox( builder, loc, box.getBoxTy(), box.nonDeferredLenParams()); builder.create(loc, nullPointer, box.getAddr()); } else { assert(box.isAllocatable() && "must be an allocatable"); // For allocatables, sync the MutableBoxValue and descriptor before the // calls in case it is tracked locally by a set of variables. fir::factory::getMutableIRBox(builder, loc, box); } if (alloc.hasCoarraySpec()) TODO(loc, "coarray allocation"); if (alloc.type.IsPolymorphic()) genSetType(alloc, box); genSetDeferredLengthParameters(alloc, box); // Set bounds for arrays mlir::Type idxTy = builder.getIndexType(); mlir::Type i32Ty = builder.getIntegerType(32); Fortran::lower::StatementContext stmtCtx; for (const auto &iter : llvm::enumerate(alloc.getShapeSpecs())) { mlir::Value lb; const auto &bounds = iter.value().t; if (const std::optional &lbExpr = std::get<0>(bounds)) lb = fir::getBase(converter.genExprValue( loc, Fortran::semantics::GetExpr(*lbExpr), stmtCtx)); else lb = builder.createIntegerConstant(loc, idxTy, 1); mlir::Value ub = fir::getBase(converter.genExprValue( loc, Fortran::semantics::GetExpr(std::get<1>(bounds)), stmtCtx)); mlir::Value dimIndex = builder.createIntegerConstant(loc, i32Ty, iter.index()); // Runtime call genRuntimeSetBounds(builder, loc, box, dimIndex, lb, ub); } mlir::Value stat = genRuntimeAllocate(builder, loc, box, errorManager); fir::factory::syncMutableBoxFromIRBox(builder, loc, box); errorManager.assignStat(builder, loc, stat); } /// Lower the length parameters that may be specified in the optional /// type specification. void lowerAllocateLengthParameters() { const Fortran::semantics::DeclTypeSpec *typeSpec = getIfAllocateStmtTypeSpec(); if (!typeSpec) return; if (const Fortran::semantics::DerivedTypeSpec *derived = typeSpec->AsDerived()) if (Fortran::semantics::CountLenParameters(*derived) > 0) TODO(loc, "TODO: setting derived type params in allocation"); if (typeSpec->category() == Fortran::semantics::DeclTypeSpec::Category::Character) { Fortran::semantics::ParamValue lenParam = typeSpec->characterTypeSpec().length(); if (Fortran::semantics::MaybeIntExpr intExpr = lenParam.GetExplicit()) { Fortran::lower::StatementContext stmtCtx; Fortran::lower::SomeExpr lenExpr{*intExpr}; lenParams.push_back( fir::getBase(converter.genExprValue(loc, lenExpr, stmtCtx))); } } } // Set length parameters in the box stored in boxAddr. // This must be called before setting the bounds because it may use // Init runtime calls that may set the bounds to zero. void genSetDeferredLengthParameters(const Allocation &alloc, const fir::MutableBoxValue &box) { if (lenParams.empty()) return; // TODO: in case a length parameter was not deferred, insert a runtime check // that the length is the same (AllocatableCheckLengthParameter runtime // call). if (box.isCharacter()) genRuntimeInitCharacter(builder, loc, box, lenParams[0]); if (box.isDerived()) TODO(loc, "derived type length parameters in allocate"); } void genSourceAllocation(const Allocation &, const fir::MutableBoxValue &) { TODO(loc, "SOURCE allocation lowering"); } void genMoldAllocation(const Allocation &, const fir::MutableBoxValue &) { TODO(loc, "MOLD allocation lowering"); } void genSetType(const Allocation &, const fir::MutableBoxValue &) { TODO(loc, "Polymorphic entity allocation lowering"); } /// Returns a pointer to the DeclTypeSpec if a type-spec is provided in the /// allocate statement. Returns a null pointer otherwise. const Fortran::semantics::DeclTypeSpec *getIfAllocateStmtTypeSpec() const { if (const auto &typeSpec = std::get>(stmt.t)) return typeSpec->declTypeSpec; return nullptr; } Fortran::lower::AbstractConverter &converter; fir::FirOpBuilder &builder; const Fortran::parser::AllocateStmt &stmt; const Fortran::lower::SomeExpr *sourceExpr{nullptr}; const Fortran::lower::SomeExpr *moldExpr{nullptr}; const Fortran::lower::SomeExpr *statExpr{nullptr}; const Fortran::lower::SomeExpr *errMsgExpr{nullptr}; // If the allocate has a type spec, lenParams contains the // value of the length parameters that were specified inside. llvm::SmallVector lenParams; ErrorManager errorManager; mlir::Location loc; }; } // namespace void Fortran::lower::genAllocateStmt( Fortran::lower::AbstractConverter &converter, const Fortran::parser::AllocateStmt &stmt, mlir::Location loc) { AllocateStmtHelper{converter, stmt, loc}.lower(); return; } //===----------------------------------------------------------------------===// // Deallocate statement implementation //===----------------------------------------------------------------------===// // Generate deallocation of a pointer/allocatable. static void genDeallocate(fir::FirOpBuilder &builder, mlir::Location loc, const fir::MutableBoxValue &box, ErrorManager &errorManager) { // Deallocate intrinsic types inline. if (!box.isDerived() && !errorManager.hasStatSpec() && !useAllocateRuntime) { fir::factory::genInlinedDeallocate(builder, loc, box); return; } // Use runtime calls to deallocate descriptor cases. Sync MutableBoxValue // with its descriptor before and after calls if needed. errorManager.genStatCheck(builder, loc); mlir::Value stat = genRuntimeDeallocate(builder, loc, box, errorManager); fir::factory::syncMutableBoxFromIRBox(builder, loc, box); errorManager.assignStat(builder, loc, stat); } void Fortran::lower::genDeallocateStmt( Fortran::lower::AbstractConverter &converter, const Fortran::parser::DeallocateStmt &stmt, mlir::Location loc) { const Fortran::lower::SomeExpr *statExpr = nullptr; const Fortran::lower::SomeExpr *errMsgExpr = nullptr; for (const Fortran::parser::StatOrErrmsg &statOrErr : std::get>(stmt.t)) std::visit(Fortran::common::visitors{ [&](const Fortran::parser::StatVariable &statVar) { statExpr = Fortran::semantics::GetExpr(statVar); }, [&](const Fortran::parser::MsgVariable &errMsgVar) { errMsgExpr = Fortran::semantics::GetExpr(errMsgVar); }, }, statOrErr.u); ErrorManager errorManager; errorManager.init(converter, loc, statExpr, errMsgExpr); fir::FirOpBuilder &builder = converter.getFirOpBuilder(); mlir::OpBuilder::InsertPoint insertPt = builder.saveInsertionPoint(); for (const Fortran::parser::AllocateObject &allocateObject : std::get>(stmt.t)) { fir::MutableBoxValue box = genMutableBoxValue(converter, loc, allocateObject); genDeallocate(builder, loc, box, errorManager); } builder.restoreInsertionPoint(insertPt); } //===----------------------------------------------------------------------===// // MutableBoxValue creation implementation //===----------------------------------------------------------------------===// /// Is this symbol a pointer to a pointer array that does not have the /// CONTIGUOUS attribute ? static inline bool isNonContiguousArrayPointer(const Fortran::semantics::Symbol &sym) { return Fortran::semantics::IsPointer(sym) && sym.Rank() != 0 && !sym.attrs().test(Fortran::semantics::Attr::CONTIGUOUS); } /// Is this a local procedure symbol in a procedure that contains internal /// procedures ? static bool mayBeCapturedInInternalProc(const Fortran::semantics::Symbol &sym) { const Fortran::semantics::Scope &owner = sym.owner(); Fortran::semantics::Scope::Kind kind = owner.kind(); // Test if this is a procedure scope that contains a subprogram scope that is // not an interface. if (kind == Fortran::semantics::Scope::Kind::Subprogram || kind == Fortran::semantics::Scope::Kind::MainProgram) for (const Fortran::semantics::Scope &childScope : owner.children()) if (childScope.kind() == Fortran::semantics::Scope::Kind::Subprogram) if (const Fortran::semantics::Symbol *childSym = childScope.symbol()) if (const auto *details = childSym->detailsIf()) if (!details->isInterface()) return true; return false; } /// In case it is safe to track the properties in variables outside a /// descriptor, create the variables to hold the mutable properties of the /// entity var. The variables are not initialized here. static fir::MutableProperties createMutableProperties(Fortran::lower::AbstractConverter &converter, mlir::Location loc, const Fortran::lower::pft::Variable &var, mlir::ValueRange nonDeferredParams) { fir::FirOpBuilder &builder = converter.getFirOpBuilder(); const Fortran::semantics::Symbol &sym = var.getSymbol(); // Globals and dummies may be associated, creating local variables would // require keeping the values and descriptor before and after every single // impure calls in the current scope (not only the ones taking the variable as // arguments. All.) Volatile means the variable may change in ways not defined // per Fortran, so lowering can most likely not keep the descriptor and values // in sync as needed. // Pointers to non contiguous arrays need to be represented with a fir.box to // account for the discontiguity. // Pointer/Allocatable in internal procedure are descriptors in the host link, // and it would increase complexity to sync this descriptor with the local // values every time the host link is escaping. if (var.isGlobal() || Fortran::semantics::IsDummy(sym) || Fortran::semantics::IsFunctionResult(sym) || sym.attrs().test(Fortran::semantics::Attr::VOLATILE) || isNonContiguousArrayPointer(sym) || useAllocateRuntime || useDescForMutableBox || mayBeCapturedInInternalProc(sym)) return {}; fir::MutableProperties mutableProperties; std::string name = converter.mangleName(sym); mlir::Type baseAddrTy = converter.genType(sym); if (auto boxType = baseAddrTy.dyn_cast()) baseAddrTy = boxType.getEleTy(); // Allocate and set a variable to hold the address. // It will be set to null in setUnallocatedStatus. mutableProperties.addr = builder.allocateLocal(loc, baseAddrTy, name + ".addr", "", /*shape=*/llvm::None, /*typeparams=*/llvm::None); // Allocate variables to hold lower bounds and extents. int rank = sym.Rank(); mlir::Type idxTy = builder.getIndexType(); for (decltype(rank) i = 0; i < rank; ++i) { mlir::Value lboundVar = builder.allocateLocal(loc, idxTy, name + ".lb" + std::to_string(i), "", /*shape=*/llvm::None, /*typeparams=*/llvm::None); mlir::Value extentVar = builder.allocateLocal(loc, idxTy, name + ".ext" + std::to_string(i), "", /*shape=*/llvm::None, /*typeparams=*/llvm::None); mutableProperties.lbounds.emplace_back(lboundVar); mutableProperties.extents.emplace_back(extentVar); } // Allocate variable to hold deferred length parameters. mlir::Type eleTy = baseAddrTy; if (auto newTy = fir::dyn_cast_ptrEleTy(eleTy)) eleTy = newTy; if (auto seqTy = eleTy.dyn_cast()) eleTy = seqTy.getEleTy(); if (auto record = eleTy.dyn_cast()) if (record.getNumLenParams() != 0) TODO(loc, "deferred length type parameters."); if (fir::isa_char(eleTy) && nonDeferredParams.empty()) { mlir::Value lenVar = builder.allocateLocal(loc, builder.getCharacterLengthType(), name + ".len", "", /*shape=*/llvm::None, /*typeparams=*/llvm::None); mutableProperties.deferredParams.emplace_back(lenVar); } return mutableProperties; } fir::MutableBoxValue Fortran::lower::createMutableBox( Fortran::lower::AbstractConverter &converter, mlir::Location loc, const Fortran::lower::pft::Variable &var, mlir::Value boxAddr, mlir::ValueRange nonDeferredParams) { fir::MutableProperties mutableProperties = createMutableProperties(converter, loc, var, nonDeferredParams); fir::MutableBoxValue box(boxAddr, nonDeferredParams, mutableProperties); fir::FirOpBuilder &builder = converter.getFirOpBuilder(); if (!var.isGlobal() && !Fortran::semantics::IsDummy(var.getSymbol())) fir::factory::disassociateMutableBox(builder, loc, box); return box; } //===----------------------------------------------------------------------===// // MutableBoxValue reading interface implementation //===----------------------------------------------------------------------===// bool Fortran::lower::isArraySectionWithoutVectorSubscript( const Fortran::lower::SomeExpr &expr) { return expr.Rank() > 0 && Fortran::evaluate::IsVariable(expr) && !Fortran::evaluate::UnwrapWholeSymbolDataRef(expr) && !Fortran::evaluate::HasVectorSubscript(expr); } void Fortran::lower::associateMutableBox( Fortran::lower::AbstractConverter &converter, mlir::Location loc, const fir::MutableBoxValue &box, const Fortran::lower::SomeExpr &source, mlir::ValueRange lbounds, Fortran::lower::StatementContext &stmtCtx) { fir::FirOpBuilder &builder = converter.getFirOpBuilder(); if (Fortran::evaluate::UnwrapExpr(source)) { fir::factory::disassociateMutableBox(builder, loc, box); return; } // The right hand side is not be evaluated into a temp. Array sections can // typically be represented as a value of type `!fir.box`. However, an // expression that uses vector subscripts cannot be emboxed. In that case, // generate a reference to avoid having to later use a fir.rebox to implement // the pointer association. fir::ExtendedValue rhs = isArraySectionWithoutVectorSubscript(source) ? converter.genExprBox(loc, source, stmtCtx) : converter.genExprAddr(loc, source, stmtCtx); fir::factory::associateMutableBox(builder, loc, box, rhs, lbounds); } bool Fortran::lower::isWholeAllocatable(const Fortran::lower::SomeExpr &expr) { if (const Fortran::semantics::Symbol *sym = Fortran::evaluate::UnwrapWholeSymbolOrComponentDataRef(expr)) return Fortran::semantics::IsAllocatable(*sym); return false; } bool Fortran::lower::isWholePointer(const Fortran::lower::SomeExpr &expr) { if (const Fortran::semantics::Symbol *sym = Fortran::evaluate::UnwrapWholeSymbolOrComponentDataRef(expr)) return Fortran::semantics::IsPointer(*sym); return false; }