9#ifndef FORTRAN_EVALUATE_FOLD_IMPLEMENTATION_H_
10#define FORTRAN_EVALUATE_FOLD_IMPLEMENTATION_H_
15#include "flang/Common/indirection.h"
16#include "flang/Common/template.h"
17#include "flang/Common/unwrap.h"
18#include "flang/Evaluate/characteristics.h"
19#include "flang/Evaluate/common.h"
20#include "flang/Evaluate/constant.h"
21#include "flang/Evaluate/expression.h"
22#include "flang/Evaluate/fold.h"
23#include "flang/Evaluate/formatting.h"
24#include "flang/Evaluate/intrinsics-library.h"
25#include "flang/Evaluate/intrinsics.h"
26#include "flang/Evaluate/shape.h"
27#include "flang/Evaluate/tools.h"
28#include "flang/Evaluate/traverse.h"
29#include "flang/Evaluate/type.h"
30#include "flang/Parser/message.h"
31#include "flang/Semantics/scope.h"
32#include "flang/Semantics/symbol.h"
33#include "flang/Semantics/tools.h"
52static constexpr bool useKahanSummation{
false};
55template <
typename T>
class Folder {
57 explicit Folder(
FoldingContext &c,
bool forOptionalArgument =
false)
58 : context_{c}, forOptionalArgument_{forOptionalArgument} {}
59 std::optional<Constant<T>> GetNamedConstant(
const Symbol &);
60 std::optional<Constant<T>> ApplySubscripts(
const Constant<T> &array,
65 std::optional<Constant<T>> GetConstantComponent(
67 std::optional<Constant<T>> Folding(
ArrayRef &);
68 std::optional<Constant<T>> Folding(
DataRef &);
70 Constant<T> *Folding(std::optional<ActualArgument> &);
85 bool forOptionalArgument_{
false};
88std::optional<Constant<SubscriptInteger>> GetConstantSubscript(
92template <
typename TR,
typename... TA>
93std::optional<std::function<Scalar<TR>(
FoldingContext &, Scalar<TA>...)>>
94GetHostRuntimeWrapper(
const std::string &name) {
95 std::vector<DynamicType> argTypes{TA{}.GetType()...};
96 if (
auto hostWrapper{GetHostRuntimeWrapper(name, TR{}.GetType(), argTypes)}) {
99 std::vector<Expr<SomeType>> genericArgs{
101 return GetScalarConstantValue<TR>(
102 (*hostWrapper)(context, std::move(genericArgs)))
117common::IfNoLvalue<Expr<ResultType<A>>, A> FoldOperation(
119 static_assert(!std::is_same_v<A, Expr<ResultType<A>>>,
120 "call Fold() instead for Expr<>");
137 return Folder<T>{context}.Folding(std::move(designator));
148std::optional<Constant<T>> Folder<T>::GetNamedConstant(
const Symbol &symbol0) {
149 const Symbol &symbol{ResolveAssociations(symbol0)};
150 if (IsNamedConstant(symbol)) {
151 if (
const auto *
object{
152 symbol.detailsIf<semantics::ObjectEntityDetails>()}) {
153 if (
const auto *constant{UnwrapConstantValue<T>(object->init())}) {
162std::optional<Constant<T>> Folder<T>::Folding(
ArrayRef &aRef) {
163 std::vector<Constant<SubscriptInteger>> subscripts;
166 if (
auto constant{GetConstantSubscript(context_, ss, aRef.base(), dim++)}) {
167 subscripts.emplace_back(std::move(*constant));
172 if (
Component * component{aRef.base().UnwrapComponent()}) {
173 return GetConstantComponent(*component, &subscripts);
175 GetNamedConstant(aRef.base().GetLastSymbol())}) {
176 return ApplySubscripts(*array, subscripts);
183std::optional<Constant<T>> Folder<T>::Folding(
DataRef &ref) {
184 return common::visit(
186 [
this](SymbolRef &sym) {
return GetNamedConstant(*sym); },
188 comp = FoldOperation(context_, std::move(comp));
189 return GetConstantComponent(comp);
192 aRef = FoldOperation(context_, std::move(aRef));
193 return Folding(aRef);
195 [](
CoarrayRef &) {
return std::optional<Constant<T>>{}; },
202std::optional<Constant<T>> Folder<T>::ApplySubscripts(
const Constant<T> &array,
204 const auto &shape{array.shape()};
205 const auto &lbounds{array.lbounds()};
206 int rank{GetRank(shape)};
207 CHECK(rank ==
static_cast<int>(subscripts.size()));
208 std::size_t elements{1};
209 ConstantSubscripts resultShape;
210 ConstantSubscripts ssLB;
211 for (
const auto &ss : subscripts) {
212 if (ss.Rank() == 1) {
213 resultShape.push_back(
static_cast<ConstantSubscript
>(ss.size()));
214 elements *= ss.size();
215 ssLB.push_back(ss.lbounds().front());
216 }
else if (ss.Rank() > 1) {
220 ConstantSubscripts ssAt(rank, 0), at(rank, 0), tmp(1, 0);
221 std::vector<Scalar<T>> values;
222 while (elements-- > 0) {
223 bool increment{
true};
225 for (
int j{0}; j < rank; ++j) {
226 if (subscripts[j].Rank() == 0) {
227 at[j] = subscripts[j].GetScalarValue().value().ToInt64();
229 CHECK(k < GetRank(resultShape));
230 tmp[0] = ssLB.at(k) + ssAt.at(k);
231 at[j] = subscripts[j].At(tmp).ToInt64();
233 if (++ssAt[k] == resultShape[k]) {
241 if (at[j] < lbounds[j] || at[j] >= lbounds[j] + shape[j]) {
242 context_.messages().Say(
243 "Subscript value (%jd) is out of range on dimension %d in reference to a constant array value"_err_en_US,
248 values.emplace_back(array.At(at));
249 CHECK(!increment || elements == 0);
250 CHECK(k == GetRank(resultShape));
252 if constexpr (T::category == TypeCategory::Character) {
253 return Constant<T>{array.LEN(), std::move(values), std::move(resultShape)};
254 }
else if constexpr (std::is_same_v<T, SomeDerived>) {
255 return Constant<T>{array.result().derivedTypeSpec(), std::move(values),
256 std::move(resultShape)};
258 return Constant<T>{std::move(values), std::move(resultShape)};
263std::optional<Constant<T>> Folder<T>::ApplyComponent(
266 if (
auto scalar{structures.GetScalarValue()}) {
267 if (std::optional<
Expr<SomeType>> expr{scalar->Find(component)}) {
268 if (
const Constant<T> *value{UnwrapConstantValue<T>(*expr)}) {
270 return ApplySubscripts(*value, *subscripts);
278 std::unique_ptr<ArrayConstructor<T>> array;
279 if (structures.empty()) {
282 ConstantSubscripts at{structures.lbounds()};
286 if (
const Constant<T> *value{UnwrapConstantValue<T>(expr.value())}) {
290 auto *typedExpr{UnwrapExpr<Expr<T>>(expr.value())};
292 array = std::make_unique<ArrayConstructor<T>>(*typedExpr);
293 if constexpr (T::category == TypeCategory::Character) {
298 if (
auto element{ApplySubscripts(*value, *subscripts)}) {
299 CHECK(element->Rank() == 0);
300 array->Push(
Expr<T>{std::move(*element)});
305 CHECK(value->Rank() == 0);
312 }
while (structures.IncrementSubscripts(at));
316 if (
auto *constant{UnwrapConstantValue<T>(result)}) {
317 return constant->Reshape(common::Clone(structures.shape()));
324std::optional<Constant<T>> Folder<T>::GetConstantComponent(
Component &component,
328 [&](
const Symbol &symbol) {
338 return std::optional<Constant<SomeDerived>>{};
341 component.base().u)}) {
342 return ApplyComponent(
343 std::move(*structures), component.GetLastSymbol(), subscripts);
350 if constexpr (T::category == TypeCategory::Character) {
351 if (
auto *substring{common::Unwrap<Substring>(designator.u)}) {
353 substring->Fold(context_)}) {
354 if (
const auto *specific{std::get_if<Expr<T>>(&folded->u)}) {
355 return std::move(*specific);
362 }
else if constexpr (T::category == TypeCategory::Real) {
363 if (
auto *zPart{std::get_if<ComplexPart>(&designator.u)}) {
364 *zPart = FoldOperation(context_, std::move(*zPart));
367 return Fold(context_,
369 zPart->part() == ComplexPart::Part::IM,
376 return common::visit(
378 [&](SymbolRef &&symbol) {
379 if (
auto constant{GetNamedConstant(*symbol)}) {
380 return Expr<T>{std::move(*constant)};
382 return Expr<T>{std::move(designator)};
385 aRef = FoldOperation(context_, std::move(aRef));
386 if (
auto c{Folding(aRef)}) {
393 component = FoldOperation(context_, std::move(component));
394 if (
auto c{GetConstantComponent(component)}) {
405 std::move(designator.u));
411Constant<T> *Folder<T>::Folding(std::optional<ActualArgument> &arg) {
412 if (
auto *expr{UnwrapExpr<Expr<SomeType>>(arg)}) {
413 *expr = Fold(context_, std::move(*expr));
414 if constexpr (T::category != TypeCategory::Derived) {
415 if (!UnwrapExpr<
Expr<T>>(*expr)) {
417 var{forOptionalArgument_
418 ? UnwrapWholeSymbolOrComponentDataRef(*expr)
420 var && (IsOptional(*var) || IsAllocatableOrObjectPointer(var))) {
422 }
else if (
auto converted{
423 ConvertToType(T::GetType(), std::move(*expr))}) {
424 *expr = Fold(context_, std::move(*converted));
428 return UnwrapConstantValue<T>(*expr);
433template <
typename... A, std::size_t... I>
434std::optional<std::tuple<const Constant<A> *...>> GetConstantArgumentsHelper(
436 bool hasOptionalArgument, std::index_sequence<I...>) {
437 static_assert(
sizeof...(A) > 0);
438 std::tuple<const Constant<A> *...> args{
439 Folder<A>{context, hasOptionalArgument}.Folding(arguments.at(I))...};
440 if ((... && (std::get<I>(args)))) {
447template <
typename... A>
448std::optional<std::tuple<const Constant<A> *...>> GetConstantArguments(
449 FoldingContext &context, ActualArguments &args,
bool hasOptionalArgument) {
450 return GetConstantArgumentsHelper<A...>(
451 context, args, hasOptionalArgument, std::index_sequence_for<A...>{});
454template <
typename... A, std::size_t... I>
455std::optional<std::tuple<Scalar<A>...>> GetScalarConstantArgumentsHelper(
456 FoldingContext &context, ActualArguments &args,
bool hasOptionalArgument,
457 std::index_sequence<I...>) {
459 GetConstantArguments<A...>(context, args, hasOptionalArgument)}) {
460 return std::tuple<Scalar<A>...>{
461 std::get<I>(*constArgs)->GetScalarValue().value()...};
467template <
typename... A>
468std::optional<std::tuple<Scalar<A>...>> GetScalarConstantArguments(
469 FoldingContext &context, ActualArguments &args,
bool hasOptionalArgument) {
470 return GetScalarConstantArgumentsHelper<A...>(
471 context, args, hasOptionalArgument, std::index_sequence_for<A...>{});
478template <
typename TR,
typename... TArgs>
479using ScalarFunc = std::function<Scalar<TR>(
const Scalar<TArgs> &...)>;
480template <
typename TR,
typename... TArgs>
481using ScalarFuncWithContext =
482 std::function<Scalar<TR>(
FoldingContext &,
const Scalar<TArgs> &...)>;
484template <
template <
typename,
typename...>
typename WrapperType,
typename TR,
485 typename... TA, std::size_t... I>
488 bool hasOptionalArgument, std::index_sequence<I...>) {
489 if (std::optional<std::tuple<
const Constant<TA> *...>> args{
490 GetConstantArguments<TA...>(
491 context, funcRef.arguments(), hasOptionalArgument)}) {
493 ConstantSubscripts shape;
495 const ConstantSubscripts *shapes[]{&std::get<I>(*args)->shape()...};
496 const int ranks[]{std::get<I>(*args)->Rank()...};
497 for (
unsigned int i{0}; i <
sizeof...(TA); ++i) {
503 if (shape != *shapes[i]) {
508 context.messages().Say(
509 "Arguments in elemental intrinsic function are not conformable"_err_en_US);
510 return Expr<TR>{std::move(funcRef)};
515 CHECK(rank == GetRank(shape));
517 std::vector<Scalar<TR>> results;
518 std::optional<uint64_t> n{TotalElementCount(shape)};
520 context.messages().Say(
521 "Too many elements in elemental intrinsic function result"_err_en_US);
522 return Expr<TR>{std::move(funcRef)};
526 ConstantSubscripts resultIndex(rank, 1);
527 ConstantSubscripts argIndex[]{std::get<I>(*args)->lbounds()...};
529 if constexpr (std::is_same_v<WrapperType<TR, TA...>,
530 ScalarFuncWithContext<TR, TA...>>) {
531 results.emplace_back(
532 func(context, std::get<I>(*args)->At(argIndex[I])...));
533 }
else if constexpr (std::is_same_v<WrapperType<TR, TA...>,
534 ScalarFunc<TR, TA...>>) {
535 results.emplace_back(func(std::get<I>(*args)->At(argIndex[I])...));
537 (std::get<I>(*args)->IncrementSubscripts(argIndex[I]), ...);
538 }
while (bounds.IncrementSubscripts(resultIndex));
541 if constexpr (TR::category == TypeCategory::Character) {
542 auto len{
static_cast<ConstantSubscript
>(
543 results.empty() ? 0 : results[0].length())};
545 }
else if constexpr (TR::category == TypeCategory::Derived) {
546 if (!results.empty()) {
550 std::move(results), std::move(shape)}};
556 return Expr<TR>{std::move(funcRef)};
559template <
typename TR,
typename... TA>
562 bool hasOptionalArgument =
false) {
563 return FoldElementalIntrinsicHelper<ScalarFunc, TR, TA...>(context,
564 std::move(funcRef), func, hasOptionalArgument,
565 std::index_sequence_for<TA...>{});
567template <
typename TR,
typename... TA>
570 bool hasOptionalArgument =
false) {
571 return FoldElementalIntrinsicHelper<ScalarFuncWithContext, TR, TA...>(context,
572 std::move(funcRef), func, hasOptionalArgument,
573 std::index_sequence_for<TA...>{});
576std::optional<std::int64_t> GetInt64ArgOr(
577 const std::optional<ActualArgument> &, std::int64_t defaultValue);
579template <
typename A,
typename B>
580std::optional<std::vector<A>> GetIntegerVector(
const B &x) {
581 static_assert(std::is_integral_v<A>);
582 if (
const auto *someInteger{UnwrapExpr<Expr<SomeInteger>>(x)}) {
583 return common::visit(
584 [](
const auto &typedExpr) -> std::optional<std::vector<A>> {
585 using T = ResultType<
decltype(typedExpr)>;
586 if (
const auto *constant{UnwrapConstantValue<T>(typedExpr)}) {
587 if (constant->Rank() == 1) {
588 std::vector<A> result;
589 for (
const auto &value : constant->values()) {
590 result.push_back(
static_cast<A
>(value.ToInt64()));
608 invalid.name = IntrinsicProcTable::InvalidName;
610 ActualArguments{std::move(funcRef.arguments())}}};
614 auto args{funcRef.arguments()};
615 CHECK(args.size() == 3);
616 const auto *array{UnwrapConstantValue<T>(args[0])};
617 const auto *shiftExpr{UnwrapExpr<Expr<SomeInteger>>(args[1])};
618 auto dim{GetInt64ArgOr(args[2], 1)};
619 if (!array || !shiftExpr || !dim) {
620 return Expr<T>{std::move(funcRef)};
622 auto convertedShift{Fold(context_,
624 const auto *shift{UnwrapConstantValue<SubscriptInteger>(convertedShift)};
626 return Expr<T>{std::move(funcRef)};
629 if (*dim < 1 || *dim > array->Rank()) {
630 context_.messages().Say(
"Invalid 'dim=' argument (%jd) in CSHIFT"_err_en_US,
631 static_cast<std::intmax_t
>(*dim));
632 }
else if (shift->Rank() > 0 && shift->Rank() != array->Rank() - 1) {
635 int rank{array->Rank()};
636 int zbDim{
static_cast<int>(*dim) - 1};
638 if (shift->Rank() > 0) {
640 for (
int j{0}; j < rank; ++j) {
642 if (array->shape()[j] != shift->shape()[k]) {
643 context_.messages().Say(
644 "Invalid 'shift=' argument in CSHIFT: extent on dimension %d is %jd but must be %jd"_err_en_US,
645 k + 1,
static_cast<std::intmax_t
>(shift->shape()[k]),
646 static_cast<std::intmax_t
>(array->shape()[j]));
654 std::vector<Scalar<T>> resultElements;
655 ConstantSubscripts arrayLB{array->lbounds()};
656 ConstantSubscripts arrayAt{arrayLB};
657 ConstantSubscript &dimIndex{arrayAt[zbDim]};
658 ConstantSubscript dimLB{dimIndex};
659 ConstantSubscript dimExtent{array->shape()[zbDim]};
660 ConstantSubscripts shiftLB{shift->lbounds()};
661 for (
auto n{GetSize(array->shape())}; n > 0; --n) {
662 ConstantSubscript origDimIndex{dimIndex};
663 ConstantSubscripts shiftAt;
664 if (shift->Rank() > 0) {
666 for (
int j{0}; j < rank; ++j) {
668 shiftAt.emplace_back(shiftLB[k++] + arrayAt[j] - arrayLB[j]);
672 ConstantSubscript shiftCount{shift->At(shiftAt).ToInt64()};
673 dimIndex = dimLB + ((dimIndex - dimLB + shiftCount) % dimExtent);
674 if (dimIndex < dimLB) {
675 dimIndex += dimExtent;
676 }
else if (dimIndex >= dimLB + dimExtent) {
677 dimIndex -= dimExtent;
679 resultElements.push_back(array->At(arrayAt));
680 dimIndex = origDimIndex;
681 array->IncrementSubscripts(arrayAt);
683 return Expr<T>{PackageConstant<T>(
684 std::move(resultElements), *array, array->shape())};
688 return MakeInvalidIntrinsic(std::move(funcRef));
692 auto args{funcRef.arguments()};
693 CHECK(args.size() == 4);
694 const auto *array{UnwrapConstantValue<T>(args[0])};
695 const auto *shiftExpr{UnwrapExpr<Expr<SomeInteger>>(args[1])};
696 auto dim{GetInt64ArgOr(args[3], 1)};
697 if (!array || !shiftExpr || !dim) {
698 return Expr<T>{std::move(funcRef)};
701 auto convertedShift{Fold(context_,
703 const auto *shift{UnwrapConstantValue<SubscriptInteger>(convertedShift)};
705 return Expr<T>{std::move(funcRef)};
708 std::optional<Expr<SomeType>> convertedBoundary;
709 if (
const auto *boundaryExpr{UnwrapExpr<Expr<SomeType>>(args[2])}) {
710 convertedBoundary = Fold(context_,
712 boundary = UnwrapExpr<Constant<T>>(convertedBoundary);
714 return Expr<T>{std::move(funcRef)};
718 if (*dim < 1 || *dim > array->Rank()) {
719 context_.messages().Say(
720 "Invalid 'dim=' argument (%jd) in EOSHIFT"_err_en_US,
721 static_cast<std::intmax_t
>(*dim));
722 }
else if (shift->Rank() > 0 && shift->Rank() != array->Rank() - 1) {
724 }
else if (boundary && boundary->Rank() > 0 &&
725 boundary->Rank() != array->Rank() - 1) {
728 int rank{array->Rank()};
729 int zbDim{
static_cast<int>(*dim) - 1};
731 if (shift->Rank() > 0) {
733 for (
int j{0}; j < rank; ++j) {
735 if (array->shape()[j] != shift->shape()[k]) {
736 context_.messages().Say(
737 "Invalid 'shift=' argument in EOSHIFT: extent on dimension %d is %jd but must be %jd"_err_en_US,
738 k + 1,
static_cast<std::intmax_t
>(shift->shape()[k]),
739 static_cast<std::intmax_t
>(array->shape()[j]));
746 if (boundary && boundary->Rank() > 0) {
748 for (
int j{0}; j < rank; ++j) {
750 if (array->shape()[j] != boundary->shape()[k]) {
751 context_.messages().Say(
752 "Invalid 'boundary=' argument in EOSHIFT: extent on dimension %d is %jd but must be %jd"_err_en_US,
753 k + 1,
static_cast<std::intmax_t
>(boundary->shape()[k]),
754 static_cast<std::intmax_t
>(array->shape()[j]));
762 std::vector<Scalar<T>> resultElements;
763 ConstantSubscripts arrayLB{array->lbounds()};
764 ConstantSubscripts arrayAt{arrayLB};
765 ConstantSubscript &dimIndex{arrayAt[zbDim]};
766 ConstantSubscript dimLB{dimIndex};
767 ConstantSubscript dimExtent{array->shape()[zbDim]};
768 ConstantSubscripts shiftLB{shift->lbounds()};
769 ConstantSubscripts boundaryLB;
771 boundaryLB = boundary->lbounds();
773 for (
auto n{GetSize(array->shape())}; n > 0; --n) {
774 ConstantSubscript origDimIndex{dimIndex};
775 ConstantSubscripts shiftAt;
776 if (shift->Rank() > 0) {
778 for (
int j{0}; j < rank; ++j) {
780 shiftAt.emplace_back(shiftLB[k++] + arrayAt[j] - arrayLB[j]);
784 ConstantSubscript shiftCount{shift->At(shiftAt).ToInt64()};
785 dimIndex += shiftCount;
786 if (dimIndex >= dimLB && dimIndex < dimLB + dimExtent) {
787 resultElements.push_back(array->At(arrayAt));
788 }
else if (boundary) {
789 ConstantSubscripts boundaryAt;
790 if (boundary->Rank() > 0) {
791 for (
int j{0}; j < rank; ++j) {
794 boundaryAt.emplace_back(
795 boundaryLB[k++] + arrayAt[j] - arrayLB[j]);
799 resultElements.push_back(boundary->At(boundaryAt));
800 }
else if constexpr (T::category == TypeCategory::Integer ||
801 T::category == TypeCategory::Unsigned ||
802 T::category == TypeCategory::Real ||
803 T::category == TypeCategory::Complex ||
804 T::category == TypeCategory::Logical) {
805 resultElements.emplace_back();
806 }
else if constexpr (T::category == TypeCategory::Character) {
807 auto len{
static_cast<std::size_t
>(array->LEN())};
808 typename Scalar<T>::value_type space{
' '};
809 resultElements.emplace_back(len, space);
811 DIE(
"no derived type boundary");
813 dimIndex = origDimIndex;
814 array->IncrementSubscripts(arrayAt);
816 return Expr<T>{PackageConstant<T>(
817 std::move(resultElements), *array, array->shape())};
821 return MakeInvalidIntrinsic(std::move(funcRef));
825 return FoldElementalIntrinsic<T, T, T, LogicalResult>(context_,
827 ScalarFunc<T, T, T, LogicalResult>(
828 [](
const Scalar<T> &ifTrue,
const Scalar<T> &ifFalse,
829 const Scalar<LogicalResult> &predicate) -> Scalar<T> {
830 return predicate.IsTrue() ? ifTrue : ifFalse;
835 auto args{funcRef.arguments()};
836 CHECK(args.size() == 3);
837 const auto *array{UnwrapConstantValue<T>(args[0])};
838 const auto *vector{UnwrapConstantValue<T>(args[2])};
839 auto convertedMask{Fold(context_,
840 ConvertToType<LogicalResult>(
842 const auto *mask{UnwrapConstantValue<LogicalResult>(convertedMask)};
843 if (!array || !mask || (args[2] && !vector)) {
844 return Expr<T>{std::move(funcRef)};
847 ConstantSubscript arrayElements{GetSize(array->shape())};
848 ConstantSubscript truths{0};
849 ConstantSubscripts maskAt{mask->lbounds()};
850 if (mask->Rank() == 0) {
851 if (mask->At(maskAt).IsTrue()) {
852 truths = arrayElements;
854 }
else if (array->shape() != mask->shape()) {
856 return MakeInvalidIntrinsic(std::move(funcRef));
858 for (ConstantSubscript j{0}; j < arrayElements;
859 ++j, mask->IncrementSubscripts(maskAt)) {
860 if (mask->At(maskAt).IsTrue()) {
865 std::vector<Scalar<T>> resultElements;
866 ConstantSubscripts arrayAt{array->lbounds()};
867 ConstantSubscript resultSize{truths};
869 resultSize = vector->shape().at(0);
870 if (resultSize < truths) {
871 context_.messages().Say(
872 "Invalid 'vector=' argument in PACK: the 'mask=' argument has %jd true elements, but the vector has only %jd elements"_err_en_US,
873 static_cast<std::intmax_t
>(truths),
874 static_cast<std::intmax_t
>(resultSize));
875 return MakeInvalidIntrinsic(std::move(funcRef));
878 for (ConstantSubscript j{0}; j < truths;) {
879 if (mask->At(maskAt).IsTrue()) {
880 resultElements.push_back(array->At(arrayAt));
883 array->IncrementSubscripts(arrayAt);
884 mask->IncrementSubscripts(maskAt);
887 ConstantSubscripts vectorAt{vector->lbounds()};
888 vectorAt.at(0) += truths;
889 for (ConstantSubscript j{truths}; j < resultSize; ++j) {
890 resultElements.push_back(vector->At(vectorAt));
894 return Expr<T>{PackageConstant<T>(std::move(resultElements), *array,
895 ConstantSubscripts{
static_cast<ConstantSubscript
>(resultSize)})};
899 auto args{funcRef.arguments()};
900 CHECK(args.size() == 4);
901 const auto *source{UnwrapConstantValue<T>(args[0])};
902 const auto *pad{UnwrapConstantValue<T>(args[2])};
903 std::optional<std::vector<ConstantSubscript>> shape{
904 GetIntegerVector<ConstantSubscript>(args[1])};
905 std::optional<std::vector<int>> order{GetIntegerVector<int>(args[3])};
906 std::optional<uint64_t> optResultElement;
907 std::optional<std::vector<int>> dimOrder;
910 if (shape->size() > common::maxRank) {
911 context_.messages().Say(
912 "Size of 'shape=' argument (%zd) must not be greater than %d"_err_en_US,
913 shape->size(), common::maxRank);
915 }
else if (HasNegativeExtent(*shape)) {
916 context_.messages().Say(
917 "'shape=' argument (%s) must not have a negative extent"_err_en_US,
918 DEREF(args[1]->UnwrapExpr()).AsFortran());
921 optResultElement = TotalElementCount(*shape);
922 if (!optResultElement) {
923 context_.messages().Say(
924 "'shape=' argument (%s) specifies an array with too many elements"_err_en_US,
925 DEREF(args[1]->UnwrapExpr()).AsFortran());
930 dimOrder = ValidateDimensionOrder(GetRank(*shape), *order);
932 context_.messages().Say(
933 "Invalid 'order=' argument (%s) in RESHAPE"_err_en_US,
934 DEREF(args[3]->UnwrapExpr()).AsFortran());
941 }
else if (!source || !shape || (args[2] && !pad) || (args[3] && !order)) {
942 return Expr<T>{std::move(funcRef)};
944 uint64_t resultElements{*optResultElement};
945 std::vector<int> *dimOrderPtr{dimOrder ? &dimOrder.value() :
nullptr};
946 if (resultElements > source->size() && (!pad || pad->empty())) {
947 context_.messages().Say(
948 "Too few elements in 'source=' argument and 'pad=' "
949 "argument is not present or has null size"_err_en_US);
953 ? source->Reshape(std::move(shape.value()))
954 : pad->Reshape(std::move(shape.value()))};
955 ConstantSubscripts subscripts{result.lbounds()};
956 auto copied{result.CopyFrom(*source,
957 std::min(
static_cast<uint64_t
>(source->size()), resultElements),
958 subscripts, dimOrderPtr)};
959 if (copied < resultElements) {
961 copied += result.CopyFrom(
962 *pad, resultElements - copied, subscripts, dimOrderPtr);
964 CHECK(copied == resultElements);
965 return Expr<T>{std::move(result)};
969 return MakeInvalidIntrinsic(std::move(funcRef));
973 auto args{funcRef.arguments()};
974 CHECK(args.size() == 3);
975 const Constant<T> *source{UnwrapConstantValue<T>(args[0])};
976 auto dim{ToInt64(args[1])};
977 auto ncopies{ToInt64(args[2])};
978 if (!source || !dim) {
979 return Expr<T>{std::move(funcRef)};
981 int sourceRank{source->Rank()};
982 if (sourceRank >= common::maxRank) {
983 context_.messages().Say(
984 "SOURCE= argument to SPREAD has rank %d but must have rank less than %d"_err_en_US,
985 sourceRank, common::maxRank);
986 }
else if (*dim < 1 || *dim > sourceRank + 1) {
987 context_.messages().Say(
988 "DIM=%d argument to SPREAD must be between 1 and %d"_err_en_US, *dim,
990 }
else if (!ncopies) {
991 return Expr<T>{std::move(funcRef)};
1001 ConstantSubscripts shape{source->shape()};
1002 shape.insert(shape.begin() + *dim - 1, *ncopies);
1003 Constant<T> spread{source->Reshape(std::move(shape))};
1004 std::optional<uint64_t> n{TotalElementCount(spread.shape())};
1006 context_.messages().Say(
"Too many elements in SPREAD result"_err_en_US);
1008 std::vector<int> dimOrder;
1009 for (
int j{0}; j < sourceRank; ++j) {
1010 dimOrder.push_back(j < *dim - 1 ? j : j + 1);
1012 dimOrder.push_back(*dim - 1);
1013 ConstantSubscripts at{spread.lbounds()};
1014 spread.CopyFrom(*source, *n, at, &dimOrder);
1015 return Expr<T>{std::move(spread)};
1019 return MakeInvalidIntrinsic(std::move(funcRef));
1023 auto args{funcRef.arguments()};
1024 CHECK(args.size() == 1);
1025 const auto *matrix{UnwrapConstantValue<T>(args[0])};
1027 return Expr<T>{std::move(funcRef)};
1030 std::vector<Scalar<T>> resultElements;
1031 ConstantSubscripts at(2);
1032 for (ConstantSubscript j{0}; j < matrix->shape()[0]; ++j) {
1033 at[0] = matrix->lbounds()[0] + j;
1034 for (ConstantSubscript k{0}; k < matrix->shape()[1]; ++k) {
1035 at[1] = matrix->lbounds()[1] + k;
1036 resultElements.push_back(matrix->At(at));
1039 at = matrix->shape();
1040 std::swap(at[0], at[1]);
1041 return Expr<T>{PackageConstant<T>(std::move(resultElements), *matrix, at)};
1045 auto args{funcRef.arguments()};
1046 CHECK(args.size() == 3);
1047 const auto *vector{UnwrapConstantValue<T>(args[0])};
1048 auto convertedMask{Fold(context_,
1049 ConvertToType<LogicalResult>(
1051 const auto *mask{UnwrapConstantValue<LogicalResult>(convertedMask)};
1052 const auto *field{UnwrapConstantValue<T>(args[2])};
1053 if (!vector || !mask || !field) {
1054 return Expr<T>{std::move(funcRef)};
1057 if (field->Rank() > 0 && field->shape() != mask->shape()) {
1059 return MakeInvalidIntrinsic(std::move(funcRef));
1061 ConstantSubscript maskElements{GetSize(mask->shape())};
1062 ConstantSubscript truths{0};
1063 ConstantSubscripts maskAt{mask->lbounds()};
1064 for (ConstantSubscript j{0}; j < maskElements;
1065 ++j, mask->IncrementSubscripts(maskAt)) {
1066 if (mask->At(maskAt).IsTrue()) {
1070 if (truths > GetSize(vector->shape())) {
1071 context_.messages().Say(
1072 "Invalid 'vector=' argument in UNPACK: the 'mask=' argument has %jd true elements, but the vector has only %jd elements"_err_en_US,
1073 static_cast<std::intmax_t
>(truths),
1074 static_cast<std::intmax_t
>(GetSize(vector->shape())));
1075 return MakeInvalidIntrinsic(std::move(funcRef));
1077 std::vector<Scalar<T>> resultElements;
1078 ConstantSubscripts vectorAt{vector->lbounds()};
1079 ConstantSubscripts fieldAt{field->lbounds()};
1080 for (ConstantSubscript j{0}; j < maskElements; ++j) {
1081 if (mask->At(maskAt).IsTrue()) {
1082 resultElements.push_back(vector->At(vectorAt));
1083 vector->IncrementSubscripts(vectorAt);
1085 resultElements.push_back(field->At(fieldAt));
1087 mask->IncrementSubscripts(maskAt);
1088 field->IncrementSubscripts(fieldAt);
1091 PackageConstant<T>(std::move(resultElements), *vector, mask->shape())};
1094std::optional<Expr<SomeType>> FoldTransfer(
1098 if (
auto folded{FoldTransfer(context_, funcRef.arguments())}) {
1099 return DEREF(UnwrapExpr<
Expr<T>>(*folded));
1101 return Expr<T>{std::move(funcRef)};
1107template <
typename T>
1110 static_assert(T::category == TypeCategory::Integer ||
1111 T::category == TypeCategory::Unsigned ||
1112 T::category == TypeCategory::Real ||
1113 T::category == TypeCategory::Character);
1136 auto &args{funcRef.arguments()};
1137 std::size_t nargs{args.size()};
1138 bool allArgsConstant{
true};
1139 bool extremumAnyway{nargs == 2 && T::category != TypeCategory::Character};
1143 if (!folder.Folding(args[0])) {
1144 allArgsConstant =
false;
1146 if (!folder.Folding(args[1])) {
1147 allArgsConstant =
false;
1153 for (std::size_t i{2}; i < nargs; ++i) {
1155 if (!folder.Folding(args[i])) {
1156 allArgsConstant =
false;
1164 if (allArgsConstant || extremumAnyway) {
1167 if (
const auto *resultp{UnwrapExpr<Expr<T>>(args[0])}) {
1169 for (std::size_t i{1}; i < nargs; ++i) {
1170 if (
const auto *tExpr{UnwrapExpr<Expr<T>>(args[i])}) {
1171 result = FoldOperation(
1172 context,
Extremum<T>{order, std::move(result), *tExpr});
1175 return Expr<T>{std::move(funcRef)};
1183 return Expr<T>{std::move(funcRef)};
1193template <
typename T>
1194Expr<T> RewriteSpecificMINorMAX(
1196 ActualArguments &args{funcRef.arguments()};
1197 auto &intrinsic{DEREF(std::get_if<SpecificIntrinsic>(&funcRef.proc().u))};
1200 std::optional<DynamicType> resultType;
1202 for (
auto j{args.size()}; j-- > 0;) {
1209 (type.category() == resultType->category() &&
1210 type.kind() > resultType->kind()) ||
1211 resultType->category() == TypeCategory::Integer) {
1213 resultTypeArg = &*args[j];
1218 return Expr<T>{std::move(funcRef)};
1221 intrinsic.name.find(
"max") != std::string::npos ?
"max"s :
"min"s;
1222 intrinsic.characteristics.value().functionResult.value().SetType(*resultType);
1223 auto insertConversion{[&](
const auto &x) ->
Expr<T> {
1224 using TR = ResultType<
decltype(x)>;
1227 return Fold(context, ConvertToType<T>(AsCategoryExpr(std::move(maxRef))));
1229 if (
auto *sx{UnwrapExpr<Expr<SomeReal>>(*resultTypeArg)}) {
1230 return common::visit(insertConversion, sx->u);
1231 }
else if (
auto *sx{UnwrapExpr<Expr<SomeInteger>>(*resultTypeArg)}) {
1232 return common::visit(insertConversion, sx->u);
1234 return Expr<T>{std::move(funcRef)};
1256template <
typename T>
1258 ActualArguments &args{funcRef.arguments()};
1259 const auto *intrinsic{std::get_if<SpecificIntrinsic>(&funcRef.proc().u)};
1260 if (!intrinsic || intrinsic->name !=
"kind") {
1263 for (std::optional<ActualArgument> &arg : args) {
1264 if (
auto *expr{UnwrapExpr<Expr<SomeType>>(arg)}) {
1265 *expr = Fold(context, std::move(*expr));
1270 const std::string name{intrinsic->name};
1271 if (name ==
"cshift") {
1272 return Folder<T>{context}.CSHIFT(std::move(funcRef));
1273 }
else if (name ==
"eoshift") {
1274 return Folder<T>{context}.EOSHIFT(std::move(funcRef));
1275 }
else if (name ==
"merge") {
1276 return Folder<T>{context}.MERGE(std::move(funcRef));
1277 }
else if (name ==
"pack") {
1278 return Folder<T>{context}.PACK(std::move(funcRef));
1279 }
else if (name ==
"reshape") {
1280 return Folder<T>{context}.RESHAPE(std::move(funcRef));
1281 }
else if (name ==
"spread") {
1282 return Folder<T>{context}.SPREAD(std::move(funcRef));
1283 }
else if (name ==
"transfer") {
1284 return Folder<T>{context}.TRANSFER(std::move(funcRef));
1285 }
else if (name ==
"transpose") {
1286 return Folder<T>{context}.TRANSPOSE(std::move(funcRef));
1287 }
else if (name ==
"unpack") {
1288 return Folder<T>{context}.UNPACK(std::move(funcRef));
1291 if constexpr (!std::is_same_v<T, SomeDerived>) {
1292 return FoldIntrinsicFunction(context, std::move(funcRef));
1295 return Expr<T>{std::move(funcRef)};
1299template <
typename T>
class ArrayConstructorFolder {
1301 explicit ArrayConstructorFolder(
FoldingContext &c) : context_{c} {}
1304 if constexpr (T::category == TypeCategory::Character) {
1305 if (
const auto *len{array.LEN()}) {
1306 charLength_ = ToInt64(Fold(context_, common::Clone(*len)));
1307 knownCharLength_ = charLength_.has_value();
1311 if (FoldArray(array)) {
1312 auto n{
static_cast<ConstantSubscript
>(elements_.size())};
1313 if constexpr (std::is_same_v<T, SomeDerived>) {
1315 std::move(elements_), ConstantSubscripts{n}}};
1316 }
else if constexpr (T::category == TypeCategory::Character) {
1319 *charLength_, std::move(elements_), ConstantSubscripts{n}}};
1323 std::move(elements_), ConstantSubscripts{n}, resultInfo_}};
1326 return Expr<T>{std::move(array)};
1330 bool FoldArray(
const Expr<T> &expr) {
1331 Expr<T> folded{Fold(context_, common::Clone(expr))};
1332 if (
const auto *c{UnwrapConstantValue<T>(folded)}) {
1335 ConstantSubscripts index{c->lbounds()};
1337 elements_.emplace_back(c->At(index));
1338 }
while (c->IncrementSubscripts(index));
1340 if constexpr (T::category == TypeCategory::Character) {
1341 if (!knownCharLength_) {
1342 charLength_ = std::max(c->LEN(), charLength_.value_or(-1));
1344 }
else if constexpr (T::category == TypeCategory::Real ||
1345 T::category == TypeCategory::Complex) {
1346 if (c->result().isFromInexactLiteralConversion()) {
1347 resultInfo_.set_isFromInexactLiteralConversion();
1355 bool FoldArray(
const common::CopyableIndirection<
Expr<T>> &expr) {
1356 return FoldArray(expr.value());
1365 std::optional<ConstantSubscript> start{ToInt64(
lower)}, end{ToInt64(upper)},
1366 step{ToInt64(stride)};
1367 if (start && end && step && *step != 0) {
1369 ConstantSubscript &j{context_.StartImpliedDo(iDo.name(), *start)};
1371 for (; j <= *end; j += *step) {
1372 result &= FoldArray(iDo.values());
1375 for (; j >= *end; j += *step) {
1376 result &= FoldArray(iDo.values());
1379 context_.EndImpliedDo(iDo.name());
1386 return common::visit([&](
const auto &y) {
return FoldArray(y); }, x.u);
1389 for (
const auto &x : xs) {
1390 if (!FoldArray(x)) {
1398 std::vector<Scalar<T>> elements_;
1399 std::optional<ConstantSubscript> charLength_;
1400 bool knownCharLength_{
false};
1401 typename Constant<T>::Result resultInfo_;
1404template <
typename T>
1417template <
typename T>
1418bool ArrayConstructorIsFlat(
const ArrayConstructorValues<T> &values) {
1419 for (
const ArrayConstructorValue<T> &x : values) {
1420 if (!std::holds_alternative<Expr<T>>(x.u)) {
1427template <
typename T>
1428std::optional<Expr<T>> AsFlatArrayConstructor(
const Expr<T> &expr) {
1429 if (
const auto *c{UnwrapConstantValue<T>(expr)}) {
1432 ConstantSubscripts at{c->lbounds()};
1435 }
while (c->IncrementSubscripts(at));
1437 return std::make_optional<Expr<T>>(std::move(result));
1438 }
else if (
const auto *a{UnwrapExpr<ArrayConstructor<T>>(expr)}) {
1439 if (ArrayConstructorIsFlat(*a)) {
1440 return std::make_optional<Expr<T>>(expr);
1442 }
else if (
const auto *p{UnwrapExpr<Parentheses<T>>(expr)}) {
1443 return AsFlatArrayConstructor(
Expr<T>{p->left()});
1445 return std::nullopt;
1448template <TypeCategory CAT>
1449std::enable_if_t<CAT != TypeCategory::Derived,
1450 std::optional<Expr<SomeKind<CAT>>>>
1452 return common::visit(
1454 if (
auto flattened{AsFlatArrayConstructor(kindExpr)}) {
1457 return std::nullopt;
1467template <
typename T>
1468std::optional<Expr<T>> FromArrayConstructor(
1470 if (
auto constShape{AsConstantExtents(context, shape)};
1471 constShape && !HasNegativeExtent(*constShape)) {
1473 if (
auto *constant{UnwrapConstantValue<T>(result)}) {
1475 return Expr<T>{constant->Reshape(std::move(*constShape))};
1477 if (constShape->size() == 1) {
1478 if (
auto elements{GetShape(context, result)}) {
1479 if (
auto constElements{AsConstantExtents(context, *elements)}) {
1480 if (constElements->size() == 1 &&
1481 constElements->at(0) == constShape->at(0)) {
1484 return std::move(result);
1490 return std::nullopt;
1501template <
typename RESULT,
typename OPERAND>
1507 if constexpr (common::HasMember<OPERAND, AllIntrinsicCategoryTypes>) {
1509 [&](
auto &&kindExpr) {
1510 using kindType = ResultType<
decltype(kindExpr)>;
1511 auto &aConst{std::get<ArrayConstructor<kindType>>(kindExpr.u)};
1512 for (
auto &acValue : aConst) {
1513 auto &scalar{std::get<Expr<kindType>>(acValue.u)};
1514 result.Push(Fold(context, f(
Expr<OPERAND>{std::move(scalar)})));
1517 std::move(values.u));
1519 auto &aConst{std::get<ArrayConstructor<OPERAND>>(values.u)};
1520 for (
auto &acValue : aConst) {
1521 auto &scalar{std::get<Expr<OPERAND>>(acValue.u)};
1522 result.Push(Fold(context, f(std::move(scalar))));
1525 if constexpr (RESULT::category == TypeCategory::Character) {
1527 result.set_LEN(std::move(*length));
1530 return FromArrayConstructor(context, std::move(result), shape);
1533template <
typename RESULT,
typename A>
1537 if constexpr (RESULT::category == TypeCategory::Character) {
1539 result.set_LEN(std::move(*length));
1545template <
typename LEFT,
typename RIGHT>
1549 auto rightIter{rightArrConst.begin()};
1550 for (
auto &leftValue : leftArrConst) {
1551 CHECK(rightIter != rightArrConst.end());
1552 auto &leftExpr{std::get<Expr<LEFT>>(leftValue.u)};
1553 auto &rightExpr{std::get<Expr<RIGHT>>(rightIter->u)};
1554 if (leftExpr.Rank() != rightExpr.Rank()) {
1557 std::optional<Shape> leftShape{GetShape(context, leftExpr)};
1558 std::optional<Shape> rightShape{GetShape(context, rightExpr)};
1559 if (!leftShape || !rightShape || *leftShape != *rightShape) {
1568template <
typename RESULT,
typename LEFT,
typename RIGHT>
1573 -> std::optional<Expr<RESULT>> {
1574 auto result{ArrayConstructorFromMold<RESULT>(leftValues, std::move(length))};
1575 auto &leftArrConst{std::get<ArrayConstructor<LEFT>>(leftValues.u)};
1576 if constexpr (common::HasMember<RIGHT, AllIntrinsicCategoryTypes>) {
1577 bool mapped{common::visit(
1578 [&](
auto &&kindExpr) ->
bool {
1579 using kindType = ResultType<
decltype(kindExpr)>;
1581 auto &rightArrConst{std::get<ArrayConstructor<kindType>>(kindExpr.u)};
1582 if (!ShapesMatch(context, leftArrConst, rightArrConst)) {
1585 auto rightIter{rightArrConst.begin()};
1586 for (
auto &leftValue : leftArrConst) {
1587 CHECK(rightIter != rightArrConst.end());
1588 auto &leftScalar{std::get<Expr<LEFT>>(leftValue.u)};
1589 auto &rightScalar{std::get<Expr<kindType>>(rightIter->u)};
1590 result.Push(Fold(context,
1591 f(std::move(leftScalar),
Expr<RIGHT>{std::move(rightScalar)})));
1596 std::move(rightValues.u))};
1598 return std::nullopt;
1601 auto &rightArrConst{std::get<ArrayConstructor<RIGHT>>(rightValues.u)};
1602 if (!ShapesMatch(context, leftArrConst, rightArrConst)) {
1603 return std::nullopt;
1605 auto rightIter{rightArrConst.begin()};
1606 for (
auto &leftValue : leftArrConst) {
1607 CHECK(rightIter != rightArrConst.end());
1608 auto &leftScalar{std::get<Expr<LEFT>>(leftValue.u)};
1609 auto &rightScalar{std::get<Expr<RIGHT>>(rightIter->u)};
1611 Fold(context, f(std::move(leftScalar), std::move(rightScalar))));
1615 return FromArrayConstructor(context, std::move(result), shape);
1619template <
typename RESULT,
typename LEFT,
typename RIGHT>
1624 -> std::optional<Expr<RESULT>> {
1625 auto result{ArrayConstructorFromMold<RESULT>(leftValues, std::move(length))};
1626 auto &leftArrConst{std::get<ArrayConstructor<LEFT>>(leftValues.u)};
1627 for (
auto &leftValue : leftArrConst) {
1628 auto &leftScalar{std::get<Expr<LEFT>>(leftValue.u)};
1630 Fold(context, f(std::move(leftScalar),
Expr<RIGHT>{rightScalar})));
1632 return FromArrayConstructor(context, std::move(result), shape);
1636template <
typename RESULT,
typename LEFT,
typename RIGHT>
1641 -> std::optional<Expr<RESULT>> {
1642 auto result{ArrayConstructorFromMold<RESULT>(leftScalar, std::move(length))};
1643 if constexpr (common::HasMember<RIGHT, AllIntrinsicCategoryTypes>) {
1645 [&](
auto &&kindExpr) {
1646 using kindType = ResultType<
decltype(kindExpr)>;
1647 auto &rightArrConst{std::get<ArrayConstructor<kindType>>(kindExpr.u)};
1648 for (
auto &rightValue : rightArrConst) {
1649 auto &rightScalar{std::get<Expr<kindType>>(rightValue.u)};
1650 result.Push(Fold(context,
1655 std::move(rightValues.u));
1657 auto &rightArrConst{std::get<ArrayConstructor<RIGHT>>(rightValues.u)};
1658 for (
auto &rightValue : rightArrConst) {
1659 auto &rightScalar{std::get<Expr<RIGHT>>(rightValue.u)};
1661 Fold(context, f(
Expr<LEFT>{leftScalar}, std::move(rightScalar))));
1664 return FromArrayConstructor(context, std::move(result), shape);
1667template <
typename DERIVED,
typename RESULT,
typename... OPD>
1668std::optional<Expr<SubscriptInteger>> ComputeResultLength(
1670 if constexpr (RESULT::category == TypeCategory::Character) {
1673 return std::nullopt;
1680template <
typename DERIVED,
typename RESULT,
typename OPERAND>
1684 -> std::optional<Expr<RESULT>> {
1685 auto &expr{operation.left()};
1686 expr = Fold(context, std::move(expr));
1687 if (expr.Rank() > 0) {
1688 if (std::optional<Shape> shape{GetShape(context, expr)}) {
1689 if (
auto values{AsFlatArrayConstructor(expr)}) {
1690 return MapOperation(context, std::move(f), *shape,
1691 ComputeResultLength(operation), std::move(*values));
1695 return std::nullopt;
1698template <
typename DERIVED,
typename RESULT,
typename OPERAND>
1699auto ApplyElementwise(
1701 -> std::optional<Expr<RESULT>> {
1702 return ApplyElementwise(context, operation,
1709template <
typename DERIVED,
typename RESULT,
typename LEFT,
typename RIGHT>
1713 -> std::optional<Expr<RESULT>> {
1714 auto resultLength{ComputeResultLength(operation)};
1715 auto &leftExpr{operation.left()};
1716 auto &rightExpr{operation.right()};
1717 if (leftExpr.Rank() != rightExpr.Rank() && leftExpr.Rank() != 0 &&
1718 rightExpr.Rank() != 0) {
1719 return std::nullopt;
1721 leftExpr = Fold(context, std::move(leftExpr));
1722 rightExpr = Fold(context, std::move(rightExpr));
1723 if (leftExpr.Rank() > 0) {
1724 if (std::optional<Shape> leftShape{GetShape(context, leftExpr)}) {
1725 if (
auto left{AsFlatArrayConstructor(leftExpr)}) {
1726 if (rightExpr.Rank() > 0) {
1727 if (std::optional<Shape> rightShape{GetShape(context, rightExpr)}) {
1728 if (
auto right{AsFlatArrayConstructor(rightExpr)}) {
1729 if (CheckConformance(context.messages(), *leftShape, *rightShape,
1730 CheckConformanceFlags::EitherScalarExpandable)
1731 .value_or(
false )) {
1732 return MapOperation(context, std::move(f), *leftShape,
1733 std::move(resultLength), std::move(*left),
1736 return std::nullopt;
1738 return MapOperation(context, std::move(f), *leftShape,
1739 std::move(resultLength), std::move(*left), std::move(*right));
1742 }
else if (IsExpandableScalar(rightExpr, context, *leftShape)) {
1743 return MapOperation(context, std::move(f), *leftShape,
1744 std::move(resultLength), std::move(*left), rightExpr);
1748 }
else if (rightExpr.Rank() > 0) {
1749 if (std::optional<Shape> rightShape{GetShape(context, rightExpr)}) {
1750 if (IsExpandableScalar(leftExpr, context, *rightShape)) {
1751 if (
auto right{AsFlatArrayConstructor(rightExpr)}) {
1752 return MapOperation(context, std::move(f), *rightShape,
1753 std::move(resultLength), leftExpr, std::move(*right));
1758 return std::nullopt;
1761template <
typename DERIVED,
typename RESULT,
typename LEFT,
typename RIGHT>
1762auto ApplyElementwise(
1764 -> std::optional<Expr<RESULT>> {
1765 return ApplyElementwise(context, operation,
1768 return Expr<RESULT>{DERIVED{std::move(left), std::move(right)}};
1774template <
typename TO,
typename FROM>
1775common::IfNoLvalue<std::optional<TO>, FROM> ConvertString(FROM &&s) {
1776 if constexpr (std::is_same_v<TO, FROM>) {
1777 return std::make_optional<TO>(std::move(s));
1782 for (
auto iter{s.cbegin()}; iter != s.cend(); ++iter) {
1783 if (
static_cast<std::uint64_t
>(*iter) > 127) {
1784 return std::nullopt;
1786 str.push_back(
static_cast<typename TO::value_type
>(*iter));
1788 return std::make_optional<TO>(std::move(str));
1792template <
typename TO, TypeCategory FROMCAT>
1795 if (
auto array{ApplyElementwise(context, convert)}) {
1801 } msvcWorkaround{context, convert};
1802 return common::visit(
1803 [&msvcWorkaround](
auto &kindExpr) ->
Expr<TO> {
1804 using Operand = ResultType<
decltype(kindExpr)>;
1807 TypeCategory
constexpr FromCat{FROMCAT};
1808 static_assert(FromCat == Operand::category);
1809 auto &convert{msvcWorkaround.convert};
1810 if (
auto value{GetScalarConstantValue<Operand>(kindExpr)}) {
1812 if constexpr (TO::category == TypeCategory::Integer) {
1813 if constexpr (FromCat == TypeCategory::Integer) {
1814 auto converted{Scalar<TO>::ConvertSigned(*value)};
1815 if (converted.overflow) {
1816 ctx.Warn(common::UsageWarning::FoldingException,
1817 "conversion of %s_%d to INTEGER(%d) overflowed; result is %s"_warn_en_US,
1818 value->SignedDecimal(), Operand::kind, TO::kind,
1819 converted.value.SignedDecimal());
1821 return ScalarConstantToExpr(std::move(converted.value));
1822 }
else if constexpr (FromCat == TypeCategory::Unsigned) {
1823 auto converted{Scalar<TO>::ConvertUnsigned(*value)};
1824 if ((converted.overflow || converted.value.IsNegative())) {
1825 ctx.Warn(common::UsageWarning::FoldingException,
1826 "conversion of %s_U%d to INTEGER(%d) overflowed; result is %s"_warn_en_US,
1827 value->UnsignedDecimal(), Operand::kind, TO::kind,
1828 converted.value.SignedDecimal());
1830 return ScalarConstantToExpr(std::move(converted.value));
1831 }
else if constexpr (FromCat == TypeCategory::Real) {
1832 auto converted{value->template ToInteger<Scalar<TO>>()};
1833 if (converted.flags.test(RealFlag::InvalidArgument)) {
1834 ctx.Warn(common::UsageWarning::FoldingException,
1835 "REAL(%d) to INTEGER(%d) conversion: invalid argument"_warn_en_US,
1836 Operand::kind, TO::kind);
1837 }
else if (converted.flags.test(RealFlag::Overflow)) {
1838 ctx.Warn(common::UsageWarning::FoldingException,
1839 "REAL(%d) to INTEGER(%d) conversion overflowed"_warn_en_US,
1840 Operand::kind, TO::kind);
1842 return ScalarConstantToExpr(std::move(converted.value));
1844 }
else if constexpr (TO::category == TypeCategory::Unsigned) {
1845 if constexpr (FromCat == TypeCategory::Integer ||
1846 FromCat == TypeCategory::Unsigned) {
1848 Constant<TO>{Scalar<TO>::ConvertUnsigned(*value).value}};
1849 }
else if constexpr (FromCat == TypeCategory::Real) {
1851 Constant<TO>{value->template ToInteger<Scalar<TO>>().value}};
1853 }
else if constexpr (TO::category == TypeCategory::Real) {
1854 if constexpr (FromCat == TypeCategory::Integer ||
1855 FromCat == TypeCategory::Unsigned) {
1856 auto converted{Scalar<TO>::FromInteger(
1857 *value, FromCat == TypeCategory::Unsigned)};
1858 if (!converted.flags.empty()) {
1860 std::snprintf(buffer,
sizeof buffer,
1861 "INTEGER(%d) to REAL(%d) conversion", Operand::kind,
1863 ctx.RealFlagWarnings(converted.flags, buffer);
1865 return ScalarConstantToExpr(std::move(converted.value));
1866 }
else if constexpr (FromCat == TypeCategory::Real) {
1867 auto converted{Scalar<TO>::Convert(*value)};
1869 if (!converted.flags.empty()) {
1870 std::snprintf(buffer,
sizeof buffer,
1871 "REAL(%d) to REAL(%d) conversion", Operand::kind, TO::kind);
1872 ctx.RealFlagWarnings(converted.flags, buffer);
1874 if (ctx.targetCharacteristics().areSubnormalsFlushedToZero()) {
1875 converted.value = converted.value.FlushSubnormalToZero();
1877 return ScalarConstantToExpr(std::move(converted.value));
1879 }
else if constexpr (TO::category == TypeCategory::Complex) {
1880 if constexpr (FromCat == TypeCategory::Complex) {
1881 return FoldOperation(ctx,
1888 }
else if constexpr (TO::category == TypeCategory::Character &&
1889 FromCat == TypeCategory::Character) {
1890 if (
auto converted{ConvertString<Scalar<TO>>(std::move(*value))}) {
1891 return ScalarConstantToExpr(std::move(*converted));
1893 }
else if constexpr (TO::category == TypeCategory::Logical &&
1894 FromCat == TypeCategory::Logical) {
1897 }
else if constexpr (TO::category == FromCat &&
1898 FromCat != TypeCategory::Character) {
1900 if constexpr (std::is_same_v<Operand, TO>) {
1901 return std::move(kindExpr);
1902 }
else if constexpr (TO::category == TypeCategory::Logical ||
1903 TO::category == TypeCategory::Integer) {
1904 if (
auto *innerConv{
1905 std::get_if<Convert<Operand, TO::category>>(&kindExpr.u)}) {
1907 if (
auto *x{std::get_if<Expr<TO>>(&innerConv->left().u)}) {
1908 if constexpr (TO::category == TypeCategory::Logical ||
1909 TO::kind <= Operand::kind) {
1910 return std::move(*x);
1912 }
else if constexpr (std::is_same_v<TO,
1913 DescriptorInquiry::Result>) {
1914 if (std::holds_alternative<DescriptorInquiry>(x->u) ||
1915 std::holds_alternative<TypeParamInquiry>(x->u)) {
1917 return std::move(*x);
1924 return Expr<TO>{std::move(convert)};
1929template <
typename T>
1931 auto &operand{x.left()};
1932 operand = Fold(context, std::move(operand));
1933 if (
auto value{GetScalarConstantValue<T>(operand)}) {
1938 return std::move(operand);
1944template <
typename T>
1946 if (
auto array{ApplyElementwise(context, x)}) {
1949 auto &operand{x.left()};
1950 if (
auto *nn{std::get_if<Negate<T>>(&x.left().u)}) {
1952 if (IsVariable(nn->left())) {
1953 return FoldOperation(context,
Parentheses<T>{std::move(nn->left())});
1955 return std::move(nn->left());
1957 }
else if (
auto value{GetScalarConstantValue<T>(operand)}) {
1958 if constexpr (T::category == TypeCategory::Integer) {
1959 auto negated{value->Negate()};
1960 if (negated.overflow) {
1961 context.Warn(common::UsageWarning::FoldingException,
1962 "INTEGER(%d) negation overflowed"_warn_en_US, T::kind);
1965 }
else if constexpr (T::category == TypeCategory::Unsigned) {
1977template <
typename LEFT,
typename RIGHT>
1978std::optional<std::pair<Scalar<LEFT>, Scalar<RIGHT>>> OperandsAreConstants(
1980 if (
auto xvalue{GetScalarConstantValue<LEFT>(x)}) {
1981 if (
auto yvalue{GetScalarConstantValue<RIGHT>(y)}) {
1982 return {std::make_pair(*xvalue, *yvalue)};
1985 return std::nullopt;
1988template <
typename DERIVED,
typename RESULT,
typename LEFT,
typename RIGHT>
1989std::optional<std::pair<Scalar<LEFT>, Scalar<RIGHT>>> OperandsAreConstants(
1991 return OperandsAreConstants(operation.left(), operation.right());
1994template <
typename T>
1996 if (
auto array{ApplyElementwise(context, x)}) {
1999 if (
auto folded{OperandsAreConstants(x)}) {
2000 if constexpr (T::category == TypeCategory::Integer) {
2001 auto sum{folded->first.AddSigned(folded->second)};
2003 context.Warn(common::UsageWarning::FoldingException,
2004 "INTEGER(%d) addition overflowed"_warn_en_US, T::kind);
2007 }
else if constexpr (T::category == TypeCategory::Unsigned) {
2009 Constant<T>{folded->first.AddUnsigned(folded->second).value}};
2011 auto sum{folded->first.Add(
2012 folded->second, context.targetCharacteristics().roundingMode())};
2013 context.RealFlagWarnings(sum.flags,
"addition");
2014 if (context.targetCharacteristics().areSubnormalsFlushedToZero()) {
2015 sum.value = sum.value.FlushSubnormalToZero();
2023template <
typename T>
2025 if (
auto array{ApplyElementwise(context, x)}) {
2028 if (
auto folded{OperandsAreConstants(x)}) {
2029 if constexpr (T::category == TypeCategory::Integer) {
2030 auto difference{folded->first.SubtractSigned(folded->second)};
2031 if (difference.overflow) {
2032 context.Warn(common::UsageWarning::FoldingException,
2033 "INTEGER(%d) subtraction overflowed"_warn_en_US, T::kind);
2036 }
else if constexpr (T::category == TypeCategory::Unsigned) {
2038 Constant<T>{folded->first.SubtractSigned(folded->second).value}};
2040 auto difference{folded->first.Subtract(
2041 folded->second, context.targetCharacteristics().roundingMode())};
2042 context.RealFlagWarnings(difference.flags,
"subtraction");
2043 if (context.targetCharacteristics().areSubnormalsFlushedToZero()) {
2044 difference.value = difference.value.FlushSubnormalToZero();
2052template <
typename T>
2054 if (
auto array{ApplyElementwise(context, x)}) {
2057 if (
auto folded{OperandsAreConstants(x)}) {
2058 if constexpr (T::category == TypeCategory::Integer) {
2059 auto product{folded->first.MultiplySigned(folded->second)};
2060 if (product.SignedMultiplicationOverflowed()) {
2061 context.Warn(common::UsageWarning::FoldingException,
2062 "INTEGER(%d) multiplication overflowed"_warn_en_US, T::kind);
2065 }
else if constexpr (T::category == TypeCategory::Unsigned) {
2067 Constant<T>{folded->first.MultiplyUnsigned(folded->second).lower}};
2069 auto product{folded->first.Multiply(
2070 folded->second, context.targetCharacteristics().roundingMode())};
2071 context.RealFlagWarnings(product.flags,
"multiplication");
2072 if (context.targetCharacteristics().areSubnormalsFlushedToZero()) {
2073 product.value = product.value.FlushSubnormalToZero();
2077 }
else if constexpr (T::category == TypeCategory::Integer) {
2078 if (
auto c{GetScalarConstantValue<T>(x.right())}) {
2079 x.right() = std::move(x.left());
2080 x.left() =
Expr<T>{std::move(*c)};
2082 if (
auto c{GetScalarConstantValue<T>(x.left())}) {
2083 if (c->IsZero() && x.right().Rank() == 0) {
2084 return std::move(x.left());
2085 }
else if (c->CompareSigned(Scalar<T>{1}) == Ordering::Equal) {
2086 if (IsVariable(x.right())) {
2087 return FoldOperation(context,
Parentheses<T>{std::move(x.right())});
2089 return std::move(x.right());
2091 }
else if (c->CompareSigned(Scalar<T>{-1}) == Ordering::Equal) {
2092 return FoldOperation(context,
Negate<T>{std::move(x.right())});
2099template <
typename T>
2101 if (
auto array{ApplyElementwise(context, x)}) {
2104 if (
auto folded{OperandsAreConstants(x)}) {
2105 if constexpr (T::category == TypeCategory::Integer) {
2106 auto quotAndRem{folded->first.DivideSigned(folded->second)};
2107 if (quotAndRem.divisionByZero) {
2108 context.Warn(common::UsageWarning::FoldingException,
2109 "INTEGER(%d) division by zero"_warn_en_US, T::kind);
2112 if (quotAndRem.overflow) {
2113 context.Warn(common::UsageWarning::FoldingException,
2114 "INTEGER(%d) division overflowed"_warn_en_US, T::kind);
2117 }
else if constexpr (T::category == TypeCategory::Unsigned) {
2118 auto quotAndRem{folded->first.DivideUnsigned(folded->second)};
2119 if (quotAndRem.divisionByZero) {
2120 context.Warn(common::UsageWarning::FoldingException,
2121 "UNSIGNED(%d) division by zero"_warn_en_US, T::kind);
2126 auto quotient{folded->first.Divide(
2127 folded->second, context.targetCharacteristics().roundingMode())};
2131 bool isCanonicalNaNOrInf{
false};
2132 if constexpr (T::category == TypeCategory::Real) {
2133 if (folded->second.IsZero() && context.moduleFileName().has_value()) {
2134 using IntType =
typename T::Scalar::Word;
2135 auto intNumerator{folded->first.template ToInteger<IntType>()};
2136 isCanonicalNaNOrInf = intNumerator.flags == RealFlags{} &&
2137 intNumerator.value >= IntType{-1} &&
2138 intNumerator.value <= IntType{1};
2141 if (!isCanonicalNaNOrInf) {
2142 context.RealFlagWarnings(quotient.flags,
"division");
2144 if (context.targetCharacteristics().areSubnormalsFlushedToZero()) {
2145 quotient.value = quotient.value.FlushSubnormalToZero();
2153template <
typename T>
2155 if (
auto array{ApplyElementwise(context, x)}) {
2158 if (
auto folded{OperandsAreConstants(x)}) {
2159 if constexpr (T::category == TypeCategory::Integer) {
2160 auto power{folded->first.Power(folded->second)};
2161 if (power.divisionByZero) {
2162 context.Warn(common::UsageWarning::FoldingException,
2163 "INTEGER(%d) zero to negative power"_warn_en_US, T::kind);
2164 }
else if (power.overflow) {
2165 context.Warn(common::UsageWarning::FoldingException,
2166 "INTEGER(%d) power overflowed"_warn_en_US, T::kind);
2167 }
else if (power.zeroToZero) {
2168 context.Warn(common::UsageWarning::FoldingException,
2169 "INTEGER(%d) 0**0 is not defined"_warn_en_US, T::kind);
2173 if (folded->first.IsZero()) {
2174 if (folded->second.IsZero()) {
2175 context.Warn(common::UsageWarning::FoldingException,
2176 "REAL/COMPLEX 0**0 is not defined"_warn_en_US);
2180 }
else if (
auto callable{GetHostRuntimeWrapper<T, T, T>(
"pow")}) {
2182 Constant<T>{(*callable)(context, folded->first, folded->second)}};
2184 context.Warn(common::UsageWarning::FoldingFailure,
2185 "Power for %s cannot be folded on host"_warn_en_US,
2193template <
typename T>
2195 if (
auto array{ApplyElementwise(context, x)}) {
2198 return common::visit(
2200 if (
auto folded{OperandsAreConstants(x.left(), y)}) {
2201 auto power{evaluate::IntPower(folded->first, folded->second)};
2202 context.RealFlagWarnings(power.flags,
"power with INTEGER exponent");
2203 if (context.targetCharacteristics().areSubnormalsFlushedToZero()) {
2204 power.value = power.value.FlushSubnormalToZero();
2214template <
typename T>
2216 if (
auto array{ApplyElementwise(context, x,
2223 if (
auto folded{OperandsAreConstants(x)}) {
2224 if constexpr (T::category == TypeCategory::Integer) {
2225 if (folded->first.CompareSigned(folded->second) == x.ordering) {
2228 }
else if constexpr (T::category == TypeCategory::Unsigned) {
2229 if (folded->first.CompareUnsigned(folded->second) == x.ordering) {
2232 }
else if constexpr (T::category == TypeCategory::Real) {
2233 if (folded->first.IsNotANumber() ||
2234 (folded->first.Compare(folded->second) == Relation::Less) ==
2235 (x.ordering == Ordering::Less)) {
2239 static_assert(T::category == TypeCategory::Character);
2242 auto maxLen{std::max(folded->first.length(), folded->second.length())};
2243 bool isFirst{x.ordering == Compare(folded->first, folded->second)};
2244 auto res{isFirst ? std::move(folded->first) : std::move(folded->second)};
2245 res = res.length() == maxLen
2247 : CharacterUtils<T::kind>::Resize(res, maxLen);
2259 std::optional<Expr<Result>> result;
2262 using From = std::decay_t<
decltype(x)>;
2263 if constexpr (std::is_same_v<From, BOZLiteralConstant>) {
2266 result = ConvertToType<Result>(std::move(x));
2267 const auto *constant{UnwrapExpr<Constant<Result>>(*result)};
2269 Scalar<Result> real{constant->GetScalarValue().value()};
2270 From converted{From::ConvertUnsigned(real.RawBits()).value};
2271 if (original != converted) {
2272 context.Warn(common::UsageWarning::FoldingValueChecks,
2273 "Nonzero bits truncated from BOZ literal constant in REAL intrinsic"_warn_en_US);
2275 }
else if constexpr (IsNumericCategoryExpr<From>()) {
2276 result = Fold(context, ConvertToType<Result>(std::move(x)));
2278 common::die(
"ToReal: bad argument expression");
2282 return result.value();
2291 if (
auto array{ApplyElementwise(context, x,
2295 x.isImaginaryPart, std::move(operand)}};
2299 auto &operand{x.left()};
2300 if (
auto value{GetScalarConstantValue<Operand>(operand)}) {
2301 if (x.isImaginaryPart) {
2310template <
typename T>
2312 return common::visit(
2314 if constexpr (IsSpecificIntrinsicType<T>) {
2315 return FoldOperation(context, std::move(x));
2316 }
else if constexpr (std::is_same_v<T, SomeDerived>) {
2317 return FoldOperation(context, std::move(x));
2318 }
else if constexpr (common::HasMember<
decltype(x),
2319 TypelessExpression>) {
2320 return std::move(expr);
2322 return Expr<T>{Fold(context, std::move(x))};
Definition fold-implementation.h:1299
Definition expression.h:438
Definition expression.h:466
Definition variable.h:205
Definition variable.h:243
Definition variable.h:357
Definition constant.h:147
Definition variable.h:381
Definition expression.h:66
Definition fold-implementation.h:55
Definition expression.h:404
Definition variable.h:101
Definition expression.h:114
Definition expression.h:740
Definition variable.h:304
Definition variable.h:160
Definition variable.h:136
Definition ParserActions.h:24
Definition expression.h:296
Definition expression.h:432
Definition expression.h:257
Definition expression.h:357
Definition expression.h:211
Definition variable.h:288
Definition expression.h:317
Definition expression.h:340
Definition expression.h:396
Definition expression.h:310
Definition expression.h:247
Definition expression.h:229
Definition expression.h:324
Definition expression.h:332
Definition variable.h:191
Definition expression.h:303