FLANG
Clauses.h
1//===-- Clauses.h -- OpenMP clause handling -------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8#ifndef FORTRAN_LOWER_OPENMP_CLAUSES_H
9#define FORTRAN_LOWER_OPENMP_CLAUSES_H
10
11#include "flang/Evaluate/expression.h"
12#include "flang/Evaluate/type.h"
13#include "flang/Parser/parse-tree.h"
14#include "flang/Semantics/expression.h"
15#include "flang/Semantics/semantics.h"
16#include "flang/Semantics/symbol.h"
17
18#include "llvm/ADT/STLExtras.h"
19#include "llvm/Frontend/OpenMP/ClauseT.h"
20#include "llvm/Frontend/OpenMP/OMP.h.inc"
21
22#include <optional>
23#include <type_traits>
24#include <utility>
25
26namespace Fortran::semantics {
27class Symbol;
28}
29
30namespace Fortran::lower::omp {
31using namespace Fortran;
32using SomeExpr = semantics::SomeExpr;
33using MaybeExpr = semantics::MaybeExpr;
34using TypeTy = evaluate::DynamicType;
35
36template <typename ExprTy>
38 // "symbol" is always non-null for id's of actual objects.
40 std::optional<ExprTy> designator;
41
42 bool operator==(const IdTyTemplate &other) const {
43 // If symbols are different, then the objects are different.
44 if (symbol != other.symbol)
45 return false;
46 if (symbol == nullptr)
47 return true;
48 // Equal symbols don't necessarily indicate identical objects,
49 // for example, a derived object component may use a single symbol,
50 // which will refer to different objects for different designators,
51 // e.g. a%c and b%c.
52 return designator == other.designator;
53 }
54
55 // Defining an "ordering" which allows types derived from this to be
56 // utilised in maps and other containers that require comparison
57 // operators for ordering
58 bool operator<(const IdTyTemplate &other) const {
59 return symbol < other.symbol;
60 }
61
62 operator bool() const { return symbol != nullptr; }
63};
64
65using ExprTy = SomeExpr;
66
67template <typename T>
68using List = tomp::ListT<T>;
69} // namespace Fortran::lower::omp
70
71// Specialization of the ObjectT template
72namespace tomp::type {
73template <>
74struct ObjectT<Fortran::lower::omp::IdTyTemplate<Fortran::lower::omp::ExprTy>,
75 Fortran::lower::omp::ExprTy> {
77 using ExprTy = Fortran::lower::omp::ExprTy;
78
79 IdTy id() const { return identity; }
80 Fortran::semantics::Symbol *sym() const { return identity.symbol; }
81 const std::optional<ExprTy> &ref() const { return identity.designator; }
82
83 bool operator<(const ObjectT<IdTy, ExprTy> &other) const {
84 return identity < other.identity;
85 }
86
87 IdTy identity;
88};
89} // namespace tomp::type
90
91namespace Fortran::lower::omp {
92using IdTy = IdTyTemplate<ExprTy>;
93}
94
95namespace std {
96template <>
97struct hash<Fortran::lower::omp::IdTy> {
98 size_t operator()(const Fortran::lower::omp::IdTy &id) const {
99 return static_cast<size_t>(reinterpret_cast<uintptr_t>(id.symbol));
100 }
101};
102} // namespace std
103
104namespace Fortran::lower::omp {
105using Object = tomp::ObjectT<IdTy, ExprTy>;
106using ObjectList = tomp::ObjectListT<IdTy, ExprTy>;
107
108Object makeObject(const parser::OmpObject &object,
110Object makeObject(const parser::Name &name,
112Object makeObject(const parser::Designator &dsg,
114Object makeObject(const parser::StructureComponent &comp,
116
117inline auto makeObjectFn(semantics::SemanticsContext &semaCtx) {
118 return [&](auto &&s) { return makeObject(s, semaCtx); };
119}
120
121template <typename T>
122SomeExpr makeExpr(T &&pftExpr, semantics::SemanticsContext &semaCtx) {
123 auto maybeExpr = evaluate::ExpressionAnalyzer(semaCtx).Analyze(pftExpr);
124 assert(maybeExpr);
125 return std::move(*maybeExpr);
126}
127
128inline auto makeExprFn(semantics::SemanticsContext &semaCtx) {
129 return [&](auto &&s) { return makeExpr(s, semaCtx); };
130}
131
132template <
133 typename ContainerTy, typename FunctionTy,
134 typename ElemTy = typename llvm::remove_cvref_t<ContainerTy>::value_type,
135 typename ResultTy = std::invoke_result_t<FunctionTy, ElemTy>>
136List<ResultTy> makeList(ContainerTy &&container, FunctionTy &&func) {
137 List<ResultTy> v;
138 llvm::transform(container, std::back_inserter(v), func);
139 return v;
140}
141
142inline ObjectList makeObjects(const parser::OmpObjectList &objects,
143 semantics::SemanticsContext &semaCtx) {
144 return makeList(objects.v, makeObjectFn(semaCtx));
145}
146
147ObjectList makeObjects(const parser::OmpArgumentList &objects,
148 semantics::SemanticsContext &semaCtx);
149
150template <typename FuncTy, //
151 typename ArgTy, //
152 typename ResultTy = std::invoke_result_t<FuncTy, ArgTy>>
153std::optional<ResultTy> maybeApply(FuncTy &&func,
154 const std::optional<ArgTy> &arg) {
155 if (!arg)
156 return std::nullopt;
157 return func(*arg);
158}
159
160template < //
161 typename FuncTy, //
162 typename ArgTy, //
163 typename ResultTy =
164 std::invoke_result_t<FuncTy, decltype(std::declval<ArgTy>().v)>>
165std::optional<ResultTy> maybeApplyToV(FuncTy &&func, const ArgTy *arg) {
166 if (!arg)
167 return std::nullopt;
168 return func(arg->v);
169}
170
171std::optional<Object> getBaseObject(const Object &object,
172 semantics::SemanticsContext &semaCtx);
173
174namespace clause {
175using Range = tomp::type::RangeT<ExprTy>;
176using Mapper = tomp::type::MapperT<IdTy, ExprTy>;
177using Iterator = tomp::type::IteratorT<TypeTy, IdTy, ExprTy>;
178using IteratorSpecifier = tomp::type::IteratorSpecifierT<TypeTy, IdTy, ExprTy>;
179using DefinedOperator = tomp::type::DefinedOperatorT<IdTy, ExprTy>;
180using ProcedureDesignator = tomp::type::ProcedureDesignatorT<IdTy, ExprTy>;
181using ReductionOperator = tomp::type::ReductionIdentifierT<IdTy, ExprTy>;
182using ReductionOperatorList = List<ReductionOperator>;
183using DependenceType = tomp::type::DependenceType;
184using Prescriptiveness = tomp::type::Prescriptiveness;
185
186// "Requires" clauses are handled early on, and the aggregated information
187// is stored in the Symbol details of modules, programs, and subprograms.
188// These clauses are still handled here to cover all alternatives in the
189// main clause variant.
190
191using Absent = tomp::clause::AbsentT<TypeTy, IdTy, ExprTy>;
192using AcqRel = tomp::clause::AcqRelT<TypeTy, IdTy, ExprTy>;
193using Acquire = tomp::clause::AcquireT<TypeTy, IdTy, ExprTy>;
194using AdjustArgs = tomp::clause::AdjustArgsT<TypeTy, IdTy, ExprTy>;
195using Affinity = tomp::clause::AffinityT<TypeTy, IdTy, ExprTy>;
196using Aligned = tomp::clause::AlignedT<TypeTy, IdTy, ExprTy>;
197using Align = tomp::clause::AlignT<TypeTy, IdTy, ExprTy>;
198using Allocate = tomp::clause::AllocateT<TypeTy, IdTy, ExprTy>;
199using Allocator = tomp::clause::AllocatorT<TypeTy, IdTy, ExprTy>;
200using AppendArgs = tomp::clause::AppendArgsT<TypeTy, IdTy, ExprTy>;
201using AtomicDefaultMemOrder =
202 tomp::clause::AtomicDefaultMemOrderT<TypeTy, IdTy, ExprTy>;
203using At = tomp::clause::AtT<TypeTy, IdTy, ExprTy>;
204using Bind = tomp::clause::BindT<TypeTy, IdTy, ExprTy>;
205using Capture = tomp::clause::CaptureT<TypeTy, IdTy, ExprTy>;
206using Collapse = tomp::clause::CollapseT<TypeTy, IdTy, ExprTy>;
207using Compare = tomp::clause::CompareT<TypeTy, IdTy, ExprTy>;
208using Contains = tomp::clause::ContainsT<TypeTy, IdTy, ExprTy>;
209using Copyin = tomp::clause::CopyinT<TypeTy, IdTy, ExprTy>;
210using Copyprivate = tomp::clause::CopyprivateT<TypeTy, IdTy, ExprTy>;
211using Defaultmap = tomp::clause::DefaultmapT<TypeTy, IdTy, ExprTy>;
212using Default = tomp::clause::DefaultT<TypeTy, IdTy, ExprTy>;
213using Depend = tomp::clause::DependT<TypeTy, IdTy, ExprTy>;
214using Destroy = tomp::clause::DestroyT<TypeTy, IdTy, ExprTy>;
215using Detach = tomp::clause::DetachT<TypeTy, IdTy, ExprTy>;
216using Device = tomp::clause::DeviceT<TypeTy, IdTy, ExprTy>;
217using DeviceType = tomp::clause::DeviceTypeT<TypeTy, IdTy, ExprTy>;
218using DistSchedule = tomp::clause::DistScheduleT<TypeTy, IdTy, ExprTy>;
219using Doacross = tomp::clause::DoacrossT<TypeTy, IdTy, ExprTy>;
220using DynamicAllocators =
221 tomp::clause::DynamicAllocatorsT<TypeTy, IdTy, ExprTy>;
222using DynGroupprivate = tomp::clause::DynGroupprivateT<TypeTy, IdTy, ExprTy>;
223using Enter = tomp::clause::EnterT<TypeTy, IdTy, ExprTy>;
224using Exclusive = tomp::clause::ExclusiveT<TypeTy, IdTy, ExprTy>;
225using Fail = tomp::clause::FailT<TypeTy, IdTy, ExprTy>;
226using Filter = tomp::clause::FilterT<TypeTy, IdTy, ExprTy>;
227using Final = tomp::clause::FinalT<TypeTy, IdTy, ExprTy>;
228using Firstprivate = tomp::clause::FirstprivateT<TypeTy, IdTy, ExprTy>;
229using From = tomp::clause::FromT<TypeTy, IdTy, ExprTy>;
230using Full = tomp::clause::FullT<TypeTy, IdTy, ExprTy>;
231using Grainsize = tomp::clause::GrainsizeT<TypeTy, IdTy, ExprTy>;
232using GraphId = tomp::clause::GraphIdT<TypeTy, IdTy, ExprTy>;
233using GraphReset = tomp::clause::GraphResetT<TypeTy, IdTy, ExprTy>;
234using HasDeviceAddr = tomp::clause::HasDeviceAddrT<TypeTy, IdTy, ExprTy>;
235using Hint = tomp::clause::HintT<TypeTy, IdTy, ExprTy>;
236using Holds = tomp::clause::HoldsT<TypeTy, IdTy, ExprTy>;
237using If = tomp::clause::IfT<TypeTy, IdTy, ExprTy>;
238using Inbranch = tomp::clause::InbranchT<TypeTy, IdTy, ExprTy>;
239using Inclusive = tomp::clause::InclusiveT<TypeTy, IdTy, ExprTy>;
240using Indirect = tomp::clause::IndirectT<TypeTy, IdTy, ExprTy>;
241using Init = tomp::clause::InitT<TypeTy, IdTy, ExprTy>;
242using Initializer = tomp::clause::InitializerT<TypeTy, IdTy, ExprTy>;
243using InReduction = tomp::clause::InReductionT<TypeTy, IdTy, ExprTy>;
244using IsDevicePtr = tomp::clause::IsDevicePtrT<TypeTy, IdTy, ExprTy>;
245using Lastprivate = tomp::clause::LastprivateT<TypeTy, IdTy, ExprTy>;
246using LoopRange = tomp::clause::LoopRangeT<TypeTy, IdTy, ExprTy>;
247using Linear = tomp::clause::LinearT<TypeTy, IdTy, ExprTy>;
248using Link = tomp::clause::LinkT<TypeTy, IdTy, ExprTy>;
249using Map = tomp::clause::MapT<TypeTy, IdTy, ExprTy>;
250using Match = tomp::clause::MatchT<TypeTy, IdTy, ExprTy>;
251using Mergeable = tomp::clause::MergeableT<TypeTy, IdTy, ExprTy>;
252using Message = tomp::clause::MessageT<TypeTy, IdTy, ExprTy>;
253using NoOpenmp = tomp::clause::NoOpenmpT<TypeTy, IdTy, ExprTy>;
254using NoOpenmpRoutines = tomp::clause::NoOpenmpRoutinesT<TypeTy, IdTy, ExprTy>;
255using NoOpenmpConstructs =
256 tomp::clause::NoOpenmpConstructsT<TypeTy, IdTy, ExprTy>;
257using NoParallelism = tomp::clause::NoParallelismT<TypeTy, IdTy, ExprTy>;
258using Nocontext = tomp::clause::NocontextT<TypeTy, IdTy, ExprTy>;
259using Nogroup = tomp::clause::NogroupT<TypeTy, IdTy, ExprTy>;
260using Nontemporal = tomp::clause::NontemporalT<TypeTy, IdTy, ExprTy>;
261using Notinbranch = tomp::clause::NotinbranchT<TypeTy, IdTy, ExprTy>;
262using Novariants = tomp::clause::NovariantsT<TypeTy, IdTy, ExprTy>;
263using Nowait = tomp::clause::NowaitT<TypeTy, IdTy, ExprTy>;
264using NumTasks = tomp::clause::NumTasksT<TypeTy, IdTy, ExprTy>;
265using NumTeams = tomp::clause::NumTeamsT<TypeTy, IdTy, ExprTy>;
266using NumThreads = tomp::clause::NumThreadsT<TypeTy, IdTy, ExprTy>;
267using OmpxAttribute = tomp::clause::OmpxAttributeT<TypeTy, IdTy, ExprTy>;
268using OmpxBare = tomp::clause::OmpxBareT<TypeTy, IdTy, ExprTy>;
269using OmpxDynCgroupMem = tomp::clause::OmpxDynCgroupMemT<TypeTy, IdTy, ExprTy>;
270using Ordered = tomp::clause::OrderedT<TypeTy, IdTy, ExprTy>;
271using Order = tomp::clause::OrderT<TypeTy, IdTy, ExprTy>;
272using Otherwise = tomp::clause::OtherwiseT<TypeTy, IdTy, ExprTy>;
273using Partial = tomp::clause::PartialT<TypeTy, IdTy, ExprTy>;
274using Priority = tomp::clause::PriorityT<TypeTy, IdTy, ExprTy>;
275using Private = tomp::clause::PrivateT<TypeTy, IdTy, ExprTy>;
276using ProcBind = tomp::clause::ProcBindT<TypeTy, IdTy, ExprTy>;
277using Read = tomp::clause::ReadT<TypeTy, IdTy, ExprTy>;
278using Reduction = tomp::clause::ReductionT<TypeTy, IdTy, ExprTy>;
279using Relaxed = tomp::clause::RelaxedT<TypeTy, IdTy, ExprTy>;
280using Release = tomp::clause::ReleaseT<TypeTy, IdTy, ExprTy>;
281using Replayable = tomp::clause::ReplayableT<TypeTy, IdTy, ExprTy>;
282using ReverseOffload = tomp::clause::ReverseOffloadT<TypeTy, IdTy, ExprTy>;
283using Safelen = tomp::clause::SafelenT<TypeTy, IdTy, ExprTy>;
284using Schedule = tomp::clause::ScheduleT<TypeTy, IdTy, ExprTy>;
285using SeqCst = tomp::clause::SeqCstT<TypeTy, IdTy, ExprTy>;
286using Severity = tomp::clause::SeverityT<TypeTy, IdTy, ExprTy>;
287using Shared = tomp::clause::SharedT<TypeTy, IdTy, ExprTy>;
288using Simdlen = tomp::clause::SimdlenT<TypeTy, IdTy, ExprTy>;
289using Simd = tomp::clause::SimdT<TypeTy, IdTy, ExprTy>;
290using Sizes = tomp::clause::SizesT<TypeTy, IdTy, ExprTy>;
291using Permutation = tomp::clause::PermutationT<TypeTy, IdTy, ExprTy>;
292using TaskReduction = tomp::clause::TaskReductionT<TypeTy, IdTy, ExprTy>;
293using ThreadLimit = tomp::clause::ThreadLimitT<TypeTy, IdTy, ExprTy>;
294using Threads = tomp::clause::ThreadsT<TypeTy, IdTy, ExprTy>;
295using Transparent = tomp::clause::TransparentT<TypeTy, IdTy, ExprTy>;
296using To = tomp::clause::ToT<TypeTy, IdTy, ExprTy>;
297using UnifiedAddress = tomp::clause::UnifiedAddressT<TypeTy, IdTy, ExprTy>;
298using UnifiedSharedMemory =
299 tomp::clause::UnifiedSharedMemoryT<TypeTy, IdTy, ExprTy>;
300using SelfMaps = tomp::clause::SelfMapsT<TypeTy, IdTy, ExprTy>;
301using Uniform = tomp::clause::UniformT<TypeTy, IdTy, ExprTy>;
302using Unknown = tomp::clause::UnknownT<TypeTy, IdTy, ExprTy>;
303using Untied = tomp::clause::UntiedT<TypeTy, IdTy, ExprTy>;
304using Update = tomp::clause::UpdateT<TypeTy, IdTy, ExprTy>;
305using UseDeviceAddr = tomp::clause::UseDeviceAddrT<TypeTy, IdTy, ExprTy>;
306using UseDevicePtr = tomp::clause::UseDevicePtrT<TypeTy, IdTy, ExprTy>;
307using UsesAllocators = tomp::clause::UsesAllocatorsT<TypeTy, IdTy, ExprTy>;
308using Use = tomp::clause::UseT<TypeTy, IdTy, ExprTy>;
309using Weak = tomp::clause::WeakT<TypeTy, IdTy, ExprTy>;
310using When = tomp::clause::WhenT<TypeTy, IdTy, ExprTy>;
311using Write = tomp::clause::WriteT<TypeTy, IdTy, ExprTy>;
312} // namespace clause
313
314using tomp::type::operator==;
315
317 using WrapperTrait = std::true_type;
318 llvm::omp::CancellationConstructType v;
319};
320struct Depobj {
321 using EmptyTrait = std::true_type;
322};
323struct Flush {
324 using EmptyTrait = std::true_type;
325};
327 using EmptyTrait = std::true_type;
328};
330 using EmptyTrait = std::true_type;
331};
333 using EmptyTrait = std::true_type;
334};
335
336using ClauseBase = tomp::ClauseT<TypeTy, IdTy, ExprTy,
337 // Extras...
340
341struct Clause : public ClauseBase {
342 Clause(ClauseBase &&base, const parser::CharBlock source = {})
343 : ClauseBase(std::move(base)), source(source) {}
344 // "source" will be ignored by tomp::type::operator==.
345 parser::CharBlock source;
346};
347
348template <typename Specific>
349Clause makeClause(llvm::omp::Clause id, Specific &&specific,
350 parser::CharBlock source = {}) {
351 return Clause(typename Clause::BaseT{id, specific}, source);
352}
353
354Clause makeClause(const parser::OmpClause &cls,
356
357List<Clause> makeClauses(const parser::OmpClauseList &clauses,
359
360bool transferLocations(const List<Clause> &from, List<Clause> &to);
361} // namespace Fortran::lower::omp
362
363#endif // FORTRAN_LOWER_OPENMP_CLAUSES_H
Definition char-block.h:28
Definition semantics.h:67
Definition symbol.h:781
Definition bit-population-count.h:20
Definition Clauses.h:341
Definition Clauses.h:320
Definition Clauses.h:323
Definition Clauses.h:332
Definition Clauses.h:37
Definition Clauses.h:326
Definition Clauses.h:329
Definition parse-tree.h:1858
Definition parse-tree.h:580
Definition parse-tree.h:4807
Definition parse-tree.h:4791
Definition parse-tree.h:3508
Definition parse-tree.h:1888