Details | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
14 | pmbaty | 1 | //===--- ExprOpenMP.h - Classes for representing expressions ----*- 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 | // This file defines the Expr interface and subclasses. |
||
10 | // |
||
11 | //===----------------------------------------------------------------------===// |
||
12 | |||
13 | #ifndef LLVM_CLANG_AST_EXPROPENMP_H |
||
14 | #define LLVM_CLANG_AST_EXPROPENMP_H |
||
15 | |||
16 | #include "clang/AST/ComputeDependence.h" |
||
17 | #include "clang/AST/Expr.h" |
||
18 | |||
19 | namespace clang { |
||
20 | /// OpenMP 5.0 [2.1.5, Array Sections]. |
||
21 | /// To specify an array section in an OpenMP construct, array subscript |
||
22 | /// expressions are extended with the following syntax: |
||
23 | /// \code |
||
24 | /// [ lower-bound : length : stride ] |
||
25 | /// [ lower-bound : length : ] |
||
26 | /// [ lower-bound : length ] |
||
27 | /// [ lower-bound : : stride ] |
||
28 | /// [ lower-bound : : ] |
||
29 | /// [ lower-bound : ] |
||
30 | /// [ : length : stride ] |
||
31 | /// [ : length : ] |
||
32 | /// [ : length ] |
||
33 | /// [ : : stride ] |
||
34 | /// [ : : ] |
||
35 | /// [ : ] |
||
36 | /// \endcode |
||
37 | /// The array section must be a subset of the original array. |
||
38 | /// Array sections are allowed on multidimensional arrays. Base language array |
||
39 | /// subscript expressions can be used to specify length-one dimensions of |
||
40 | /// multidimensional array sections. |
||
41 | /// Each of the lower-bound, length, and stride expressions if specified must be |
||
42 | /// an integral type expressions of the base language. When evaluated |
||
43 | /// they represent a set of integer values as follows: |
||
44 | /// \code |
||
45 | /// { lower-bound, lower-bound + stride, lower-bound + 2 * stride,... , |
||
46 | /// lower-bound + ((length - 1) * stride) } |
||
47 | /// \endcode |
||
48 | /// The lower-bound and length must evaluate to non-negative integers. |
||
49 | /// The stride must evaluate to a positive integer. |
||
50 | /// When the size of the array dimension is not known, the length must be |
||
51 | /// specified explicitly. |
||
52 | /// When the stride is absent it defaults to 1. |
||
53 | /// When the length is absent it defaults to ⌈(size − lower-bound)/stride⌉, |
||
54 | /// where size is the size of the array dimension. When the lower-bound is |
||
55 | /// absent it defaults to 0. |
||
56 | class OMPArraySectionExpr : public Expr { |
||
57 | enum { BASE, LOWER_BOUND, LENGTH, STRIDE, END_EXPR }; |
||
58 | Stmt *SubExprs[END_EXPR]; |
||
59 | SourceLocation ColonLocFirst; |
||
60 | SourceLocation ColonLocSecond; |
||
61 | SourceLocation RBracketLoc; |
||
62 | |||
63 | public: |
||
64 | OMPArraySectionExpr(Expr *Base, Expr *LowerBound, Expr *Length, Expr *Stride, |
||
65 | QualType Type, ExprValueKind VK, ExprObjectKind OK, |
||
66 | SourceLocation ColonLocFirst, |
||
67 | SourceLocation ColonLocSecond, SourceLocation RBracketLoc) |
||
68 | : Expr(OMPArraySectionExprClass, Type, VK, OK), |
||
69 | ColonLocFirst(ColonLocFirst), ColonLocSecond(ColonLocSecond), |
||
70 | RBracketLoc(RBracketLoc) { |
||
71 | SubExprs[BASE] = Base; |
||
72 | SubExprs[LOWER_BOUND] = LowerBound; |
||
73 | SubExprs[LENGTH] = Length; |
||
74 | SubExprs[STRIDE] = Stride; |
||
75 | setDependence(computeDependence(this)); |
||
76 | } |
||
77 | |||
78 | /// Create an empty array section expression. |
||
79 | explicit OMPArraySectionExpr(EmptyShell Shell) |
||
80 | : Expr(OMPArraySectionExprClass, Shell) {} |
||
81 | |||
82 | /// An array section can be written only as Base[LowerBound:Length]. |
||
83 | |||
84 | /// Get base of the array section. |
||
85 | Expr *getBase() { return cast<Expr>(SubExprs[BASE]); } |
||
86 | const Expr *getBase() const { return cast<Expr>(SubExprs[BASE]); } |
||
87 | /// Set base of the array section. |
||
88 | void setBase(Expr *E) { SubExprs[BASE] = E; } |
||
89 | |||
90 | /// Return original type of the base expression for array section. |
||
91 | static QualType getBaseOriginalType(const Expr *Base); |
||
92 | |||
93 | /// Get lower bound of array section. |
||
94 | Expr *getLowerBound() { return cast_or_null<Expr>(SubExprs[LOWER_BOUND]); } |
||
95 | const Expr *getLowerBound() const { |
||
96 | return cast_or_null<Expr>(SubExprs[LOWER_BOUND]); |
||
97 | } |
||
98 | /// Set lower bound of the array section. |
||
99 | void setLowerBound(Expr *E) { SubExprs[LOWER_BOUND] = E; } |
||
100 | |||
101 | /// Get length of array section. |
||
102 | Expr *getLength() { return cast_or_null<Expr>(SubExprs[LENGTH]); } |
||
103 | const Expr *getLength() const { return cast_or_null<Expr>(SubExprs[LENGTH]); } |
||
104 | /// Set length of the array section. |
||
105 | void setLength(Expr *E) { SubExprs[LENGTH] = E; } |
||
106 | |||
107 | /// Get stride of array section. |
||
108 | Expr *getStride() { return cast_or_null<Expr>(SubExprs[STRIDE]); } |
||
109 | const Expr *getStride() const { return cast_or_null<Expr>(SubExprs[STRIDE]); } |
||
110 | /// Set length of the array section. |
||
111 | void setStride(Expr *E) { SubExprs[STRIDE] = E; } |
||
112 | |||
113 | SourceLocation getBeginLoc() const LLVM_READONLY { |
||
114 | return getBase()->getBeginLoc(); |
||
115 | } |
||
116 | SourceLocation getEndLoc() const LLVM_READONLY { return RBracketLoc; } |
||
117 | |||
118 | SourceLocation getColonLocFirst() const { return ColonLocFirst; } |
||
119 | void setColonLocFirst(SourceLocation L) { ColonLocFirst = L; } |
||
120 | |||
121 | SourceLocation getColonLocSecond() const { return ColonLocSecond; } |
||
122 | void setColonLocSecond(SourceLocation L) { ColonLocSecond = L; } |
||
123 | |||
124 | SourceLocation getRBracketLoc() const { return RBracketLoc; } |
||
125 | void setRBracketLoc(SourceLocation L) { RBracketLoc = L; } |
||
126 | |||
127 | SourceLocation getExprLoc() const LLVM_READONLY { |
||
128 | return getBase()->getExprLoc(); |
||
129 | } |
||
130 | |||
131 | static bool classof(const Stmt *T) { |
||
132 | return T->getStmtClass() == OMPArraySectionExprClass; |
||
133 | } |
||
134 | |||
135 | child_range children() { |
||
136 | return child_range(&SubExprs[BASE], &SubExprs[END_EXPR]); |
||
137 | } |
||
138 | |||
139 | const_child_range children() const { |
||
140 | return const_child_range(&SubExprs[BASE], &SubExprs[END_EXPR]); |
||
141 | } |
||
142 | }; |
||
143 | |||
144 | /// An explicit cast in C or a C-style cast in C++, which uses the syntax |
||
145 | /// ([s1][s2]...[sn])expr. For example: @c ([3][3])f. |
||
146 | class OMPArrayShapingExpr final |
||
147 | : public Expr, |
||
148 | private llvm::TrailingObjects<OMPArrayShapingExpr, Expr *, SourceRange> { |
||
149 | friend TrailingObjects; |
||
150 | friend class ASTStmtReader; |
||
151 | friend class ASTStmtWriter; |
||
152 | /// Base node. |
||
153 | SourceLocation LPLoc; /// The location of the left paren |
||
154 | SourceLocation RPLoc; /// The location of the right paren |
||
155 | unsigned NumDims = 0; /// Number of dimensions in the shaping expression. |
||
156 | |||
157 | /// Construct full expression. |
||
158 | OMPArrayShapingExpr(QualType ExprTy, Expr *Op, SourceLocation L, |
||
159 | SourceLocation R, ArrayRef<Expr *> Dims); |
||
160 | |||
161 | /// Construct an empty expression. |
||
162 | explicit OMPArrayShapingExpr(EmptyShell Shell, unsigned NumDims) |
||
163 | : Expr(OMPArrayShapingExprClass, Shell), NumDims(NumDims) {} |
||
164 | |||
165 | /// Sets the dimensions for the array shaping. |
||
166 | void setDimensions(ArrayRef<Expr *> Dims); |
||
167 | |||
168 | /// Sets the base expression for array shaping operation. |
||
169 | void setBase(Expr *Op) { getTrailingObjects<Expr *>()[NumDims] = Op; } |
||
170 | |||
171 | /// Sets source ranges for the brackets in the array shaping operation. |
||
172 | void setBracketsRanges(ArrayRef<SourceRange> BR); |
||
173 | |||
174 | unsigned numTrailingObjects(OverloadToken<Expr *>) const { |
||
175 | // Add an extra one for the base expression. |
||
176 | return NumDims + 1; |
||
177 | } |
||
178 | |||
179 | unsigned numTrailingObjects(OverloadToken<SourceRange>) const { |
||
180 | return NumDims; |
||
181 | } |
||
182 | |||
183 | public: |
||
184 | static OMPArrayShapingExpr *Create(const ASTContext &Context, QualType T, |
||
185 | Expr *Op, SourceLocation L, |
||
186 | SourceLocation R, ArrayRef<Expr *> Dims, |
||
187 | ArrayRef<SourceRange> BracketRanges); |
||
188 | |||
189 | static OMPArrayShapingExpr *CreateEmpty(const ASTContext &Context, |
||
190 | unsigned NumDims); |
||
191 | |||
192 | SourceLocation getLParenLoc() const { return LPLoc; } |
||
193 | void setLParenLoc(SourceLocation L) { LPLoc = L; } |
||
194 | |||
195 | SourceLocation getRParenLoc() const { return RPLoc; } |
||
196 | void setRParenLoc(SourceLocation L) { RPLoc = L; } |
||
197 | |||
198 | SourceLocation getBeginLoc() const LLVM_READONLY { return LPLoc; } |
||
199 | SourceLocation getEndLoc() const LLVM_READONLY { |
||
200 | return getBase()->getEndLoc(); |
||
201 | } |
||
202 | |||
203 | /// Fetches the dimensions for array shaping expression. |
||
204 | ArrayRef<Expr *> getDimensions() const { |
||
205 | return llvm::ArrayRef(getTrailingObjects<Expr *>(), NumDims); |
||
206 | } |
||
207 | |||
208 | /// Fetches source ranges for the brackets os the array shaping expression. |
||
209 | ArrayRef<SourceRange> getBracketsRanges() const { |
||
210 | return llvm::ArrayRef(getTrailingObjects<SourceRange>(), NumDims); |
||
211 | } |
||
212 | |||
213 | /// Fetches base expression of array shaping expression. |
||
214 | Expr *getBase() { return getTrailingObjects<Expr *>()[NumDims]; } |
||
215 | const Expr *getBase() const { return getTrailingObjects<Expr *>()[NumDims]; } |
||
216 | |||
217 | static bool classof(const Stmt *T) { |
||
218 | return T->getStmtClass() == OMPArrayShapingExprClass; |
||
219 | } |
||
220 | |||
221 | // Iterators |
||
222 | child_range children() { |
||
223 | Stmt **Begin = reinterpret_cast<Stmt **>(getTrailingObjects<Expr *>()); |
||
224 | return child_range(Begin, Begin + NumDims + 1); |
||
225 | } |
||
226 | const_child_range children() const { |
||
227 | Stmt *const *Begin = |
||
228 | reinterpret_cast<Stmt *const *>(getTrailingObjects<Expr *>()); |
||
229 | return const_child_range(Begin, Begin + NumDims + 1); |
||
230 | } |
||
231 | }; |
||
232 | |||
233 | /// Helper expressions and declaration for OMPIteratorExpr class for each |
||
234 | /// iteration space. |
||
235 | struct OMPIteratorHelperData { |
||
236 | /// Internal normalized counter. |
||
237 | VarDecl *CounterVD = nullptr; |
||
238 | /// Normalized upper bound. Normalized loop iterates from 0 to Upper with |
||
239 | /// step 1. |
||
240 | Expr *Upper = nullptr; |
||
241 | /// Update expression for the originally specified iteration variable, |
||
242 | /// calculated as VD = Begin + CounterVD * Step; |
||
243 | Expr *Update = nullptr; |
||
244 | /// Updater for the internal counter: ++CounterVD; |
||
245 | Expr *CounterUpdate = nullptr; |
||
246 | }; |
||
247 | |||
248 | /// OpenMP 5.0 [2.1.6 Iterators] |
||
249 | /// Iterators are identifiers that expand to multiple values in the clause on |
||
250 | /// which they appear. |
||
251 | /// The syntax of the iterator modifier is as follows: |
||
252 | /// \code |
||
253 | /// iterator(iterators-definition) |
||
254 | /// \endcode |
||
255 | /// where iterators-definition is one of the following: |
||
256 | /// \code |
||
257 | /// iterator-specifier [, iterators-definition ] |
||
258 | /// \endcode |
||
259 | /// where iterator-specifier is one of the following: |
||
260 | /// \code |
||
261 | /// [ iterator-type ] identifier = range-specification |
||
262 | /// \endcode |
||
263 | /// where identifier is a base language identifier. |
||
264 | /// iterator-type is a type name. |
||
265 | /// range-specification is of the form begin:end[:step], where begin and end are |
||
266 | /// expressions for which their types can be converted to iterator-type and step |
||
267 | /// is an integral expression. |
||
268 | /// In an iterator-specifier, if the iterator-type is not specified then the |
||
269 | /// type of that iterator is of int type. |
||
270 | /// The iterator-type must be an integral or pointer type. |
||
271 | /// The iterator-type must not be const qualified. |
||
272 | class OMPIteratorExpr final |
||
273 | : public Expr, |
||
274 | private llvm::TrailingObjects<OMPIteratorExpr, Decl *, Expr *, |
||
275 | SourceLocation, OMPIteratorHelperData> { |
||
276 | public: |
||
277 | /// Iterator range representation begin:end[:step]. |
||
278 | struct IteratorRange { |
||
279 | Expr *Begin = nullptr; |
||
280 | Expr *End = nullptr; |
||
281 | Expr *Step = nullptr; |
||
282 | }; |
||
283 | /// Iterator definition representation. |
||
284 | struct IteratorDefinition { |
||
285 | Decl *IteratorDecl = nullptr; |
||
286 | IteratorRange Range; |
||
287 | SourceLocation AssignmentLoc; |
||
288 | SourceLocation ColonLoc, SecondColonLoc; |
||
289 | }; |
||
290 | |||
291 | private: |
||
292 | friend TrailingObjects; |
||
293 | friend class ASTStmtReader; |
||
294 | friend class ASTStmtWriter; |
||
295 | |||
296 | /// Offset in the list of expressions for subelements of the ranges. |
||
297 | enum class RangeExprOffset { |
||
298 | Begin = 0, |
||
299 | End = 1, |
||
300 | Step = 2, |
||
301 | Total = 3, |
||
302 | }; |
||
303 | /// Offset in the list of locations for subelements of colon symbols |
||
304 | /// locations. |
||
305 | enum class RangeLocOffset { |
||
306 | AssignLoc = 0, |
||
307 | FirstColonLoc = 1, |
||
308 | SecondColonLoc = 2, |
||
309 | Total = 3, |
||
310 | }; |
||
311 | /// Location of 'iterator' keyword. |
||
312 | SourceLocation IteratorKwLoc; |
||
313 | /// Location of '('. |
||
314 | SourceLocation LPLoc; |
||
315 | /// Location of ')'. |
||
316 | SourceLocation RPLoc; |
||
317 | /// Number of iterator definitions. |
||
318 | unsigned NumIterators = 0; |
||
319 | |||
320 | OMPIteratorExpr(QualType ExprTy, SourceLocation IteratorKwLoc, |
||
321 | SourceLocation L, SourceLocation R, |
||
322 | ArrayRef<IteratorDefinition> Data, |
||
323 | ArrayRef<OMPIteratorHelperData> Helpers); |
||
324 | |||
325 | /// Construct an empty expression. |
||
326 | explicit OMPIteratorExpr(EmptyShell Shell, unsigned NumIterators) |
||
327 | : Expr(OMPIteratorExprClass, Shell), NumIterators(NumIterators) {} |
||
328 | |||
329 | /// Sets basic declaration for the specified iterator definition. |
||
330 | void setIteratorDeclaration(unsigned I, Decl *D); |
||
331 | |||
332 | /// Sets the location of the assignment symbol for the specified iterator |
||
333 | /// definition. |
||
334 | void setAssignmentLoc(unsigned I, SourceLocation Loc); |
||
335 | |||
336 | /// Sets begin, end and optional step expressions for specified iterator |
||
337 | /// definition. |
||
338 | void setIteratorRange(unsigned I, Expr *Begin, SourceLocation ColonLoc, |
||
339 | Expr *End, SourceLocation SecondColonLoc, Expr *Step); |
||
340 | |||
341 | /// Sets helpers for the specified iteration space. |
||
342 | void setHelper(unsigned I, const OMPIteratorHelperData &D); |
||
343 | |||
344 | unsigned numTrailingObjects(OverloadToken<Decl *>) const { |
||
345 | return NumIterators; |
||
346 | } |
||
347 | |||
348 | unsigned numTrailingObjects(OverloadToken<Expr *>) const { |
||
349 | return NumIterators * static_cast<int>(RangeExprOffset::Total); |
||
350 | } |
||
351 | |||
352 | unsigned numTrailingObjects(OverloadToken<SourceLocation>) const { |
||
353 | return NumIterators * static_cast<int>(RangeLocOffset::Total); |
||
354 | } |
||
355 | |||
356 | public: |
||
357 | static OMPIteratorExpr *Create(const ASTContext &Context, QualType T, |
||
358 | SourceLocation IteratorKwLoc, SourceLocation L, |
||
359 | SourceLocation R, |
||
360 | ArrayRef<IteratorDefinition> Data, |
||
361 | ArrayRef<OMPIteratorHelperData> Helpers); |
||
362 | |||
363 | static OMPIteratorExpr *CreateEmpty(const ASTContext &Context, |
||
364 | unsigned NumIterators); |
||
365 | |||
366 | SourceLocation getLParenLoc() const { return LPLoc; } |
||
367 | void setLParenLoc(SourceLocation L) { LPLoc = L; } |
||
368 | |||
369 | SourceLocation getRParenLoc() const { return RPLoc; } |
||
370 | void setRParenLoc(SourceLocation L) { RPLoc = L; } |
||
371 | |||
372 | SourceLocation getIteratorKwLoc() const { return IteratorKwLoc; } |
||
373 | void setIteratorKwLoc(SourceLocation L) { IteratorKwLoc = L; } |
||
374 | SourceLocation getBeginLoc() const LLVM_READONLY { return IteratorKwLoc; } |
||
375 | SourceLocation getEndLoc() const LLVM_READONLY { return RPLoc; } |
||
376 | |||
377 | /// Gets the iterator declaration for the given iterator. |
||
378 | Decl *getIteratorDecl(unsigned I); |
||
379 | const Decl *getIteratorDecl(unsigned I) const { |
||
380 | return const_cast<OMPIteratorExpr *>(this)->getIteratorDecl(I); |
||
381 | } |
||
382 | |||
383 | /// Gets the iterator range for the given iterator. |
||
384 | IteratorRange getIteratorRange(unsigned I); |
||
385 | const IteratorRange getIteratorRange(unsigned I) const { |
||
386 | return const_cast<OMPIteratorExpr *>(this)->getIteratorRange(I); |
||
387 | } |
||
388 | |||
389 | /// Gets the location of '=' for the given iterator definition. |
||
390 | SourceLocation getAssignLoc(unsigned I) const; |
||
391 | /// Gets the location of the first ':' in the range for the given iterator |
||
392 | /// definition. |
||
393 | SourceLocation getColonLoc(unsigned I) const; |
||
394 | /// Gets the location of the second ':' (if any) in the range for the given |
||
395 | /// iteratori definition. |
||
396 | SourceLocation getSecondColonLoc(unsigned I) const; |
||
397 | |||
398 | /// Returns number of iterator definitions. |
||
399 | unsigned numOfIterators() const { return NumIterators; } |
||
400 | |||
401 | /// Fetches helper data for the specified iteration space. |
||
402 | OMPIteratorHelperData &getHelper(unsigned I); |
||
403 | const OMPIteratorHelperData &getHelper(unsigned I) const; |
||
404 | |||
405 | static bool classof(const Stmt *T) { |
||
406 | return T->getStmtClass() == OMPIteratorExprClass; |
||
407 | } |
||
408 | |||
409 | // Iterators |
||
410 | child_range children() { |
||
411 | Stmt **Begin = reinterpret_cast<Stmt **>(getTrailingObjects<Expr *>()); |
||
412 | return child_range( |
||
413 | Begin, Begin + NumIterators * static_cast<int>(RangeExprOffset::Total)); |
||
414 | } |
||
415 | const_child_range children() const { |
||
416 | Stmt *const *Begin = |
||
417 | reinterpret_cast<Stmt *const *>(getTrailingObjects<Expr *>()); |
||
418 | return const_child_range( |
||
419 | Begin, Begin + NumIterators * static_cast<int>(RangeExprOffset::Total)); |
||
420 | } |
||
421 | }; |
||
422 | |||
423 | } // end namespace clang |
||
424 | |||
425 | #endif |