FLANG
check-omp-structure.h
1//===-- lib/Semantics/check-omp-structure.h ---------------------*- C++ -*-===//
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
9// OpenMP structure validity check list
10// 1. invalid clauses on directive
11// 2. invalid repeated clauses on directive
12// 3. TODO: invalid nesting of regions
13
14#ifndef FORTRAN_SEMANTICS_CHECK_OMP_STRUCTURE_H_
15#define FORTRAN_SEMANTICS_CHECK_OMP_STRUCTURE_H_
16
17#include "check-directive-structure.h"
18#include "flang/Common/enum-set.h"
19#include "flang/Parser/parse-tree.h"
20#include "flang/Semantics/openmp-directive-sets.h"
21#include "flang/Semantics/semantics.h"
22
23using OmpClauseSet =
25
26#define GEN_FLANG_DIRECTIVE_CLAUSE_SETS
27#include "llvm/Frontend/OpenMP/OMP.inc"
28
29namespace llvm {
30namespace omp {
31static OmpClauseSet privateSet{
32 Clause::OMPC_private, Clause::OMPC_firstprivate, Clause::OMPC_lastprivate};
33static OmpClauseSet privateReductionSet{
34 OmpClauseSet{Clause::OMPC_reduction} | privateSet};
35// omp.td cannot differentiate allowed/not allowed clause list for few
36// directives for fortran. nowait is not allowed on begin directive clause list
37// for below list of directives. Directives with conflicting list of clauses are
38// included in below list.
39static const OmpDirectiveSet noWaitClauseNotAllowedSet{
40 Directive::OMPD_do,
41 Directive::OMPD_do_simd,
42 Directive::OMPD_sections,
43 Directive::OMPD_single,
44 Directive::OMPD_workshare,
45};
46} // namespace omp
47} // namespace llvm
48
49namespace Fortran::semantics {
50struct AnalyzedCondStmt;
51
52namespace omp {
53struct LoopSequence;
54}
55
56// Mapping from 'Symbol' to 'Source' to keep track of the variables
57// used in multiple clauses
58using SymbolSourceMap = std::multimap<const Symbol *, parser::CharBlock>;
59// Multimap to check the triple <current_dir, enclosing_dir, enclosing_clause>
60using DirectivesClauseTriple = std::multimap<llvm::omp::Directive,
61 std::pair<llvm::omp::Directive, const OmpClauseSet>>;
62
63using OmpStructureCheckerBase = DirectiveStructureChecker<llvm::omp::Directive,
64 llvm::omp::Clause, parser::OmpClause, llvm::omp::Clause_enumSize>;
65
66class OmpStructureChecker : public OmpStructureCheckerBase {
67public:
68 using Base = OmpStructureCheckerBase;
69
70 OmpStructureChecker(SemanticsContext &context);
71
72 using llvmOmpClause = const llvm::omp::Clause;
73
74 bool Enter(const parser::MainProgram &);
75 void Leave(const parser::MainProgram &);
76 bool Enter(const parser::BlockData &);
77 void Leave(const parser::BlockData &);
78 bool Enter(const parser::Module &);
79 void Leave(const parser::Module &);
80 bool Enter(const parser::Submodule &);
81 void Leave(const parser::Submodule &);
82 bool Enter(const parser::SubroutineStmt &);
83 bool Enter(const parser::EndSubroutineStmt &);
84 bool Enter(const parser::FunctionStmt &);
85 bool Enter(const parser::EndFunctionStmt &);
86 bool Enter(const parser::BlockConstruct &);
87 void Leave(const parser::BlockConstruct &);
88
89 void Enter(const parser::SpecificationPart &);
90 void Leave(const parser::SpecificationPart &);
91 void Enter(const parser::ExecutionPart &);
92 void Leave(const parser::ExecutionPart &);
93
94 void Enter(const parser::OpenMPConstruct &);
95 void Leave(const parser::OpenMPConstruct &);
96 void Enter(const parser::OpenMPDeclarativeConstruct &);
97 void Leave(const parser::OpenMPDeclarativeConstruct &);
98
99 void Enter(const parser::OpenMPMisplacedEndDirective &);
100 void Leave(const parser::OpenMPMisplacedEndDirective &);
101 void Enter(const parser::OpenMPInvalidDirective &);
102 void Leave(const parser::OpenMPInvalidDirective &);
103
104 void Enter(const parser::OpenMPLoopConstruct &);
105 void Leave(const parser::OpenMPLoopConstruct &);
106 void Enter(const parser::OmpEndLoopDirective &);
107 void Leave(const parser::OmpEndLoopDirective &);
108
109 void Enter(const parser::OpenMPAssumeConstruct &);
110 void Leave(const parser::OpenMPAssumeConstruct &);
111 void Enter(const parser::OpenMPDeclarativeAssumes &);
112 void Leave(const parser::OpenMPDeclarativeAssumes &);
113 void Enter(const parser::OpenMPInteropConstruct &);
114 void Leave(const parser::OpenMPInteropConstruct &);
115 void Enter(const parser::OmpBlockConstruct &);
116 void Leave(const parser::OmpBlockConstruct &);
117 void Leave(const parser::OmpBeginDirective &);
118 void Enter(const parser::OmpEndDirective &);
119 void Leave(const parser::OmpEndDirective &);
120
121 void Enter(const parser::OpenMPSectionsConstruct &);
122 void Leave(const parser::OpenMPSectionsConstruct &);
123 void Enter(const parser::OmpEndSectionsDirective &);
124 void Leave(const parser::OmpEndSectionsDirective &);
125
126 void Enter(const parser::OmpDeclareVariantDirective &);
127 void Leave(const parser::OmpDeclareVariantDirective &);
128 void Enter(const parser::OpenMPDeclareSimdConstruct &);
129 void Leave(const parser::OpenMPDeclareSimdConstruct &);
130 void Enter(const parser::OmpAllocateDirective &);
131 void Leave(const parser::OmpAllocateDirective &);
132 void Enter(const parser::OpenMPDeclareMapperConstruct &);
133 void Leave(const parser::OpenMPDeclareMapperConstruct &);
134 void Enter(const parser::OpenMPDeclareReductionConstruct &);
135 void Leave(const parser::OpenMPDeclareReductionConstruct &);
136 void Enter(const parser::OpenMPDeclareTargetConstruct &);
137 void Leave(const parser::OpenMPDeclareTargetConstruct &);
138 void Enter(const parser::OpenMPDepobjConstruct &);
139 void Leave(const parser::OpenMPDepobjConstruct &);
140 void Enter(const parser::OpenMPDispatchConstruct &);
141 void Leave(const parser::OpenMPDispatchConstruct &);
142 void Enter(const parser::OmpErrorDirective &);
143 void Leave(const parser::OmpErrorDirective &);
144 void Enter(const parser::OmpNothingDirective &);
145 void Leave(const parser::OmpNothingDirective &);
146 void Enter(const parser::OpenMPAllocatorsConstruct &);
147 void Leave(const parser::OpenMPAllocatorsConstruct &);
148 void Enter(const parser::OpenMPRequiresConstruct &);
149 void Leave(const parser::OpenMPRequiresConstruct &);
150 void Enter(const parser::OpenMPGroupprivate &);
151 void Leave(const parser::OpenMPGroupprivate &);
152 void Enter(const parser::OpenMPThreadprivate &);
153 void Leave(const parser::OpenMPThreadprivate &);
154
155 void Enter(const parser::OpenMPSimpleStandaloneConstruct &);
156 void Leave(const parser::OpenMPSimpleStandaloneConstruct &);
157 void Enter(const parser::OpenMPFlushConstruct &);
158 void Leave(const parser::OpenMPFlushConstruct &);
159 void Enter(const parser::OpenMPCancelConstruct &);
160 void Leave(const parser::OpenMPCancelConstruct &);
163 void Enter(const parser::OpenMPCriticalConstruct &);
164 void Leave(const parser::OpenMPCriticalConstruct &);
165 void Enter(const parser::OpenMPAtomicConstruct &);
166 void Leave(const parser::OpenMPAtomicConstruct &);
167
168 void Leave(const parser::OmpClauseList &);
169 void Enter(const parser::OmpClause &);
170
171 void Enter(const parser::DoConstruct &);
172 void Leave(const parser::DoConstruct &);
173
174 void Enter(const parser::OmpDirectiveSpecification &);
175 void Leave(const parser::OmpDirectiveSpecification &);
176
177 void Enter(const parser::OmpMetadirectiveDirective &);
178 void Leave(const parser::OmpMetadirectiveDirective &);
179
180 void Enter(const parser::OmpContextSelector &);
181 void Leave(const parser::OmpContextSelector &);
182
183#define GEN_FLANG_CLAUSE_CHECK_ENTER
184#include "llvm/Frontend/OpenMP/OMP.inc"
185
186private:
187 // Most of these functions are defined in check-omp-structure.cpp, but
188 // some groups have their own files.
189
190 // check-omp-atomic.cpp
191 void CheckStorageOverlap(const evaluate::Expr<evaluate::SomeType> &,
193 void ErrorShouldBeVariable(const MaybeExpr &expr, parser::CharBlock source);
194 void CheckAtomicType(SymbolRef sym, parser::CharBlock source,
195 std::string_view name, bool checkTypeOnPointer = true);
196 void CheckAtomicVariable(const evaluate::Expr<evaluate::SomeType> &,
197 parser::CharBlock, bool checkTypeOnPointer = true);
198 std::pair<const parser::ExecutionPartConstruct *,
200 CheckUpdateCapture(const parser::ExecutionPartConstruct *ec1,
202 void CheckAtomicCaptureAssignment(const evaluate::Assignment &capture,
203 const SomeExpr &atom, parser::CharBlock source);
204 void CheckAtomicReadAssignment(
205 const evaluate::Assignment &read, parser::CharBlock source);
206 void CheckAtomicWriteAssignment(
207 const evaluate::Assignment &write, parser::CharBlock source);
208 std::optional<evaluate::Assignment> CheckAtomicUpdateAssignment(
209 const evaluate::Assignment &update, parser::CharBlock source);
210 std::pair<bool, bool> CheckAtomicUpdateAssignmentRhs(const SomeExpr &atom,
211 const SomeExpr &rhs, parser::CharBlock source, bool suppressDiagnostics);
212 void CheckAtomicConditionalUpdateAssignment(const SomeExpr &cond,
213 parser::CharBlock condSource, const evaluate::Assignment &assign,
214 parser::CharBlock assignSource);
215 void CheckAtomicConditionalUpdateStmt(
216 const AnalyzedCondStmt &update, parser::CharBlock source);
217 void CheckAtomicUpdateOnly(const parser::OpenMPAtomicConstruct &x,
218 const parser::Block &body, parser::CharBlock source);
219 void CheckAtomicConditionalUpdate(const parser::OpenMPAtomicConstruct &x,
220 const parser::Block &body, parser::CharBlock source);
221 void CheckAtomicUpdateCapture(const parser::OpenMPAtomicConstruct &x,
222 const parser::Block &body, parser::CharBlock source);
223 void CheckAtomicConditionalUpdateCapture(
224 const parser::OpenMPAtomicConstruct &x, const parser::Block &body,
225 parser::CharBlock source);
226 void CheckAtomicRead(const parser::OpenMPAtomicConstruct &x);
227 void CheckAtomicWrite(const parser::OpenMPAtomicConstruct &x);
228 void CheckAtomicUpdate(const parser::OpenMPAtomicConstruct &x);
229
230 // check-omp-loop.cpp
231 void HasInvalidDistributeNesting(const parser::OpenMPLoopConstruct &x);
232 void HasInvalidLoopBinding(const parser::OpenMPLoopConstruct &x);
233 void CheckSIMDNest(const parser::OpenMPConstruct &x);
234 void CheckRectangularNest(const parser::OmpDirectiveSpecification &spec,
235 const omp::LoopSequence &nest);
236 void CheckNestedConstruct(const parser::OpenMPLoopConstruct &x);
237 const parser::Name GetLoopIndex(const parser::DoConstruct *x);
238 void SetLoopInfo(const parser::OpenMPLoopConstruct &x);
239 void CheckIterationVariableType(const parser::OpenMPLoopConstruct &x);
240 std::int64_t GetOrdCollapseLevel(const parser::OpenMPLoopConstruct &x);
241 void CheckAssociatedLoopConstraints(const parser::OpenMPLoopConstruct &x);
242 void CheckScanModifier(const parser::OmpClause::Reduction &x);
243 void CheckDistLinear(const parser::OpenMPLoopConstruct &x);
244
245 // check-omp-metadirective.cpp
246 const std::list<parser::OmpTraitProperty> &GetTraitPropertyList(
248 std::optional<llvm::omp::Clause> GetClauseFromProperty(
250
251 void CheckTraitSelectorList(const std::list<parser::OmpTraitSelector> &);
252 void CheckTraitSetSelector(const parser::OmpTraitSetSelector &);
253 void CheckTraitScore(const parser::OmpTraitScore &);
254 bool VerifyTraitPropertyLists(
256 void CheckTraitSelector(
258 void CheckTraitADMO(
260 void CheckTraitCondition(
262 void CheckTraitDeviceNum(
264 void CheckTraitRequires(
266 void CheckTraitSimd(
268
269 // check-omp-structure.cpp
270 bool IsAllowedClause(llvm::omp::Clause clauseId);
271 bool CheckAllowedClause(llvmOmpClause clause);
272 void CheckVariableListItem(const SymbolSourceMap &symbols);
273 void CheckDirectiveSpelling(
274 parser::CharBlock spelling, llvm::omp::Directive id);
275 void AnalyzeObject(const parser::OmpObject &object);
276 void AnalyzeObjects(const parser::OmpObjectList &objects);
277 void CheckMultipleOccurrence(semantics::UnorderedSymbolSet &listVars,
278 const std::list<parser::Name> &nameList, const parser::CharBlock &item,
279 const std::string &clauseName);
280 void CheckMultListItems();
281 void CheckStructureComponent(
282 const parser::OmpObjectList &objects, llvm::omp::Clause clauseId);
283 bool HasInvalidWorksharingNesting(
284 const parser::OmpDirectiveName &name, const OmpDirectiveSet &);
285
286 bool IsCloselyNestedRegion(const OmpDirectiveSet &set);
287 bool IsNestedInDirective(llvm::omp::Directive directive);
288 bool IsCombinedParallelWorksharing(llvm::omp::Directive directive) const;
289 bool InTargetRegion();
290 void HasInvalidTeamsNesting(
291 const llvm::omp::Directive &dir, const parser::CharBlock &source);
292 bool HasRequires(llvm::omp::Clause req);
293 void CheckAllowedMapTypes(
294 parser::OmpMapType::Value, llvm::ArrayRef<parser::OmpMapType::Value>);
295
296 llvm::StringRef getClauseName(llvm::omp::Clause clause) override;
297 llvm::StringRef getDirectiveName(llvm::omp::Directive directive) override;
298
299 template < //
300 typename LessTy, typename RangeTy,
301 typename IterTy = decltype(std::declval<RangeTy>().begin())>
302 std::optional<IterTy> FindDuplicate(RangeTy &&);
303
304 void CheckDependList(const parser::DataRef &);
305 void CheckDoacross(const parser::OmpDoacross &doa);
306 void CheckDimsModifier(parser::CharBlock source, size_t numValues,
307 const parser::OmpDimsModifier &x);
308 bool IsDataRefTypeParamInquiry(const parser::DataRef *dataRef);
309 void CheckVarIsNotPartOfAnotherVar(const parser::CharBlock &source,
310 const parser::OmpObject &obj, llvm::StringRef clause = "");
311 void CheckVarIsNotPartOfAnotherVar(const parser::CharBlock &source,
312 const parser::OmpObjectList &objList, llvm::StringRef clause = "");
313 void CheckThreadprivateOrDeclareTargetVar(const parser::Designator &);
314 void CheckThreadprivateOrDeclareTargetVar(const parser::Name &);
315 void CheckThreadprivateOrDeclareTargetVar(const parser::OmpObject &);
316 void CheckThreadprivateOrDeclareTargetVar(const parser::OmpObjectList &);
317 void CheckSymbolName(
318 const parser::CharBlock &source, const parser::OmpObject &object);
319 void CheckSymbolNames(
320 const parser::CharBlock &source, const parser::OmpObjectList &objList);
321 void CheckIntentInPointer(SymbolSourceMap &, const llvm::omp::Clause);
322 void CheckAssumedSizeArray(SymbolSourceMap &, const llvm::omp::Clause);
323 void CheckProcedurePointer(SymbolSourceMap &, const llvm::omp::Clause);
324 void CheckCrayPointee(const parser::OmpObjectList &objectList,
325 llvm::StringRef clause, bool suggestToUseCrayPointer = true);
326 void GetSymbolsInObjectList(const parser::OmpObjectList &, SymbolSourceMap &);
327 void CheckDefinableObjects(SymbolSourceMap &, const llvm::omp::Clause);
328 void CheckCopyingPolymorphicAllocatable(
329 SymbolSourceMap &, const llvm::omp::Clause);
330 void CheckPrivateSymbolsInOuterCxt(
331 SymbolSourceMap &, DirectivesClauseTriple &, const llvm::omp::Clause);
332 void CheckIsLoopIvPartOfClause(
333 llvmOmpClause clause, const parser::OmpObjectList &ompObjectList);
334 bool CheckTargetBlockOnlyTeams(const parser::Block &);
335 void CheckWorkshareBlockStmts(const parser::Block &, parser::CharBlock);
336 void CheckWorkdistributeBlockStmts(const parser::Block &, parser::CharBlock);
337 void CheckIndividualAllocateDirective(
338 const parser::OmpAllocateDirective &x, bool isExecutable);
339 void CheckExecutableAllocateDirective(const parser::OmpAllocateDirective &x);
340
341 void CheckIteratorRange(const parser::OmpIteratorSpecifier &x);
342 void CheckIteratorModifier(const parser::OmpIterator &x);
343
344 void CheckTargetNest(const parser::OpenMPConstruct &x);
345 void CheckTargetUpdate();
346 void CheckTaskgraph(const parser::OmpBlockConstruct &x);
347 void CheckDependenceType(const parser::OmpDependenceType::Value &x);
348 void CheckTaskDependenceType(const parser::OmpTaskDependenceType::Value &x);
349 std::optional<llvm::omp::Directive> GetCancelType(
350 llvm::omp::Directive cancelDir, const parser::CharBlock &cancelSource,
351 const std::optional<parser::OmpClauseList> &maybeClauses);
352 void CheckCancellationNest(
353 const parser::CharBlock &source, llvm::omp::Directive type);
354 void CheckReductionObjects(
355 const parser::OmpObjectList &objects, llvm::omp::Clause clauseId);
356 bool CheckReductionOperator(const parser::OmpReductionIdentifier &ident,
357 parser::CharBlock source, llvm::omp::Clause clauseId);
358 void CheckReductionObjectTypes(const parser::OmpObjectList &objects,
359 const parser::OmpReductionIdentifier &ident);
360 void CheckReductionModifier(const parser::OmpReductionModifier &);
361 void CheckLastprivateModifier(const parser::OmpLastprivateModifier &);
362 void CheckMasterNesting(const parser::OmpBlockConstruct &x);
363 void ChecksOnOrderedAsBlock();
364 void CheckBarrierNesting(const parser::OpenMPSimpleStandaloneConstruct &x);
365 void CheckScan(const parser::OpenMPSimpleStandaloneConstruct &x);
366 void ChecksOnOrderedAsStandalone();
367 void CheckOrderedDependClause(std::optional<std::int64_t> orderedValue);
368 void CheckReductionArraySection(
369 const parser::OmpObjectList &ompObjectList, llvm::omp::Clause clauseId);
370 void CheckArraySection(const parser::ArrayElement &arrayElement,
371 const parser::Name &name, const llvm::omp::Clause clause);
372 void CheckLastPartRefForArraySection(
373 const parser::Designator &designator, llvm::omp::Clause clauseId);
374 void CheckSharedBindingInOuterContext(
375 const parser::OmpObjectList &ompObjectList);
376 void CheckIfContiguous(const parser::OmpObject &object);
377 const parser::Name *GetObjectName(const parser::OmpObject &object);
378 void CheckInitOnDepobj(const parser::OpenMPDepobjConstruct &depobj,
379 const parser::OmpClause &initClause);
380 void CheckAllowedRequiresClause(llvmOmpClause clause);
381 void AddEndDirectiveClauses(const parser::OmpClauseList &clauses);
382
383 void EnterDirectiveNest(const int index) { directiveNest_[index]++; }
384 void ExitDirectiveNest(const int index) { directiveNest_[index]--; }
385 int GetDirectiveNest(const int index) { return directiveNest_[index]; }
386
387 bool deviceConstructFound_{false};
388 enum directiveNestType : int {
389 SIMDNest,
390 TargetBlockOnlyTeams,
391 TargetNest,
392 DeclarativeNest,
393 ContextSelectorNest,
394 MetadirectiveNest,
395 LastType = MetadirectiveNest,
396 };
397 int directiveNest_[LastType + 1] = {0};
398
399 int allocateDirectiveLevel_{0};
400 parser::CharBlock visitedAtomicSource_;
401 SymbolSourceMap deferredNonVariables_;
402
403 using LoopConstruct = std::variant<const parser::DoConstruct *,
405 std::vector<LoopConstruct> loopStack_;
406 // Scopes for scoping units.
407 std::vector<const Scope *> scopeStack_;
408 // Stack of directive specifications (except for SECTION).
409 // This is to allow visitor functions to see all specified clauses, since
410 // they are only recorded in DirContext as they are processed.
411 std::vector<const parser::OmpDirectiveSpecification *> dirStack_;
412
413 enum class PartKind : int {
414 // There are also other "parts", such as internal-subprogram-part, etc,
415 // but we're keeping track of these two for now.
416 SpecificationPart,
417 ExecutionPart,
418 };
419 std::vector<PartKind> partStack_;
420};
421
424template <typename LessTy, typename RangeTy, typename IterTy>
425std::optional<IterTy> OmpStructureChecker::FindDuplicate(RangeTy &&range) {
426 // Deal with iterators, since the actual elements may be rvalues (i.e.
427 // have no addresses), for example with custom-constructed ranges that
428 // are not simple c.begin()..c.end().
429 std::set<IterTy, LessTy> uniq;
430 for (auto it{range.begin()}, end{range.end()}; it != end; ++it) {
431 if (!uniq.insert(it).second) {
432 return it;
433 }
434 }
435 return std::nullopt;
436}
437
438} // namespace Fortran::semantics
439#endif // FORTRAN_SEMANTICS_CHECK_OMP_STRUCTURE_H_
Definition enum-set.h:28
Definition expression.h:908
Definition common.h:215
Definition char-block.h:28
Definition check-directive-structure.h:208
Definition semantics.h:67
Definition FIRType.h:103
Definition parse-tree.h:1896
Definition parse-tree.h:2180
Definition parse-tree.h:3018
Definition parse-tree.h:1806
Definition parse-tree.h:1845
Definition parse-tree.h:2323
Definition parse-tree.h:554
Definition parse-tree.h:3137
Definition parse-tree.h:2927
Definition parse-tree.h:2962
Definition parse-tree.h:587
Definition parse-tree.h:5278
Definition parse-tree.h:5072
Definition parse-tree.h:5082
Definition parse-tree.h:5041
Definition parse-tree.h:5025
Definition parse-tree.h:3492
Definition parse-tree.h:5048
Definition parse-tree.h:4472
Definition parse-tree.h:5077
Definition parse-tree.h:5419
Definition parse-tree.h:5148
Definition parse-tree.h:5110
Definition parse-tree.h:5095
Definition parse-tree.h:5104
Definition parse-tree.h:3537
Definition parse-tree.h:3525
Definition parse-tree.h:3581
Definition parse-tree.h:5304
Definition parse-tree.h:5137
Definition parse-tree.h:5309
Definition parse-tree.h:5346
Definition parse-tree.h:5450
Definition parse-tree.h:5294
Definition parse-tree.h:5125
Definition parse-tree.h:5357
Definition parse-tree.h:5371
Definition parse-tree.h:5387
Definition parse-tree.h:5233
Definition parse-tree.h:5395
Definition parse-tree.h:5468
Definition parse-tree.h:5424
Definition parse-tree.h:5239
Definition parse-tree.h:5163
Definition parse-tree.h:5245
Definition parse-tree.h:451
Definition parse-tree.h:3004
Definition parse-tree.h:3156
Definition parse-tree.h:3974
Definition parse-tree.h:4040
Definition parse-tree.h:4050
Definition parse-tree.h:4058
Definition parse-tree.h:4199
Definition parse-tree.h:3727
Definition parse-tree.h:3693
Definition parse-tree.h:3766
Definition parse-tree.h:3788
Definition check-omp-atomic.cpp:242
Definition openmp-utils.h:254