13#ifndef FORTRAN_SEMANTICS_OPENMP_UTILS_H
14#define FORTRAN_SEMANTICS_OPENMP_UTILS_H
16#include "flang/Common/indirection.h"
17#include "flang/Evaluate/type.h"
18#include "flang/Parser/char-block.h"
19#include "flang/Parser/message.h"
20#include "flang/Parser/openmp-utils.h"
21#include "flang/Parser/parse-tree.h"
22#include "flang/Parser/tools.h"
23#include "flang/Semantics/tools.h"
25#include "llvm/ADT/APInt.h"
26#include "llvm/ADT/ArrayRef.h"
27#include "llvm/Frontend/OpenMP/OMPContext.h"
37namespace Fortran::semantics {
44using Fortran::parser::omp::BlockRange;
45using Fortran::parser::omp::ExecutionPartIterator;
46using Fortran::parser::omp::is_range_v;
47using Fortran::parser::omp::LoopNestIterator;
48using Fortran::parser::omp::LoopRange;
50template <
typename T,
typename U = std::remove_const_t<T>> U AsRvalue(T &t) {
54template <
typename T> T &&AsRvalue(T &&t) {
return std::move(t); }
56const Scope &GetScopingUnit(
const Scope &scope);
57const Scope &GetProgramUnit(
const Scope &scope);
59template <
typename T>
struct WithSource {
61 typename U = std::remove_reference_t<T>,
62 typename = std::enable_if_t<std::is_default_constructible_v<U>>>
63 WithSource() : value(), source() {}
64 WithSource(
const WithSource<T> &) =
default;
65 WithSource(WithSource<T> &&) =
default;
68 WithSource &operator=(
const WithSource<T> &) =
default;
69 WithSource &operator=(WithSource<T> &&) =
default;
80 using WithSource<value_type>::WithSource;
81 value_type stmt()
const {
return value; }
82 operator bool()
const {
return stmt() !=
nullptr; }
88std::string ThisVersion(
unsigned version);
89std::string TryVersion(
unsigned version);
91const Symbol *GetObjectSymbol(
93const Symbol *GetArgumentSymbol(
96bool IsCommonBlock(
const Symbol &sym);
97bool IsExtendedListItem(
const Symbol &sym);
98bool IsVariableListItem(
const Symbol &sym);
99bool IsTypeParamInquiry(
const Symbol &sym);
100bool IsComplexPart(
const Symbol &sym);
101bool IsStructureComponent(
const Symbol &sym);
102bool IsPrivatizable(
const Symbol &sym);
103bool IsVarOrFunctionRef(
const MaybeExpr &expr);
107bool IsExtendedListItem(
109bool IsLocatorListItem(
111bool IsVariableListItem(
119bool IsMapEnteringType(parser::OmpMapType::Value type);
120bool IsMapExitingType(parser::OmpMapType::Value type);
122MaybeExpr GetEvaluateExpr(
const parser::Expr &parserExpr);
123template <
typename T> MaybeExpr GetEvaluateExpr(
const T &inp) {
124 return GetEvaluateExpr(parser::UnwrapRef<parser::Expr>(inp));
127std::optional<evaluate::DynamicType> GetDynamicType(
130std::optional<bool> GetLogicalValue(
const SomeExpr &expr);
131std::optional<int64_t> GetIntValueFromExpr(
135std::optional<int64_t> GetIntValueFromExpr(
137 if (
auto *parserExpr{parser::Unwrap<parser::Expr>(wrappedExpr)}) {
138 return GetIntValueFromExpr(*parserExpr, semaCtx);
147template <
typename ClauseTy>
148std::optional<bool> GetLogicalArgument(
149 const std::optional<ClauseTy> &maybeClause, SemanticsContext &semaCtx) {
152 auto &parserExpr{parser::UnwrapRef<parser::Expr>(*maybeClause)};
153 evaluate::ExpressionAnalyzer ea{semaCtx};
154 if (
auto &&maybeExpr{ea.Analyze(parserExpr)}) {
155 if (
auto v{GetLogicalValue(*maybeExpr)}) {
163std::optional<bool> IsContiguous(
164 SemanticsContext &semaCtx,
const parser::OmpObject &
object);
166std::vector<SomeExpr> GetTopLevelDesignators(
const SomeExpr &expr);
167const SomeExpr *HasStorageOverlap(
168 const SomeExpr &base, llvm::ArrayRef<SomeExpr> exprs);
170bool IsAssignment(
const parser::ActionStmt *x);
171bool IsPointerAssignment(
const evaluate::Assignment &x);
173MaybeExpr MakeEvaluateExpr(
const parser::OmpStylizedInstance &inp);
175enum struct ListItemKind : uint32_t {
178 DirectiveSpecification,
189std::optional<ListItemKind> GetArgumentListItemKind(
190 llvm::omp::Clause clause,
unsigned version);
192bool IsLoopTransforming(llvm::omp::Directive dir);
193bool HasDataEnvironment(llvm::omp::Directive dir);
195bool IsFullUnroll(
const parser::OmpDirectiveSpecification &spec);
197inline bool IsDoConcurrentLegal(
unsigned version) {
200 return version >= 60;
204 LoopControl(LoopControl &&x) =
default;
205 LoopControl(
const LoopControl &x) =
default;
206 LoopControl(
const parser::LoopControl::Bounds &x);
221 Reason(Reason &&) =
default;
222 Reason(
const Reason &);
223 Reason &operator=(Reason &&) =
default;
224 Reason &operator=(
const Reason &);
228 template <
typename... Ts> Reason &Say(Ts &&...args) {
229 msgs.Say(std::forward<Ts>(args)...);
233 Reason &Append(
const Reason &other) {
237 operator bool()
const {
return !msgs.empty(); }
240 void CopyFrom(
const Reason &other);
245template <
typename T>
struct WithReason {
246 std::optional<T> value;
249 WithReason() =
default;
250 WithReason(std::optional<T> v,
const Reason &r =
Reason())
251 : value(v), reason(r) {}
252 operator bool()
const {
return value.has_value(); }
267std::pair<WithReason<int64_t>,
bool> GetAffectedNestDepthWithReason(
272std::pair<WithReason<int64_t>,
bool> GetGeneratedNestDepthWithReason(
289std::optional<int64_t> GetMinimumSequenceCount(
290 std::optional<int64_t> first, std::optional<int64_t> count);
291std::optional<int64_t> GetMinimumSequenceCount(
292 std::optional<std::pair<int64_t, int64_t>> range);
299std::optional<std::vector<const parser::DoConstruct *>> CollectAffectedDoLoops(
307 template <
typename R,
typename = std::enable_if_t<is_range_v<R>>>
308 LoopSequence(
const R &range,
unsigned version,
bool allowAllLoops =
false,
310 : version_(version), allowAllLoops_(allowAllLoops), semaCtx_(semaCtx) {
311 entry_ = std::make_unique<Construct>(range,
nullptr);
312 createChildrenFromRange(entry_->location);
325 bool isNest()
const {
return length_.value == 1; }
328 const Depth &depth()
const {
return depth_; }
329 const std::vector<LoopSequence> &children()
const {
return children_; }
330 const parser::ExecutionPartConstruct *owner()
const {
return entry_->owner; }
332 WithReason<bool> isWellFormedSequence()
const;
333 WithReason<bool> isWellFormedNest()
const;
339 std::vector<LoopControl> getLoopControls()
const;
342 WithReason<bool> isRectangular(
343 const std::vector<const LoopSequence *> &outer)
const;
346 using Construct = ExecutionPartIterator::Construct;
348 LoopSequence(std::unique_ptr<Construct> entry,
unsigned version,
349 bool allowAllLoops, SemanticsContext *semaCtx =
nullptr);
351 template <
typename R,
typename = std::enable_if_t<is_range_v<R>>>
352 void createChildrenFromRange(
const R &range) {
353 createChildrenFromRange(range.begin(), range.end());
356 std::unique_ptr<Construct> createConstructEntry(
357 const parser::ExecutionPartConstruct &code);
359 void createChildrenFromRange(
360 ExecutionPartIterator::IteratorType begin,
361 ExecutionPartIterator::IteratorType end);
366 WithReason<int64_t> calculateLength()
const;
367 WithReason<int64_t> getNestedLength()
const;
368 Depth calculateDepths()
const;
369 Depth getNestedDepths()
const;
370 WithReason<int64_t> calculateHeight()
const;
375 const parser::ExecutionPartConstruct *invalidIC_{
nullptr};
379 const parser::ExecutionPartConstruct *opaqueIC_{
nullptr};
385 WithReason<int64_t> length_;
394 WithReason<int64_t> height_;
399 std::unique_ptr<Construct> entry_;
400 std::vector<LoopSequence> children_;
401 SemanticsContext *semaCtx_{
nullptr};
410llvm::omp::TraitSet MapTraitSet(parser::OmpTraitSetSelectorName::Value name);
414llvm::omp::TraitSelector MapTraitSelector(
415 const parser::OmpTraitSelectorName &name, llvm::omp::TraitSet set);
418std::optional<bool> EvaluateUserCondition(
419 SemanticsContext &semaCtx,
const parser::ScalarExpr &scalarExpr);
422llvm::APInt *GetTraitScore(
423 const std::optional<parser::OmpTraitSelector::Properties> &props,
424 SemanticsContext &semaCtx, std::optional<llvm::APInt> &scoreStorage);
429void ProcessTraitProperties(llvm::omp::VariantMatchInfo &vmi,
430 llvm::omp::TraitSet set, llvm::omp::TraitSelector selector,
431 const std::optional<parser::OmpTraitSelector::Properties> &props,
432 llvm::APInt *scorePtr);
Definition char-block.h:28
Definition semantics.h:68
Definition parse-tree.h:2279
Definition parse-tree.h:2366
Definition parse-tree.h:559
Definition parse-tree.h:1736
Definition parse-tree.h:592
Definition parse-tree.h:5129
Definition parse-tree.h:3606
Definition parse-tree.h:5502
Definition parse-tree.h:3737
Definition openmp-utils.h:316
const LoopSequence * getNestedDoConcurrent() const
Definition openmp-utils.cpp:1677
A representation of a "because" message.
Definition openmp-utils.h:219
Definition openmp-utils.h:79
Definition openmp-utils.h:245
Definition openmp-utils.h:59