2020-02-25 23:11:52 +08:00
|
|
|
//===-- lib/Evaluate/formatting.cpp ---------------------------------------===//
|
2019-03-28 06:27:33 +08:00
|
|
|
//
|
2019-12-21 04:52:07 +08:00
|
|
|
// 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
|
2019-03-28 06:27:33 +08:00
|
|
|
//
|
2020-01-11 04:12:03 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2019-03-28 06:27:33 +08:00
|
|
|
|
2020-02-25 23:11:52 +08:00
|
|
|
#include "flang/Evaluate/formatting.h"
|
|
|
|
#include "flang/Evaluate/call.h"
|
|
|
|
#include "flang/Evaluate/constant.h"
|
|
|
|
#include "flang/Evaluate/expression.h"
|
|
|
|
#include "flang/Evaluate/fold.h"
|
|
|
|
#include "flang/Evaluate/tools.h"
|
|
|
|
#include "flang/Parser/characters.h"
|
|
|
|
#include "flang/Semantics/symbol.h"
|
2020-01-15 09:31:25 +08:00
|
|
|
#include <sstream>
|
2019-03-28 06:27:33 +08:00
|
|
|
|
|
|
|
namespace Fortran::evaluate {
|
|
|
|
|
2019-04-19 05:11:15 +08:00
|
|
|
static void ShapeAsFortran(std::ostream &o, const ConstantSubscripts &shape) {
|
2019-05-22 07:58:46 +08:00
|
|
|
if (GetRank(shape) > 1) {
|
2019-03-28 06:27:33 +08:00
|
|
|
o << ",shape=";
|
|
|
|
char ch{'['};
|
|
|
|
for (auto dim : shape) {
|
|
|
|
o << ch << dim;
|
|
|
|
ch = ',';
|
|
|
|
}
|
|
|
|
o << "])";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RESULT, typename VALUE>
|
|
|
|
std::ostream &ConstantBase<RESULT, VALUE>::AsFortran(std::ostream &o) const {
|
|
|
|
if (Rank() > 1) {
|
|
|
|
o << "reshape(";
|
|
|
|
}
|
|
|
|
if (Rank() > 0) {
|
|
|
|
o << '[' << GetType().AsFortran() << "::";
|
|
|
|
}
|
|
|
|
bool first{true};
|
|
|
|
for (const auto &value : values_) {
|
|
|
|
if (first) {
|
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
o << ',';
|
|
|
|
}
|
|
|
|
if constexpr (Result::category == TypeCategory::Integer) {
|
|
|
|
o << value.SignedDecimal() << '_' << Result::kind;
|
|
|
|
} else if constexpr (Result::category == TypeCategory::Real ||
|
|
|
|
Result::category == TypeCategory::Complex) {
|
|
|
|
value.AsFortran(o, Result::kind);
|
|
|
|
} else if constexpr (Result::category == TypeCategory::Character) {
|
2019-06-19 04:46:54 +08:00
|
|
|
o << Result::kind << '_' << parser::QuoteCharacterLiteral(value, true);
|
2019-03-28 06:27:33 +08:00
|
|
|
} else if constexpr (Result::category == TypeCategory::Logical) {
|
|
|
|
if (value.IsTrue()) {
|
|
|
|
o << ".true.";
|
|
|
|
} else {
|
|
|
|
o << ".false.";
|
|
|
|
}
|
|
|
|
o << '_' << Result::kind;
|
|
|
|
} else {
|
2019-05-14 00:33:18 +08:00
|
|
|
StructureConstructor{result_.derivedTypeSpec(), value}.AsFortran(o);
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (Rank() > 0) {
|
|
|
|
o << ']';
|
|
|
|
}
|
2019-07-20 07:17:07 +08:00
|
|
|
ShapeAsFortran(o, shape());
|
2019-03-28 06:27:33 +08:00
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<int KIND>
|
|
|
|
std::ostream &Constant<Type<TypeCategory::Character, KIND>>::AsFortran(
|
|
|
|
std::ostream &o) const {
|
|
|
|
if (Rank() > 1) {
|
|
|
|
o << "reshape(";
|
|
|
|
}
|
|
|
|
if (Rank() > 0) {
|
|
|
|
o << '[' << GetType().AsFortran(std::to_string(length_)) << "::";
|
|
|
|
}
|
2019-06-08 00:57:10 +08:00
|
|
|
auto total{static_cast<ConstantSubscript>(size())};
|
|
|
|
for (ConstantSubscript j{0}; j < total; ++j) {
|
2019-05-14 00:33:18 +08:00
|
|
|
Scalar<Result> value{values_.substr(j * length_, length_)};
|
2019-03-28 06:27:33 +08:00
|
|
|
if (j > 0) {
|
|
|
|
o << ',';
|
2019-08-13 06:29:58 +08:00
|
|
|
}
|
2020-03-18 03:35:31 +08:00
|
|
|
if (Result::kind != 1) {
|
2019-03-28 06:27:33 +08:00
|
|
|
o << Result::kind << '_';
|
|
|
|
}
|
2019-06-18 07:12:28 +08:00
|
|
|
o << parser::QuoteCharacterLiteral(value);
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
|
|
|
if (Rank() > 0) {
|
|
|
|
o << ']';
|
|
|
|
}
|
2019-07-20 07:17:07 +08:00
|
|
|
ShapeAsFortran(o, shape());
|
2019-03-28 06:27:33 +08:00
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
2019-04-19 05:11:15 +08:00
|
|
|
std::ostream &ActualArgument::AssumedType::AsFortran(std::ostream &o) const {
|
|
|
|
return o << symbol_->name().ToString();
|
|
|
|
}
|
|
|
|
|
2019-03-28 06:27:33 +08:00
|
|
|
std::ostream &ActualArgument::AsFortran(std::ostream &o) const {
|
2019-12-06 02:24:18 +08:00
|
|
|
if (keyword_) {
|
|
|
|
o << keyword_->ToString() << '=';
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
2019-12-06 02:24:18 +08:00
|
|
|
if (isAlternateReturn_) {
|
2019-03-28 06:27:33 +08:00
|
|
|
o << '*';
|
|
|
|
}
|
2019-05-22 07:58:46 +08:00
|
|
|
if (const auto *expr{UnwrapExpr()}) {
|
2019-04-19 05:11:15 +08:00
|
|
|
return expr->AsFortran(o);
|
|
|
|
} else {
|
|
|
|
return std::get<AssumedType>(u_).AsFortran(o);
|
|
|
|
}
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &SpecificIntrinsic::AsFortran(std::ostream &o) const {
|
|
|
|
return o << name;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &ProcedureRef::AsFortran(std::ostream &o) const {
|
2019-12-10 05:52:12 +08:00
|
|
|
for (const auto &arg : arguments_) {
|
|
|
|
if (arg && arg->isPassedObject()) {
|
|
|
|
arg->AsFortran(o) << '%';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-03-28 06:27:33 +08:00
|
|
|
proc_.AsFortran(o);
|
|
|
|
char separator{'('};
|
|
|
|
for (const auto &arg : arguments_) {
|
2019-12-10 05:52:12 +08:00
|
|
|
if (arg && !arg->isPassedObject()) {
|
2019-03-28 06:27:33 +08:00
|
|
|
arg->AsFortran(o << separator);
|
|
|
|
separator = ',';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (separator == '(') {
|
|
|
|
o << '(';
|
|
|
|
}
|
|
|
|
return o << ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
// Operator precedence formatting; insert parentheses around operands
|
|
|
|
// only when necessary.
|
|
|
|
|
2019-04-02 04:01:42 +08:00
|
|
|
enum class Precedence { // in increasing order for sane comparisons
|
2019-04-02 03:30:08 +08:00
|
|
|
DefinedBinary,
|
2019-04-02 04:01:42 +08:00
|
|
|
Or,
|
|
|
|
And,
|
|
|
|
Equivalence, // .EQV., .NEQV.
|
|
|
|
Not, // which binds *less* tightly in Fortran than relations
|
2019-04-02 03:30:08 +08:00
|
|
|
Relational,
|
2019-04-02 04:01:42 +08:00
|
|
|
Additive, // +, -, and (arbitrarily) //
|
|
|
|
Negate, // which binds *less* tightly than *, /, **
|
2019-04-02 03:30:08 +08:00
|
|
|
Multiplicative, // *, /
|
|
|
|
Power, // **, which is right-associative unlike the other dyadic operators
|
|
|
|
DefinedUnary,
|
2019-12-18 07:59:25 +08:00
|
|
|
Top,
|
2019-03-28 06:27:33 +08:00
|
|
|
};
|
|
|
|
|
2019-12-18 07:59:25 +08:00
|
|
|
template<typename A> constexpr Precedence ToPrecedence(const A &) {
|
|
|
|
return Precedence::Top;
|
|
|
|
}
|
2019-03-28 06:27:33 +08:00
|
|
|
template<int KIND>
|
2019-12-18 07:59:25 +08:00
|
|
|
static Precedence ToPrecedence(const LogicalOperation<KIND> &x) {
|
|
|
|
switch (x.logicalOperator) {
|
|
|
|
SWITCH_COVERS_ALL_CASES
|
|
|
|
case LogicalOperator::And: return Precedence::And;
|
|
|
|
case LogicalOperator::Or: return Precedence::Or;
|
|
|
|
case LogicalOperator::Not: return Precedence::Not;
|
|
|
|
case LogicalOperator::Eqv:
|
|
|
|
case LogicalOperator::Neqv: return Precedence::Equivalence;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
template<int KIND> constexpr Precedence ToPrecedence(const Not<KIND> &) {
|
|
|
|
return Precedence::Not;
|
|
|
|
}
|
|
|
|
template<typename T> constexpr Precedence ToPrecedence(const Relational<T> &) {
|
|
|
|
return Precedence::Relational;
|
|
|
|
}
|
|
|
|
template<typename T> constexpr Precedence ToPrecedence(const Add<T> &) {
|
|
|
|
return Precedence::Additive;
|
|
|
|
}
|
|
|
|
template<typename T> constexpr Precedence ToPrecedence(const Subtract<T> &) {
|
|
|
|
return Precedence::Additive;
|
|
|
|
}
|
|
|
|
template<int KIND> constexpr Precedence ToPrecedence(const Concat<KIND> &) {
|
|
|
|
return Precedence::Additive;
|
|
|
|
}
|
|
|
|
template<typename T> constexpr Precedence ToPrecedence(const Negate<T> &) {
|
|
|
|
return Precedence::Negate;
|
|
|
|
}
|
|
|
|
template<typename T> constexpr Precedence ToPrecedence(const Multiply<T> &) {
|
|
|
|
return Precedence::Multiplicative;
|
|
|
|
}
|
|
|
|
template<typename T> constexpr Precedence ToPrecedence(const Divide<T> &) {
|
|
|
|
return Precedence::Multiplicative;
|
|
|
|
}
|
|
|
|
template<typename T> constexpr Precedence ToPrecedence(const Power<T> &) {
|
|
|
|
return Precedence::Power;
|
|
|
|
}
|
2019-03-28 06:27:33 +08:00
|
|
|
template<typename T>
|
2019-12-18 07:59:25 +08:00
|
|
|
constexpr Precedence ToPrecedence(const RealToIntPower<T> &) {
|
|
|
|
return Precedence::Power;
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
2019-12-18 07:59:25 +08:00
|
|
|
template<typename T> static Precedence ToPrecedence(const Constant<T> &x) {
|
|
|
|
static constexpr TypeCategory cat{T::category};
|
|
|
|
if constexpr (cat == TypeCategory::Integer || cat == TypeCategory::Real) {
|
|
|
|
if (auto n{GetScalarConstantValue<T>(x)}) {
|
|
|
|
if (n->IsNegative()) {
|
|
|
|
return Precedence::Negate;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-21 02:44:43 +08:00
|
|
|
return Precedence::Top;
|
2019-12-19 09:06:13 +08:00
|
|
|
}
|
2019-12-18 07:59:25 +08:00
|
|
|
template<typename T> static Precedence ToPrecedence(const Expr<T> &expr) {
|
|
|
|
return std::visit([](const auto &x) { return ToPrecedence(x); }, expr.u);
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
|
|
|
|
2019-04-02 03:30:08 +08:00
|
|
|
template<typename T> static bool IsNegatedScalarConstant(const Expr<T> &expr) {
|
|
|
|
static constexpr TypeCategory cat{T::category};
|
|
|
|
if constexpr (cat == TypeCategory::Integer || cat == TypeCategory::Real) {
|
2019-05-14 00:33:18 +08:00
|
|
|
if (auto n{GetScalarConstantValue<T>(expr)}) {
|
|
|
|
return n->IsNegative();
|
2019-04-02 03:30:08 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<TypeCategory CAT>
|
|
|
|
static bool IsNegatedScalarConstant(const Expr<SomeKind<CAT>> &expr) {
|
|
|
|
return std::visit(
|
|
|
|
[](const auto &x) { return IsNegatedScalarConstant(x); }, expr.u);
|
|
|
|
}
|
|
|
|
|
2019-12-18 07:59:25 +08:00
|
|
|
struct OperatorSpelling {
|
|
|
|
const char *prefix{""}, *infix{","}, *suffix{""};
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename A> constexpr OperatorSpelling SpellOperator(const A &) {
|
|
|
|
return OperatorSpelling{};
|
|
|
|
}
|
|
|
|
template<typename A>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const Negate<A> &) {
|
|
|
|
return OperatorSpelling{"-", "", ""};
|
|
|
|
}
|
2019-12-21 02:44:43 +08:00
|
|
|
template<typename A>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const Parentheses<A> &) {
|
|
|
|
return OperatorSpelling{"(", "", ")"};
|
|
|
|
}
|
2019-12-18 07:59:25 +08:00
|
|
|
template<int KIND>
|
|
|
|
static OperatorSpelling SpellOperator(const ComplexComponent<KIND> &x) {
|
2019-12-21 02:44:43 +08:00
|
|
|
return {x.isImaginaryPart ? "aimag(" : "real(", "", ")"};
|
2019-12-18 07:59:25 +08:00
|
|
|
}
|
|
|
|
template<int KIND> constexpr OperatorSpelling SpellOperator(const Not<KIND> &) {
|
|
|
|
return OperatorSpelling{".NOT.", "", ""};
|
|
|
|
}
|
|
|
|
template<int KIND>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const SetLength<KIND> &) {
|
|
|
|
return OperatorSpelling{"%SET_LENGTH(", ",", ")"};
|
|
|
|
}
|
|
|
|
template<int KIND>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const ComplexConstructor<KIND> &) {
|
|
|
|
return OperatorSpelling{"(", ",", ")"};
|
|
|
|
}
|
|
|
|
template<typename A> constexpr OperatorSpelling SpellOperator(const Add<A> &) {
|
|
|
|
return OperatorSpelling{"", "+", ""};
|
|
|
|
}
|
|
|
|
template<typename A>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const Subtract<A> &) {
|
|
|
|
return OperatorSpelling{"", "-", ""};
|
|
|
|
}
|
|
|
|
template<typename A>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const Multiply<A> &) {
|
|
|
|
return OperatorSpelling{"", "*", ""};
|
|
|
|
}
|
|
|
|
template<typename A>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const Divide<A> &) {
|
|
|
|
return OperatorSpelling{"", "/", ""};
|
|
|
|
}
|
|
|
|
template<typename A>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const Power<A> &) {
|
|
|
|
return OperatorSpelling{"", "**", ""};
|
|
|
|
}
|
|
|
|
template<typename A>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const RealToIntPower<A> &) {
|
|
|
|
return OperatorSpelling{"", "**", ""};
|
|
|
|
}
|
|
|
|
template<typename A>
|
|
|
|
static OperatorSpelling SpellOperator(const Extremum<A> &x) {
|
|
|
|
return OperatorSpelling{
|
2019-12-19 09:06:13 +08:00
|
|
|
x.ordering == Ordering::Less ? "min(" : "max(", ",", ")"};
|
2019-12-18 07:59:25 +08:00
|
|
|
}
|
|
|
|
template<int KIND>
|
|
|
|
constexpr OperatorSpelling SpellOperator(const Concat<KIND> &) {
|
|
|
|
return OperatorSpelling{"", "//", ""};
|
|
|
|
}
|
|
|
|
template<int KIND>
|
|
|
|
static OperatorSpelling SpellOperator(const LogicalOperation<KIND> &x) {
|
|
|
|
return OperatorSpelling{"", AsFortran(x.logicalOperator), ""};
|
|
|
|
}
|
|
|
|
template<typename T>
|
|
|
|
static OperatorSpelling SpellOperator(const Relational<T> &x) {
|
|
|
|
return OperatorSpelling{"", AsFortran(x.opr), ""};
|
|
|
|
}
|
|
|
|
|
2019-03-28 06:27:33 +08:00
|
|
|
template<typename D, typename R, typename... O>
|
|
|
|
std::ostream &Operation<D, R, O...>::AsFortran(std::ostream &o) const {
|
2019-12-18 07:59:25 +08:00
|
|
|
Precedence lhsPrec{ToPrecedence(left())};
|
|
|
|
OperatorSpelling spelling{SpellOperator(derived())};
|
|
|
|
o << spelling.prefix;
|
|
|
|
Precedence thisPrec{ToPrecedence(derived())};
|
2019-03-28 06:27:33 +08:00
|
|
|
if constexpr (operands == 1) {
|
2019-12-21 02:44:43 +08:00
|
|
|
if (thisPrec != Precedence::Top && lhsPrec < thisPrec) {
|
2020-01-15 09:31:25 +08:00
|
|
|
left().AsFortran(o << '(') << ')';
|
2019-12-18 07:59:25 +08:00
|
|
|
} else {
|
2020-01-15 09:31:25 +08:00
|
|
|
left().AsFortran(o);
|
2019-12-18 07:59:25 +08:00
|
|
|
}
|
2019-03-28 06:27:33 +08:00
|
|
|
} else {
|
2019-12-21 02:44:43 +08:00
|
|
|
if (thisPrec != Precedence::Top &&
|
|
|
|
(lhsPrec < thisPrec ||
|
|
|
|
(lhsPrec == Precedence::Power && thisPrec == Precedence::Power))) {
|
2020-01-15 09:31:25 +08:00
|
|
|
left().AsFortran(o << '(') << ')';
|
2019-12-18 07:59:25 +08:00
|
|
|
} else {
|
2020-01-15 09:31:25 +08:00
|
|
|
left().AsFortran(o);
|
2019-12-18 07:59:25 +08:00
|
|
|
}
|
|
|
|
o << spelling.infix;
|
|
|
|
Precedence rhsPrec{ToPrecedence(right())};
|
2019-12-21 02:44:43 +08:00
|
|
|
if (thisPrec != Precedence::Top && rhsPrec < thisPrec) {
|
2020-01-15 09:31:25 +08:00
|
|
|
right().AsFortran(o << '(') << ')';
|
2019-12-18 07:59:25 +08:00
|
|
|
} else {
|
2020-01-15 09:31:25 +08:00
|
|
|
right().AsFortran(o);
|
2019-12-18 07:59:25 +08:00
|
|
|
}
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
2019-12-18 07:59:25 +08:00
|
|
|
return o << spelling.suffix;
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename TO, TypeCategory FROMCAT>
|
|
|
|
std::ostream &Convert<TO, FROMCAT>::AsFortran(std::ostream &o) const {
|
|
|
|
static_assert(TO::category == TypeCategory::Integer ||
|
2019-10-03 03:40:52 +08:00
|
|
|
TO::category == TypeCategory::Real ||
|
|
|
|
TO::category == TypeCategory::Character ||
|
|
|
|
TO::category == TypeCategory::Logical,
|
|
|
|
"Convert<> to bad category!");
|
2019-03-28 06:27:33 +08:00
|
|
|
if constexpr (TO::category == TypeCategory::Character) {
|
|
|
|
this->left().AsFortran(o << "achar(iachar(") << ')';
|
|
|
|
} else if constexpr (TO::category == TypeCategory::Integer) {
|
|
|
|
this->left().AsFortran(o << "int(");
|
|
|
|
} else if constexpr (TO::category == TypeCategory::Real) {
|
|
|
|
this->left().AsFortran(o << "real(");
|
|
|
|
} else {
|
|
|
|
this->left().AsFortran(o << "logical(");
|
|
|
|
}
|
|
|
|
return o << ",kind=" << TO::kind << ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &Relational<SomeType>::AsFortran(std::ostream &o) const {
|
|
|
|
std::visit([&](const auto &rel) { rel.AsFortran(o); }, u);
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2019-04-06 02:38:10 +08:00
|
|
|
std::ostream &EmitArray(std::ostream &o, const Expr<T> &expr) {
|
|
|
|
return expr.AsFortran(o);
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2019-04-02 01:54:22 +08:00
|
|
|
std::ostream &EmitArray(std::ostream &, const ArrayConstructorValues<T> &);
|
2019-03-28 06:27:33 +08:00
|
|
|
|
|
|
|
template<typename T>
|
2019-04-02 01:54:22 +08:00
|
|
|
std::ostream &EmitArray(std::ostream &o, const ImpliedDo<T> &implDo) {
|
2019-03-28 06:27:33 +08:00
|
|
|
o << '(';
|
2019-04-02 01:54:22 +08:00
|
|
|
EmitArray(o, implDo.values());
|
2019-03-28 06:27:33 +08:00
|
|
|
o << ',' << ImpliedDoIndex::Result::AsFortran()
|
|
|
|
<< "::" << implDo.name().ToString() << '=';
|
|
|
|
implDo.lower().AsFortran(o) << ',';
|
|
|
|
implDo.upper().AsFortran(o) << ',';
|
|
|
|
implDo.stride().AsFortran(o) << ')';
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2019-04-02 03:30:08 +08:00
|
|
|
std::ostream &EmitArray(
|
|
|
|
std::ostream &o, const ArrayConstructorValues<T> &values) {
|
2019-03-28 06:27:33 +08:00
|
|
|
const char *sep{""};
|
2019-04-19 05:11:15 +08:00
|
|
|
for (const auto &value : values) {
|
2019-03-28 06:27:33 +08:00
|
|
|
o << sep;
|
2019-04-02 01:54:22 +08:00
|
|
|
std::visit([&](const auto &x) { EmitArray(o, x); }, value.u);
|
2019-03-28 06:27:33 +08:00
|
|
|
sep = ",";
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
std::ostream &ArrayConstructor<T>::AsFortran(std::ostream &o) const {
|
|
|
|
o << '[' << GetType().AsFortran() << "::";
|
2019-04-02 01:54:22 +08:00
|
|
|
EmitArray(o, *this);
|
2019-03-28 06:27:33 +08:00
|
|
|
return o << ']';
|
|
|
|
}
|
|
|
|
|
|
|
|
template<int KIND>
|
|
|
|
std::ostream &ArrayConstructor<Type<TypeCategory::Character, KIND>>::AsFortran(
|
|
|
|
std::ostream &o) const {
|
2020-01-15 09:31:25 +08:00
|
|
|
o << '[' << GetType().AsFortran(LEN().AsFortran()) << "::";
|
2019-04-02 01:54:22 +08:00
|
|
|
EmitArray(o, *this);
|
2019-03-28 06:27:33 +08:00
|
|
|
return o << ']';
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &ArrayConstructor<SomeDerived>::AsFortran(std::ostream &o) const {
|
|
|
|
o << '[' << GetType().AsFortran() << "::";
|
2019-04-02 01:54:22 +08:00
|
|
|
EmitArray(o, *this);
|
2019-03-28 06:27:33 +08:00
|
|
|
return o << ']';
|
|
|
|
}
|
|
|
|
|
2020-01-15 09:31:25 +08:00
|
|
|
template<typename RESULT>
|
|
|
|
std::string ExpressionBase<RESULT>::AsFortran() const {
|
|
|
|
std::ostringstream ss;
|
|
|
|
AsFortran(ss);
|
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
2019-03-28 06:27:33 +08:00
|
|
|
template<typename RESULT>
|
|
|
|
std::ostream &ExpressionBase<RESULT>::AsFortran(std::ostream &o) const {
|
|
|
|
std::visit(
|
|
|
|
common::visitors{
|
|
|
|
[&](const BOZLiteralConstant &x) {
|
|
|
|
o << "z'" << x.Hexadecimal() << "'";
|
|
|
|
},
|
|
|
|
[&](const NullPointer &) { o << "NULL()"; },
|
|
|
|
[&](const common::CopyableIndirection<Substring> &s) {
|
|
|
|
s.value().AsFortran(o);
|
|
|
|
},
|
|
|
|
[&](const ImpliedDoIndex &i) { o << i.name.ToString(); },
|
|
|
|
[&](const auto &x) { x.AsFortran(o); },
|
|
|
|
},
|
|
|
|
derived().u);
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &StructureConstructor::AsFortran(std::ostream &o) const {
|
2019-05-14 00:33:18 +08:00
|
|
|
o << DerivedTypeSpecAsFortran(result_.derivedTypeSpec());
|
2019-03-28 06:27:33 +08:00
|
|
|
if (values_.empty()) {
|
|
|
|
o << '(';
|
|
|
|
} else {
|
|
|
|
char ch{'('};
|
|
|
|
for (const auto &[symbol, value] : values_) {
|
|
|
|
value.value().AsFortran(o << ch << symbol->name().ToString() << '=');
|
|
|
|
ch = ',';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return o << ')';
|
|
|
|
}
|
|
|
|
|
2019-05-22 01:24:30 +08:00
|
|
|
std::string DynamicType::AsFortran() const {
|
2019-11-10 01:29:31 +08:00
|
|
|
if (derived_) {
|
2019-05-22 01:24:30 +08:00
|
|
|
CHECK(category_ == TypeCategory::Derived);
|
2019-05-30 04:22:35 +08:00
|
|
|
return DerivedTypeSpecAsFortran(*derived_);
|
2019-11-10 01:29:31 +08:00
|
|
|
} else if (charLength_) {
|
2019-05-22 01:24:30 +08:00
|
|
|
std::string result{"CHARACTER(KIND="s + std::to_string(kind_) + ",LEN="};
|
|
|
|
if (charLength_->isAssumed()) {
|
|
|
|
result += '*';
|
|
|
|
} else if (charLength_->isDeferred()) {
|
|
|
|
result += ':';
|
|
|
|
} else if (const auto &length{charLength_->GetExplicit()}) {
|
2020-01-15 09:31:25 +08:00
|
|
|
result += length->AsFortran();
|
2019-05-22 01:24:30 +08:00
|
|
|
}
|
|
|
|
return result + ')';
|
2019-07-02 07:54:53 +08:00
|
|
|
} else if (IsUnlimitedPolymorphic()) {
|
|
|
|
return "CLASS(*)";
|
|
|
|
} else if (IsAssumedType()) {
|
|
|
|
return "TYPE(*)";
|
2019-10-09 06:21:09 +08:00
|
|
|
} else if (IsTypelessIntrinsicArgument()) {
|
2019-05-22 01:24:30 +08:00
|
|
|
return "(typeless intrinsic function argument)";
|
|
|
|
} else {
|
2020-01-07 06:04:17 +08:00
|
|
|
return parser::ToUpperCaseLetters(EnumToString(category_)) + '(' +
|
|
|
|
std::to_string(kind_) + ')';
|
2019-05-22 01:24:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string DynamicType::AsFortran(std::string &&charLenExpr) const {
|
|
|
|
if (!charLenExpr.empty() && category_ == TypeCategory::Character) {
|
|
|
|
return "CHARACTER(KIND=" + std::to_string(kind_) +
|
|
|
|
",LEN=" + std::move(charLenExpr) + ')';
|
|
|
|
} else {
|
|
|
|
return AsFortran();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string SomeDerived::AsFortran() const {
|
|
|
|
if (IsUnlimitedPolymorphic()) {
|
|
|
|
return "CLASS(*)";
|
|
|
|
} else {
|
|
|
|
return "TYPE("s + DerivedTypeSpecAsFortran(derivedTypeSpec()) + ')';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-14 00:33:18 +08:00
|
|
|
std::string DerivedTypeSpecAsFortran(const semantics::DerivedTypeSpec &spec) {
|
2019-09-17 07:58:13 +08:00
|
|
|
std::stringstream ss;
|
|
|
|
ss << spec.name().ToString();
|
|
|
|
char ch{'('};
|
|
|
|
for (const auto &[name, value] : spec.parameters()) {
|
|
|
|
ss << ch << name.ToString() << '=';
|
|
|
|
ch = ',';
|
|
|
|
if (value.isAssumed()) {
|
|
|
|
ss << '*';
|
|
|
|
} else if (value.isDeferred()) {
|
|
|
|
ss << ':';
|
|
|
|
} else {
|
|
|
|
value.GetExplicit()->AsFortran(ss);
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
2019-09-17 07:58:13 +08:00
|
|
|
}
|
|
|
|
if (ch != '(') {
|
2019-05-14 00:33:18 +08:00
|
|
|
ss << ')';
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
2019-09-17 07:58:13 +08:00
|
|
|
return ss.str();
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|
|
|
|
|
2019-04-02 01:54:22 +08:00
|
|
|
std::ostream &EmitVar(std::ostream &o, const Symbol &symbol) {
|
|
|
|
return o << symbol.name().ToString();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &EmitVar(std::ostream &o, const std::string &lit) {
|
|
|
|
return o << parser::QuoteCharacterLiteral(lit);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &EmitVar(std::ostream &o, const std::u16string &lit) {
|
|
|
|
return o << parser::QuoteCharacterLiteral(lit);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &EmitVar(std::ostream &o, const std::u32string &lit) {
|
|
|
|
return o << parser::QuoteCharacterLiteral(lit);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename A> std::ostream &EmitVar(std::ostream &o, const A &x) {
|
|
|
|
return x.AsFortran(o);
|
|
|
|
}
|
|
|
|
|
2019-10-23 07:53:29 +08:00
|
|
|
template<typename A>
|
|
|
|
std::ostream &EmitVar(std::ostream &o, common::Reference<A> x) {
|
|
|
|
return EmitVar(o, *x);
|
|
|
|
}
|
|
|
|
|
2019-04-02 01:54:22 +08:00
|
|
|
template<typename A>
|
|
|
|
std::ostream &EmitVar(std::ostream &o, const A *p, const char *kw = nullptr) {
|
2019-11-10 01:29:31 +08:00
|
|
|
if (p) {
|
|
|
|
if (kw) {
|
2019-04-02 01:54:22 +08:00
|
|
|
o << kw;
|
|
|
|
}
|
|
|
|
EmitVar(o, *p);
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename A>
|
|
|
|
std::ostream &EmitVar(
|
|
|
|
std::ostream &o, const std::optional<A> &x, const char *kw = nullptr) {
|
2019-11-10 01:29:31 +08:00
|
|
|
if (x) {
|
|
|
|
if (kw) {
|
2019-04-02 01:54:22 +08:00
|
|
|
o << kw;
|
|
|
|
}
|
|
|
|
EmitVar(o, *x);
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename A, bool COPY>
|
|
|
|
std::ostream &EmitVar(std::ostream &o, const common::Indirection<A, COPY> &p,
|
|
|
|
const char *kw = nullptr) {
|
2019-11-10 01:29:31 +08:00
|
|
|
if (kw) {
|
2019-04-02 01:54:22 +08:00
|
|
|
o << kw;
|
|
|
|
}
|
|
|
|
EmitVar(o, p.value());
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename A>
|
|
|
|
std::ostream &EmitVar(std::ostream &o, const std::shared_ptr<A> &p) {
|
2019-11-10 01:29:31 +08:00
|
|
|
CHECK(p);
|
2019-04-02 01:54:22 +08:00
|
|
|
return EmitVar(o, *p);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename... A>
|
|
|
|
std::ostream &EmitVar(std::ostream &o, const std::variant<A...> &u) {
|
|
|
|
std::visit([&](const auto &x) { EmitVar(o, x); }, u);
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &BaseObject::AsFortran(std::ostream &o) const {
|
|
|
|
return EmitVar(o, u);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<int KIND>
|
|
|
|
std::ostream &TypeParamInquiry<KIND>::AsFortran(std::ostream &o) const {
|
2019-11-10 01:29:31 +08:00
|
|
|
if (base_) {
|
2019-06-26 04:07:32 +08:00
|
|
|
return base_->AsFortran(o) << '%';
|
|
|
|
}
|
2019-10-23 07:53:29 +08:00
|
|
|
return EmitVar(o, parameter_);
|
2019-04-02 01:54:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &Component::AsFortran(std::ostream &o) const {
|
|
|
|
base_.value().AsFortran(o);
|
2019-10-23 07:53:29 +08:00
|
|
|
return EmitVar(o << '%', symbol_);
|
2019-04-02 01:54:22 +08:00
|
|
|
}
|
|
|
|
|
2019-06-26 04:07:32 +08:00
|
|
|
std::ostream &NamedEntity::AsFortran(std::ostream &o) const {
|
|
|
|
std::visit(
|
|
|
|
common::visitors{
|
2019-10-23 07:53:29 +08:00
|
|
|
[&](SymbolRef s) { EmitVar(o, s); },
|
2019-06-26 04:07:32 +08:00
|
|
|
[&](const Component &c) { c.AsFortran(o); },
|
|
|
|
},
|
|
|
|
u_);
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
2019-04-02 01:54:22 +08:00
|
|
|
std::ostream &Triplet::AsFortran(std::ostream &o) const {
|
|
|
|
EmitVar(o, lower_) << ':';
|
|
|
|
EmitVar(o, upper_);
|
|
|
|
EmitVar(o << ':', stride_.value());
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
2019-04-02 03:30:08 +08:00
|
|
|
std::ostream &Subscript::AsFortran(std::ostream &o) const {
|
|
|
|
return EmitVar(o, u);
|
|
|
|
}
|
2019-04-02 01:54:22 +08:00
|
|
|
|
|
|
|
std::ostream &ArrayRef::AsFortran(std::ostream &o) const {
|
2019-06-26 04:07:32 +08:00
|
|
|
base_.AsFortran(o);
|
2019-04-02 01:54:22 +08:00
|
|
|
char separator{'('};
|
|
|
|
for (const Subscript &ss : subscript_) {
|
|
|
|
ss.AsFortran(o << separator);
|
|
|
|
separator = ',';
|
|
|
|
}
|
|
|
|
return o << ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &CoarrayRef::AsFortran(std::ostream &o) const {
|
2019-04-02 06:22:45 +08:00
|
|
|
bool first{true};
|
2019-10-23 07:53:29 +08:00
|
|
|
for (const Symbol &part : base_) {
|
2019-04-02 06:22:45 +08:00
|
|
|
if (first) {
|
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
o << '%';
|
|
|
|
}
|
2019-10-23 07:53:29 +08:00
|
|
|
EmitVar(o, part);
|
2019-04-04 03:57:24 +08:00
|
|
|
}
|
|
|
|
char separator{'('};
|
|
|
|
for (const auto &sscript : subscript_) {
|
|
|
|
EmitVar(o << separator, sscript);
|
|
|
|
separator = ',';
|
|
|
|
}
|
|
|
|
if (separator == ',') {
|
|
|
|
o << ')';
|
2019-04-02 01:54:22 +08:00
|
|
|
}
|
2019-04-04 03:57:24 +08:00
|
|
|
separator = '[';
|
2019-04-02 01:54:22 +08:00
|
|
|
for (const auto &css : cosubscript_) {
|
|
|
|
EmitVar(o << separator, css);
|
|
|
|
separator = ',';
|
|
|
|
}
|
2019-11-10 01:29:31 +08:00
|
|
|
if (stat_) {
|
2019-04-02 01:54:22 +08:00
|
|
|
EmitVar(o << separator, stat_, "STAT=");
|
|
|
|
separator = ',';
|
|
|
|
}
|
2019-11-10 01:29:31 +08:00
|
|
|
if (team_) {
|
2019-04-02 03:30:08 +08:00
|
|
|
EmitVar(
|
|
|
|
o << separator, team_, teamIsTeamNumber_ ? "TEAM_NUMBER=" : "TEAM=");
|
2019-04-02 01:54:22 +08:00
|
|
|
}
|
|
|
|
return o << ']';
|
|
|
|
}
|
|
|
|
|
2019-04-02 03:30:08 +08:00
|
|
|
std::ostream &DataRef::AsFortran(std::ostream &o) const {
|
|
|
|
return EmitVar(o, u);
|
|
|
|
}
|
2019-04-02 01:54:22 +08:00
|
|
|
|
|
|
|
std::ostream &Substring::AsFortran(std::ostream &o) const {
|
|
|
|
EmitVar(o, parent_) << '(';
|
|
|
|
EmitVar(o, lower_) << ':';
|
2019-06-26 04:07:32 +08:00
|
|
|
return EmitVar(o, upper_) << ')';
|
2019-04-02 01:54:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &ComplexPart::AsFortran(std::ostream &o) const {
|
|
|
|
return complex_.AsFortran(o) << '%' << EnumToString(part_);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream &ProcedureDesignator::AsFortran(std::ostream &o) const {
|
|
|
|
return EmitVar(o, u);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
std::ostream &Designator<T>::AsFortran(std::ostream &o) const {
|
|
|
|
std::visit(
|
|
|
|
common::visitors{
|
2019-10-23 07:53:29 +08:00
|
|
|
[&](SymbolRef symbol) { EmitVar(o, symbol); },
|
2019-04-02 01:54:22 +08:00
|
|
|
[&](const auto &x) { x.AsFortran(o); },
|
|
|
|
},
|
|
|
|
u);
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
2019-04-02 06:22:45 +08:00
|
|
|
std::ostream &DescriptorInquiry::AsFortran(std::ostream &o) const {
|
|
|
|
switch (field_) {
|
|
|
|
case Field::LowerBound: o << "lbound("; break;
|
2019-04-05 04:12:21 +08:00
|
|
|
case Field::Extent: o << "size("; break;
|
2019-04-02 06:22:45 +08:00
|
|
|
case Field::Stride: o << "%STRIDE("; break;
|
2019-04-05 04:12:21 +08:00
|
|
|
case Field::Rank: o << "rank("; break;
|
2019-12-12 07:06:24 +08:00
|
|
|
case Field::Len: break;
|
2019-04-02 06:22:45 +08:00
|
|
|
}
|
2019-06-26 04:07:32 +08:00
|
|
|
base_.AsFortran(o);
|
2019-12-12 07:06:24 +08:00
|
|
|
if (field_ == Field::Len) {
|
|
|
|
return o << "%len";
|
|
|
|
} else {
|
|
|
|
if (dimension_ >= 0) {
|
|
|
|
o << ",dim=" << (dimension_ + 1);
|
|
|
|
}
|
|
|
|
return o << ')';
|
2019-04-02 06:22:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-14 00:52:15 +08:00
|
|
|
std::ostream &Assignment::AsFortran(std::ostream &o) const {
|
|
|
|
std::visit(
|
|
|
|
common::visitors{
|
|
|
|
[&](const Assignment::Intrinsic &) {
|
|
|
|
rhs.AsFortran(lhs.AsFortran(o) << '=');
|
|
|
|
},
|
|
|
|
[&](const ProcedureRef &proc) { proc.AsFortran(o << "CALL "); },
|
|
|
|
[&](const BoundsSpec &bounds) {
|
|
|
|
lhs.AsFortran(o);
|
|
|
|
if (!bounds.empty()) {
|
|
|
|
char sep{'('};
|
|
|
|
for (const auto &bound : bounds) {
|
|
|
|
bound.AsFortran(o << sep) << ':';
|
|
|
|
sep = ',';
|
|
|
|
}
|
|
|
|
o << ')';
|
|
|
|
}
|
|
|
|
rhs.AsFortran(o << " => ");
|
|
|
|
},
|
|
|
|
[&](const BoundsRemapping &bounds) {
|
|
|
|
lhs.AsFortran(o);
|
|
|
|
if (!bounds.empty()) {
|
|
|
|
char sep{'('};
|
|
|
|
for (const auto &bound : bounds) {
|
|
|
|
bound.first.AsFortran(o << sep) << ':';
|
|
|
|
bound.second.AsFortran(o);
|
|
|
|
sep = ',';
|
|
|
|
}
|
|
|
|
o << ')';
|
|
|
|
}
|
|
|
|
rhs.AsFortran(o << " => ");
|
|
|
|
},
|
|
|
|
},
|
|
|
|
u);
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
2019-03-28 06:27:33 +08:00
|
|
|
INSTANTIATE_CONSTANT_TEMPLATES
|
|
|
|
INSTANTIATE_EXPRESSION_TEMPLATES
|
2019-04-02 01:54:22 +08:00
|
|
|
INSTANTIATE_VARIABLE_TEMPLATES
|
2019-03-28 06:27:33 +08:00
|
|
|
}
|