Krzysztof Parzyszek d542fac6b1
[flang] Add traits to more AST nodes (#175578)
Follow-up to PR175211.

There are still a few AST nodes that don't have any of the standard
traits (Wrapper/Tuple/etc). Because of that they require special
handling in the parse tree visitor.

Convert a subset of these nodes to the typical format, and remove the
special cases from the parse tree visitor.

The members of these nodes were frequently used, so instead of
extracting them by hand each time use helper member functions to access
them.
2026-01-20 09:57:35 -06:00

192 lines
6.0 KiB
C++

//===-- lib/Parser/tools.cpp ----------------------------------------------===//
//
// 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
//
//===----------------------------------------------------------------------===//
#include "flang/Parser/tools.h"
namespace Fortran::parser {
const Name &GetLastName(const Name &x) { return x; }
const Name &GetLastName(const StructureComponent &x) {
return GetLastName(x.Component());
}
const Name &GetLastName(const DataRef &x) {
return common::visit(
common::visitors{
[](const Name &name) -> const Name & { return name; },
[](const common::Indirection<StructureComponent> &sc)
-> const Name & { return GetLastName(sc.value()); },
[](const common::Indirection<ArrayElement> &sc) -> const Name & {
return GetLastName(sc.value().Base());
},
[](const common::Indirection<CoindexedNamedObject> &ci)
-> const Name & {
return GetLastName(std::get<DataRef>(ci.value().t));
},
},
x.u);
}
const Name &GetLastName(const Substring &x) {
return GetLastName(std::get<DataRef>(x.t));
}
const Name &GetLastName(const Designator &x) {
return common::visit(
[](const auto &y) -> const Name & { return GetLastName(y); }, x.u);
}
const Name &GetLastName(const ProcComponentRef &x) {
return GetLastName(x.v.thing);
}
const Name &GetLastName(const ProcedureDesignator &x) {
return common::visit(
[](const auto &y) -> const Name & { return GetLastName(y); }, x.u);
}
const Name &GetLastName(const Call &x) {
return GetLastName(std::get<ProcedureDesignator>(x.t));
}
const Name &GetLastName(const FunctionReference &x) { return GetLastName(x.v); }
const Name &GetLastName(const Variable &x) {
return common::visit(
[](const auto &indirection) -> const Name & {
return GetLastName(indirection.value());
},
x.u);
}
const Name &GetLastName(const AllocateObject &x) {
return common::visit(
[](const auto &y) -> const Name & { return GetLastName(y); }, x.u);
}
const Name &GetFirstName(const Name &x) { return x; }
const Name &GetFirstName(const StructureComponent &x) {
return GetFirstName(x.Base());
}
const Name &GetFirstName(const DataRef &x) {
return common::visit(
common::visitors{
[](const Name &name) -> const Name & { return name; },
[](const common::Indirection<StructureComponent> &sc)
-> const Name & { return GetFirstName(sc.value()); },
[](const common::Indirection<ArrayElement> &sc) -> const Name & {
return GetFirstName(sc.value().Base());
},
[](const common::Indirection<CoindexedNamedObject> &ci)
-> const Name & {
return GetFirstName(std::get<DataRef>(ci.value().t));
},
},
x.u);
}
const Name &GetFirstName(const Substring &x) {
return GetFirstName(std::get<DataRef>(x.t));
}
const Name &GetFirstName(const Designator &x) {
return common::visit(
[](const auto &y) -> const Name & { return GetFirstName(y); }, x.u);
}
const Name &GetFirstName(const ProcComponentRef &x) {
return GetFirstName(x.v.thing);
}
const Name &GetFirstName(const ProcedureDesignator &x) {
return common::visit(
[](const auto &y) -> const Name & { return GetFirstName(y); }, x.u);
}
const Name &GetFirstName(const Call &x) {
return GetFirstName(std::get<ProcedureDesignator>(x.t));
}
const Name &GetFirstName(const FunctionReference &x) {
return GetFirstName(x.v);
}
const Name &GetFirstName(const Variable &x) {
return common::visit(
[](const auto &indirect) -> const Name & {
return GetFirstName(indirect.value());
},
x.u);
}
const Name &GetFirstName(const EntityDecl &x) {
return std::get<ObjectName>(x.t);
}
const CoindexedNamedObject *GetCoindexedNamedObject(const DataRef &base) {
return common::visit(
common::visitors{
[](const Name &) -> const CoindexedNamedObject * { return nullptr; },
[](const common::Indirection<CoindexedNamedObject> &x)
-> const CoindexedNamedObject * { return &x.value(); },
[](const auto &x) -> const CoindexedNamedObject * {
return GetCoindexedNamedObject(x.value().Base());
},
},
base.u);
}
const CoindexedNamedObject *GetCoindexedNamedObject(
const Designator &designator) {
return common::visit(
common::visitors{
[](const DataRef &x) -> const CoindexedNamedObject * {
return GetCoindexedNamedObject(x);
},
[](const Substring &x) -> const CoindexedNamedObject * {
return GetCoindexedNamedObject(std::get<DataRef>(x.t));
},
},
designator.u);
}
const CoindexedNamedObject *GetCoindexedNamedObject(const Variable &variable) {
return common::visit(
common::visitors{
[](const common::Indirection<Designator> &designator)
-> const CoindexedNamedObject * {
return GetCoindexedNamedObject(designator.value());
},
[](const auto &) -> const CoindexedNamedObject * { return nullptr; },
},
variable.u);
}
const CoindexedNamedObject *GetCoindexedNamedObject(
const AllocateObject &allocateObject) {
return common::visit(
common::visitors{
[](const StructureComponent &x) -> const CoindexedNamedObject * {
return GetCoindexedNamedObject(x.Base());
},
[](const auto &) -> const CoindexedNamedObject * { return nullptr; },
},
allocateObject.u);
}
bool CheckForSingleVariableOnRHS(const AssignmentStmt &assignmentStmt) {
return Unwrap<Designator>(std::get<Expr>(assignmentStmt.t)) != nullptr;
}
const Name *GetDesignatorNameIfDataRef(const Designator &designator) {
const auto *dataRef{std::get_if<DataRef>(&designator.u)};
return dataRef ? std::get_if<Name>(&dataRef->u) : nullptr;
}
} // namespace Fortran::parser