164ab3302SCarolineConcatto //===-- lib/Evaluate/fold.cpp ---------------------------------------------===// 264ab3302SCarolineConcatto // 364ab3302SCarolineConcatto // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 464ab3302SCarolineConcatto // See https://llvm.org/LICENSE.txt for license information. 564ab3302SCarolineConcatto // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 664ab3302SCarolineConcatto // 764ab3302SCarolineConcatto //===----------------------------------------------------------------------===// 864ab3302SCarolineConcatto 964ab3302SCarolineConcatto #include "flang/Evaluate/fold.h" 1064ab3302SCarolineConcatto #include "fold-implementation.h" 11*a50bb84eSpeter klausler #include "flang/Evaluate/characteristics.h" 1264ab3302SCarolineConcatto 1364ab3302SCarolineConcatto namespace Fortran::evaluate { 1464ab3302SCarolineConcatto 15*a50bb84eSpeter klausler characteristics::TypeAndShape Fold( 16*a50bb84eSpeter klausler FoldingContext &context, characteristics::TypeAndShape &&x) { 17*a50bb84eSpeter klausler x.Rewrite(context); 18*a50bb84eSpeter klausler return std::move(x); 19*a50bb84eSpeter klausler } 20*a50bb84eSpeter klausler 2164ab3302SCarolineConcatto std::optional<Constant<SubscriptInteger>> GetConstantSubscript( 2264ab3302SCarolineConcatto FoldingContext &context, Subscript &ss, const NamedEntity &base, int dim) { 2364ab3302SCarolineConcatto ss = FoldOperation(context, std::move(ss)); 2464ab3302SCarolineConcatto return std::visit( 2564ab3302SCarolineConcatto common::visitors{ 2664ab3302SCarolineConcatto [](IndirectSubscriptIntegerExpr &expr) 2764ab3302SCarolineConcatto -> std::optional<Constant<SubscriptInteger>> { 2864ab3302SCarolineConcatto if (const auto *constant{ 2964ab3302SCarolineConcatto UnwrapConstantValue<SubscriptInteger>(expr.value())}) { 3064ab3302SCarolineConcatto return *constant; 3164ab3302SCarolineConcatto } else { 3264ab3302SCarolineConcatto return std::nullopt; 3364ab3302SCarolineConcatto } 3464ab3302SCarolineConcatto }, 3564ab3302SCarolineConcatto [&](Triplet &triplet) -> std::optional<Constant<SubscriptInteger>> { 3664ab3302SCarolineConcatto auto lower{triplet.lower()}, upper{triplet.upper()}; 3764ab3302SCarolineConcatto std::optional<ConstantSubscript> stride{ToInt64(triplet.stride())}; 3864ab3302SCarolineConcatto if (!lower) { 3964ab3302SCarolineConcatto lower = GetLowerBound(context, base, dim); 4064ab3302SCarolineConcatto } 4164ab3302SCarolineConcatto if (!upper) { 4264ab3302SCarolineConcatto upper = 4364ab3302SCarolineConcatto ComputeUpperBound(context, GetLowerBound(context, base, dim), 4464ab3302SCarolineConcatto GetExtent(context, base, dim)); 4564ab3302SCarolineConcatto } 4664ab3302SCarolineConcatto auto lbi{ToInt64(lower)}, ubi{ToInt64(upper)}; 4764ab3302SCarolineConcatto if (lbi && ubi && stride && *stride != 0) { 4864ab3302SCarolineConcatto std::vector<SubscriptInteger::Scalar> values; 4964ab3302SCarolineConcatto while ((*stride > 0 && *lbi <= *ubi) || 5064ab3302SCarolineConcatto (*stride < 0 && *lbi >= *ubi)) { 5164ab3302SCarolineConcatto values.emplace_back(*lbi); 5264ab3302SCarolineConcatto *lbi += *stride; 5364ab3302SCarolineConcatto } 5464ab3302SCarolineConcatto return Constant<SubscriptInteger>{std::move(values), 5564ab3302SCarolineConcatto ConstantSubscripts{ 5664ab3302SCarolineConcatto static_cast<ConstantSubscript>(values.size())}}; 5764ab3302SCarolineConcatto } else { 5864ab3302SCarolineConcatto return std::nullopt; 5964ab3302SCarolineConcatto } 6064ab3302SCarolineConcatto }, 6164ab3302SCarolineConcatto }, 6264ab3302SCarolineConcatto ss.u); 6364ab3302SCarolineConcatto } 6464ab3302SCarolineConcatto 6564ab3302SCarolineConcatto Expr<SomeDerived> FoldOperation( 6664ab3302SCarolineConcatto FoldingContext &context, StructureConstructor &&structure) { 674171f80dSpeter klausler StructureConstructor ctor{structure.derivedTypeSpec()}; 684171f80dSpeter klausler bool constantExtents{true}; 6964ab3302SCarolineConcatto for (auto &&[symbol, value] : std::move(structure)) { 704171f80dSpeter klausler auto expr{Fold(context, std::move(value.value()))}; 714171f80dSpeter klausler if (!IsPointer(symbol)) { 72641ede93Speter klausler bool ok{false}; 73641ede93Speter klausler if (auto valueShape{GetConstantExtents(context, expr)}) { 744171f80dSpeter klausler if (auto componentShape{GetConstantExtents(context, symbol)}) { 754171f80dSpeter klausler if (GetRank(*componentShape) > 0 && GetRank(*valueShape) == 0) { 76641ede93Speter klausler expr = ScalarConstantExpander{std::move(*componentShape)}.Expand( 774171f80dSpeter klausler std::move(expr)); 78641ede93Speter klausler ok = expr.Rank() > 0; 794171f80dSpeter klausler } else { 80641ede93Speter klausler ok = *valueShape == *componentShape; 814171f80dSpeter klausler } 824171f80dSpeter klausler } 83641ede93Speter klausler } 84641ede93Speter klausler if (!ok) { 854171f80dSpeter klausler constantExtents = false; 864171f80dSpeter klausler } 874171f80dSpeter klausler } 884171f80dSpeter klausler ctor.Add(symbol, Fold(context, std::move(expr))); 894171f80dSpeter klausler } 904171f80dSpeter klausler if (constantExtents && IsConstantExpr(ctor)) { 914171f80dSpeter klausler return Expr<SomeDerived>{Constant<SomeDerived>{std::move(ctor)}}; 924171f80dSpeter klausler } else { 934171f80dSpeter klausler return Expr<SomeDerived>{std::move(ctor)}; 944171f80dSpeter klausler } 9564ab3302SCarolineConcatto } 9664ab3302SCarolineConcatto 9764ab3302SCarolineConcatto Component FoldOperation(FoldingContext &context, Component &&component) { 9864ab3302SCarolineConcatto return {FoldOperation(context, std::move(component.base())), 9964ab3302SCarolineConcatto component.GetLastSymbol()}; 10064ab3302SCarolineConcatto } 10164ab3302SCarolineConcatto 10264ab3302SCarolineConcatto NamedEntity FoldOperation(FoldingContext &context, NamedEntity &&x) { 10364ab3302SCarolineConcatto if (Component * c{x.UnwrapComponent()}) { 10464ab3302SCarolineConcatto return NamedEntity{FoldOperation(context, std::move(*c))}; 10564ab3302SCarolineConcatto } else { 10664ab3302SCarolineConcatto return std::move(x); 10764ab3302SCarolineConcatto } 10864ab3302SCarolineConcatto } 10964ab3302SCarolineConcatto 11064ab3302SCarolineConcatto Triplet FoldOperation(FoldingContext &context, Triplet &&triplet) { 11164ab3302SCarolineConcatto MaybeExtentExpr lower{triplet.lower()}; 11264ab3302SCarolineConcatto MaybeExtentExpr upper{triplet.upper()}; 11364ab3302SCarolineConcatto return {Fold(context, std::move(lower)), Fold(context, std::move(upper)), 11464ab3302SCarolineConcatto Fold(context, triplet.stride())}; 11564ab3302SCarolineConcatto } 11664ab3302SCarolineConcatto 11764ab3302SCarolineConcatto Subscript FoldOperation(FoldingContext &context, Subscript &&subscript) { 1181f879005STim Keith return std::visit(common::visitors{ 11964ab3302SCarolineConcatto [&](IndirectSubscriptIntegerExpr &&expr) { 12064ab3302SCarolineConcatto expr.value() = Fold(context, std::move(expr.value())); 12164ab3302SCarolineConcatto return Subscript(std::move(expr)); 12264ab3302SCarolineConcatto }, 12364ab3302SCarolineConcatto [&](Triplet &&triplet) { 1241f879005STim Keith return Subscript( 1251f879005STim Keith FoldOperation(context, std::move(triplet))); 12664ab3302SCarolineConcatto }, 12764ab3302SCarolineConcatto }, 12864ab3302SCarolineConcatto std::move(subscript.u)); 12964ab3302SCarolineConcatto } 13064ab3302SCarolineConcatto 13164ab3302SCarolineConcatto ArrayRef FoldOperation(FoldingContext &context, ArrayRef &&arrayRef) { 13264ab3302SCarolineConcatto NamedEntity base{FoldOperation(context, std::move(arrayRef.base()))}; 13364ab3302SCarolineConcatto for (Subscript &subscript : arrayRef.subscript()) { 13464ab3302SCarolineConcatto subscript = FoldOperation(context, std::move(subscript)); 13564ab3302SCarolineConcatto } 13664ab3302SCarolineConcatto return ArrayRef{std::move(base), std::move(arrayRef.subscript())}; 13764ab3302SCarolineConcatto } 13864ab3302SCarolineConcatto 13964ab3302SCarolineConcatto CoarrayRef FoldOperation(FoldingContext &context, CoarrayRef &&coarrayRef) { 14064ab3302SCarolineConcatto std::vector<Subscript> subscript; 14164ab3302SCarolineConcatto for (Subscript x : coarrayRef.subscript()) { 14264ab3302SCarolineConcatto subscript.emplace_back(FoldOperation(context, std::move(x))); 14364ab3302SCarolineConcatto } 14464ab3302SCarolineConcatto std::vector<Expr<SubscriptInteger>> cosubscript; 14564ab3302SCarolineConcatto for (Expr<SubscriptInteger> x : coarrayRef.cosubscript()) { 14664ab3302SCarolineConcatto cosubscript.emplace_back(Fold(context, std::move(x))); 14764ab3302SCarolineConcatto } 14864ab3302SCarolineConcatto CoarrayRef folded{std::move(coarrayRef.base()), std::move(subscript), 14964ab3302SCarolineConcatto std::move(cosubscript)}; 15064ab3302SCarolineConcatto if (std::optional<Expr<SomeInteger>> stat{coarrayRef.stat()}) { 15164ab3302SCarolineConcatto folded.set_stat(Fold(context, std::move(*stat))); 15264ab3302SCarolineConcatto } 15364ab3302SCarolineConcatto if (std::optional<Expr<SomeInteger>> team{coarrayRef.team()}) { 15464ab3302SCarolineConcatto folded.set_team( 15564ab3302SCarolineConcatto Fold(context, std::move(*team)), coarrayRef.teamIsTeamNumber()); 15664ab3302SCarolineConcatto } 15764ab3302SCarolineConcatto return folded; 15864ab3302SCarolineConcatto } 15964ab3302SCarolineConcatto 16064ab3302SCarolineConcatto DataRef FoldOperation(FoldingContext &context, DataRef &&dataRef) { 1611f879005STim Keith return std::visit(common::visitors{ 16264ab3302SCarolineConcatto [&](SymbolRef symbol) { return DataRef{*symbol}; }, 16364ab3302SCarolineConcatto [&](auto &&x) { 16464ab3302SCarolineConcatto return DataRef{FoldOperation(context, std::move(x))}; 16564ab3302SCarolineConcatto }, 16664ab3302SCarolineConcatto }, 16764ab3302SCarolineConcatto std::move(dataRef.u)); 16864ab3302SCarolineConcatto } 16964ab3302SCarolineConcatto 17064ab3302SCarolineConcatto Substring FoldOperation(FoldingContext &context, Substring &&substring) { 17164ab3302SCarolineConcatto auto lower{Fold(context, substring.lower())}; 17264ab3302SCarolineConcatto auto upper{Fold(context, substring.upper())}; 17364ab3302SCarolineConcatto if (const DataRef * dataRef{substring.GetParentIf<DataRef>()}) { 17464ab3302SCarolineConcatto return Substring{FoldOperation(context, DataRef{*dataRef}), 17564ab3302SCarolineConcatto std::move(lower), std::move(upper)}; 17664ab3302SCarolineConcatto } else { 17764ab3302SCarolineConcatto auto p{*substring.GetParentIf<StaticDataObject::Pointer>()}; 17864ab3302SCarolineConcatto return Substring{std::move(p), std::move(lower), std::move(upper)}; 17964ab3302SCarolineConcatto } 18064ab3302SCarolineConcatto } 18164ab3302SCarolineConcatto 18264ab3302SCarolineConcatto ComplexPart FoldOperation(FoldingContext &context, ComplexPart &&complexPart) { 18364ab3302SCarolineConcatto DataRef complex{complexPart.complex()}; 18464ab3302SCarolineConcatto return ComplexPart{ 18564ab3302SCarolineConcatto FoldOperation(context, std::move(complex)), complexPart.part()}; 18664ab3302SCarolineConcatto } 18764ab3302SCarolineConcatto 18864ab3302SCarolineConcatto std::optional<std::int64_t> GetInt64Arg( 18964ab3302SCarolineConcatto const std::optional<ActualArgument> &arg) { 19064ab3302SCarolineConcatto if (const auto *intExpr{UnwrapExpr<Expr<SomeInteger>>(arg)}) { 19164ab3302SCarolineConcatto return ToInt64(*intExpr); 19264ab3302SCarolineConcatto } else { 19364ab3302SCarolineConcatto return std::nullopt; 19464ab3302SCarolineConcatto } 19564ab3302SCarolineConcatto } 19664ab3302SCarolineConcatto 19764ab3302SCarolineConcatto std::optional<std::int64_t> GetInt64ArgOr( 19864ab3302SCarolineConcatto const std::optional<ActualArgument> &arg, std::int64_t defaultValue) { 19964ab3302SCarolineConcatto if (!arg) { 20064ab3302SCarolineConcatto return defaultValue; 20164ab3302SCarolineConcatto } else if (const auto *intExpr{UnwrapExpr<Expr<SomeInteger>>(arg)}) { 20264ab3302SCarolineConcatto return ToInt64(*intExpr); 20364ab3302SCarolineConcatto } else { 20464ab3302SCarolineConcatto return std::nullopt; 20564ab3302SCarolineConcatto } 20664ab3302SCarolineConcatto } 20764ab3302SCarolineConcatto 20864ab3302SCarolineConcatto Expr<ImpliedDoIndex::Result> FoldOperation( 20964ab3302SCarolineConcatto FoldingContext &context, ImpliedDoIndex &&iDo) { 21064ab3302SCarolineConcatto if (std::optional<ConstantSubscript> value{context.GetImpliedDo(iDo.name)}) { 21164ab3302SCarolineConcatto return Expr<ImpliedDoIndex::Result>{*value}; 21264ab3302SCarolineConcatto } else { 21364ab3302SCarolineConcatto return Expr<ImpliedDoIndex::Result>{std::move(iDo)}; 21464ab3302SCarolineConcatto } 21564ab3302SCarolineConcatto } 21664ab3302SCarolineConcatto 21764ab3302SCarolineConcatto template class ExpressionBase<SomeDerived>; 21864ab3302SCarolineConcatto template class ExpressionBase<SomeType>; 21964ab3302SCarolineConcatto 2201f879005STim Keith } // namespace Fortran::evaluate 221