1 //===- DeclOpenMP.h - Classes for representing OpenMP directives -*- 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 /// \file 10 /// This file defines OpenMP nodes for declarative directives. 11 /// 12 //===----------------------------------------------------------------------===// 13 14 #ifndef LLVM_CLANG_AST_DECLOPENMP_H 15 #define LLVM_CLANG_AST_DECLOPENMP_H 16 17 #include "clang/AST/ASTContext.h" 18 #include "clang/AST/Decl.h" 19 #include "clang/AST/Expr.h" 20 #include "clang/AST/ExternalASTSource.h" 21 #include "clang/AST/OpenMPClause.h" 22 #include "clang/AST/Type.h" 23 #include "llvm/ADT/ArrayRef.h" 24 #include "llvm/Support/TrailingObjects.h" 25 26 namespace clang { 27 28 /// This is a basic class for representing single OpenMP declarative directive. 29 /// 30 template <typename U> class OMPDeclarativeDirective : public U { 31 friend class ASTDeclReader; 32 friend class ASTDeclWriter; 33 34 /// Get the clauses storage. getClauses()35 MutableArrayRef<OMPClause *> getClauses() { 36 if (!Data) 37 return llvm::None; 38 return Data->getClauses(); 39 } 40 41 protected: 42 /// Data, associated with the directive. 43 OMPChildren *Data = nullptr; 44 45 /// Build instance of directive. 46 template <typename... Params> OMPDeclarativeDirective(Params &&...P)47 OMPDeclarativeDirective(Params &&... P) : U(std::forward<Params>(P)...) {} 48 49 template <typename T, typename... Params> createDirective(const ASTContext & C,DeclContext * DC,ArrayRef<OMPClause * > Clauses,unsigned NumChildren,Params &&...P)50 static T *createDirective(const ASTContext &C, DeclContext *DC, 51 ArrayRef<OMPClause *> Clauses, unsigned NumChildren, 52 Params &&... P) { 53 auto *Inst = new (C, DC, size(Clauses.size(), NumChildren)) 54 T(DC, std::forward<Params>(P)...); 55 Inst->Data = OMPChildren::Create(Inst + 1, Clauses, 56 /*AssociatedStmt=*/nullptr, NumChildren); 57 Inst->Data->setClauses(Clauses); 58 return Inst; 59 } 60 61 template <typename T, typename... Params> createEmptyDirective(const ASTContext & C,unsigned ID,unsigned NumClauses,unsigned NumChildren,Params &&...P)62 static T *createEmptyDirective(const ASTContext &C, unsigned ID, 63 unsigned NumClauses, unsigned NumChildren, 64 Params &&... P) { 65 auto *Inst = new (C, ID, size(NumClauses, NumChildren)) 66 T(nullptr, std::forward<Params>(P)...); 67 Inst->Data = OMPChildren::CreateEmpty( 68 Inst + 1, NumClauses, /*HasAssociatedStmt=*/false, NumChildren); 69 return Inst; 70 } 71 size(unsigned NumClauses,unsigned NumChildren)72 static size_t size(unsigned NumClauses, unsigned NumChildren) { 73 return OMPChildren::size(NumClauses, /*HasAssociatedStmt=*/false, 74 NumChildren); 75 } 76 77 public: 78 /// Get number of clauses. getNumClauses()79 unsigned getNumClauses() const { 80 if (!Data) 81 return 0; 82 return Data->getNumClauses(); 83 } 84 85 /// Returns specified clause. 86 /// 87 /// \param I Number of clause. 88 /// getClause(unsigned I)89 OMPClause *getClause(unsigned I) const { return clauses()[I]; } 90 clauses()91 ArrayRef<OMPClause *> clauses() const { 92 if (!Data) 93 return llvm::None; 94 return Data->getClauses(); 95 } 96 }; 97 98 /// This represents '#pragma omp threadprivate ...' directive. 99 /// For example, in the following, both 'a' and 'A::b' are threadprivate: 100 /// 101 /// \code 102 /// int a; 103 /// #pragma omp threadprivate(a) 104 /// struct A { 105 /// static int b; 106 /// #pragma omp threadprivate(b) 107 /// }; 108 /// \endcode 109 /// 110 class OMPThreadPrivateDecl final : public OMPDeclarativeDirective<Decl> { 111 friend class OMPDeclarativeDirective<Decl>; 112 113 virtual void anchor(); 114 115 OMPThreadPrivateDecl(DeclContext *DC = nullptr, 116 SourceLocation L = SourceLocation()) 117 : OMPDeclarativeDirective<Decl>(OMPThreadPrivate, DC, L) {} 118 getVars()119 ArrayRef<const Expr *> getVars() const { 120 auto **Storage = reinterpret_cast<Expr **>(Data->getChildren().data()); 121 return llvm::makeArrayRef(Storage, Data->getNumChildren()); 122 } 123 getVars()124 MutableArrayRef<Expr *> getVars() { 125 auto **Storage = reinterpret_cast<Expr **>(Data->getChildren().data()); 126 return llvm::makeMutableArrayRef(Storage, Data->getNumChildren()); 127 } 128 129 void setVars(ArrayRef<Expr *> VL); 130 131 public: 132 static OMPThreadPrivateDecl *Create(ASTContext &C, DeclContext *DC, 133 SourceLocation L, 134 ArrayRef<Expr *> VL); 135 static OMPThreadPrivateDecl *CreateDeserialized(ASTContext &C, 136 unsigned ID, unsigned N); 137 138 typedef MutableArrayRef<Expr *>::iterator varlist_iterator; 139 typedef ArrayRef<const Expr *>::iterator varlist_const_iterator; 140 typedef llvm::iterator_range<varlist_iterator> varlist_range; 141 typedef llvm::iterator_range<varlist_const_iterator> varlist_const_range; 142 varlist_size()143 unsigned varlist_size() const { return Data->getNumChildren(); } varlist_empty()144 bool varlist_empty() const { return Data->getChildren().empty(); } 145 varlists()146 varlist_range varlists() { 147 return varlist_range(varlist_begin(), varlist_end()); 148 } varlists()149 varlist_const_range varlists() const { 150 return varlist_const_range(varlist_begin(), varlist_end()); 151 } varlist_begin()152 varlist_iterator varlist_begin() { return getVars().begin(); } varlist_end()153 varlist_iterator varlist_end() { return getVars().end(); } varlist_begin()154 varlist_const_iterator varlist_begin() const { return getVars().begin(); } varlist_end()155 varlist_const_iterator varlist_end() const { return getVars().end(); } 156 classof(const Decl * D)157 static bool classof(const Decl *D) { return classofKind(D->getKind()); } classofKind(Kind K)158 static bool classofKind(Kind K) { return K == OMPThreadPrivate; } 159 }; 160 161 /// This represents '#pragma omp declare reduction ...' directive. 162 /// For example, in the following, declared reduction 'foo' for types 'int' and 163 /// 'float': 164 /// 165 /// \code 166 /// #pragma omp declare reduction (foo : int,float : omp_out += omp_in) \ 167 /// initializer (omp_priv = 0) 168 /// \endcode 169 /// 170 /// Here 'omp_out += omp_in' is a combiner and 'omp_priv = 0' is an initializer. 171 class OMPDeclareReductionDecl final : public ValueDecl, public DeclContext { 172 // This class stores some data in DeclContext::OMPDeclareReductionDeclBits 173 // to save some space. Use the provided accessors to access it. 174 public: 175 enum InitKind { 176 CallInit, // Initialized by function call. 177 DirectInit, // omp_priv(<expr>) 178 CopyInit // omp_priv = <expr> 179 }; 180 181 private: 182 friend class ASTDeclReader; 183 /// Combiner for declare reduction construct. 184 Expr *Combiner = nullptr; 185 /// Initializer for declare reduction construct. 186 Expr *Initializer = nullptr; 187 /// In parameter of the combiner. 188 Expr *In = nullptr; 189 /// Out parameter of the combiner. 190 Expr *Out = nullptr; 191 /// Priv parameter of the initializer. 192 Expr *Priv = nullptr; 193 /// Orig parameter of the initializer. 194 Expr *Orig = nullptr; 195 196 /// Reference to the previous declare reduction construct in the same 197 /// scope with the same name. Required for proper templates instantiation if 198 /// the declare reduction construct is declared inside compound statement. 199 LazyDeclPtr PrevDeclInScope; 200 201 void anchor() override; 202 203 OMPDeclareReductionDecl(Kind DK, DeclContext *DC, SourceLocation L, 204 DeclarationName Name, QualType Ty, 205 OMPDeclareReductionDecl *PrevDeclInScope); 206 setPrevDeclInScope(OMPDeclareReductionDecl * Prev)207 void setPrevDeclInScope(OMPDeclareReductionDecl *Prev) { 208 PrevDeclInScope = Prev; 209 } 210 211 public: 212 /// Create declare reduction node. 213 static OMPDeclareReductionDecl * 214 Create(ASTContext &C, DeclContext *DC, SourceLocation L, DeclarationName Name, 215 QualType T, OMPDeclareReductionDecl *PrevDeclInScope); 216 /// Create deserialized declare reduction node. 217 static OMPDeclareReductionDecl *CreateDeserialized(ASTContext &C, 218 unsigned ID); 219 220 /// Get combiner expression of the declare reduction construct. getCombiner()221 Expr *getCombiner() { return Combiner; } getCombiner()222 const Expr *getCombiner() const { return Combiner; } 223 /// Get In variable of the combiner. getCombinerIn()224 Expr *getCombinerIn() { return In; } getCombinerIn()225 const Expr *getCombinerIn() const { return In; } 226 /// Get Out variable of the combiner. getCombinerOut()227 Expr *getCombinerOut() { return Out; } getCombinerOut()228 const Expr *getCombinerOut() const { return Out; } 229 /// Set combiner expression for the declare reduction construct. setCombiner(Expr * E)230 void setCombiner(Expr *E) { Combiner = E; } 231 /// Set combiner In and Out vars. setCombinerData(Expr * InE,Expr * OutE)232 void setCombinerData(Expr *InE, Expr *OutE) { 233 In = InE; 234 Out = OutE; 235 } 236 237 /// Get initializer expression (if specified) of the declare reduction 238 /// construct. getInitializer()239 Expr *getInitializer() { return Initializer; } getInitializer()240 const Expr *getInitializer() const { return Initializer; } 241 /// Get initializer kind. getInitializerKind()242 InitKind getInitializerKind() const { 243 return static_cast<InitKind>(OMPDeclareReductionDeclBits.InitializerKind); 244 } 245 /// Get Orig variable of the initializer. getInitOrig()246 Expr *getInitOrig() { return Orig; } getInitOrig()247 const Expr *getInitOrig() const { return Orig; } 248 /// Get Priv variable of the initializer. getInitPriv()249 Expr *getInitPriv() { return Priv; } getInitPriv()250 const Expr *getInitPriv() const { return Priv; } 251 /// Set initializer expression for the declare reduction construct. setInitializer(Expr * E,InitKind IK)252 void setInitializer(Expr *E, InitKind IK) { 253 Initializer = E; 254 OMPDeclareReductionDeclBits.InitializerKind = IK; 255 } 256 /// Set initializer Orig and Priv vars. setInitializerData(Expr * OrigE,Expr * PrivE)257 void setInitializerData(Expr *OrigE, Expr *PrivE) { 258 Orig = OrigE; 259 Priv = PrivE; 260 } 261 262 /// Get reference to previous declare reduction construct in the same 263 /// scope with the same name. 264 OMPDeclareReductionDecl *getPrevDeclInScope(); 265 const OMPDeclareReductionDecl *getPrevDeclInScope() const; 266 classof(const Decl * D)267 static bool classof(const Decl *D) { return classofKind(D->getKind()); } classofKind(Kind K)268 static bool classofKind(Kind K) { return K == OMPDeclareReduction; } castToDeclContext(const OMPDeclareReductionDecl * D)269 static DeclContext *castToDeclContext(const OMPDeclareReductionDecl *D) { 270 return static_cast<DeclContext *>(const_cast<OMPDeclareReductionDecl *>(D)); 271 } castFromDeclContext(const DeclContext * DC)272 static OMPDeclareReductionDecl *castFromDeclContext(const DeclContext *DC) { 273 return static_cast<OMPDeclareReductionDecl *>( 274 const_cast<DeclContext *>(DC)); 275 } 276 }; 277 278 /// This represents '#pragma omp declare mapper ...' directive. Map clauses are 279 /// allowed to use with this directive. The following example declares a user 280 /// defined mapper for the type 'struct vec'. This example instructs the fields 281 /// 'len' and 'data' should be mapped when mapping instances of 'struct vec'. 282 /// 283 /// \code 284 /// #pragma omp declare mapper(mid: struct vec v) map(v.len, v.data[0:N]) 285 /// \endcode 286 class OMPDeclareMapperDecl final : public OMPDeclarativeDirective<ValueDecl>, 287 public DeclContext { 288 friend class OMPDeclarativeDirective<ValueDecl>; 289 friend class ASTDeclReader; 290 friend class ASTDeclWriter; 291 292 /// Mapper variable, which is 'v' in the example above 293 Expr *MapperVarRef = nullptr; 294 295 /// Name of the mapper variable 296 DeclarationName VarName; 297 298 LazyDeclPtr PrevDeclInScope; 299 300 void anchor() override; 301 OMPDeclareMapperDecl(DeclContext * DC,SourceLocation L,DeclarationName Name,QualType Ty,DeclarationName VarName,OMPDeclareMapperDecl * PrevDeclInScope)302 OMPDeclareMapperDecl(DeclContext *DC, SourceLocation L, DeclarationName Name, 303 QualType Ty, DeclarationName VarName, 304 OMPDeclareMapperDecl *PrevDeclInScope) 305 : OMPDeclarativeDirective<ValueDecl>(OMPDeclareMapper, DC, L, Name, Ty), 306 DeclContext(OMPDeclareMapper), VarName(VarName), 307 PrevDeclInScope(PrevDeclInScope) {} 308 setPrevDeclInScope(OMPDeclareMapperDecl * Prev)309 void setPrevDeclInScope(OMPDeclareMapperDecl *Prev) { 310 PrevDeclInScope = Prev; 311 } 312 313 public: 314 /// Creates declare mapper node. 315 static OMPDeclareMapperDecl *Create(ASTContext &C, DeclContext *DC, 316 SourceLocation L, DeclarationName Name, 317 QualType T, DeclarationName VarName, 318 ArrayRef<OMPClause *> Clauses, 319 OMPDeclareMapperDecl *PrevDeclInScope); 320 /// Creates deserialized declare mapper node. 321 static OMPDeclareMapperDecl *CreateDeserialized(ASTContext &C, unsigned ID, 322 unsigned N); 323 324 using clauselist_iterator = MutableArrayRef<OMPClause *>::iterator; 325 using clauselist_const_iterator = ArrayRef<const OMPClause *>::iterator; 326 using clauselist_range = llvm::iterator_range<clauselist_iterator>; 327 using clauselist_const_range = 328 llvm::iterator_range<clauselist_const_iterator>; 329 clauselist_size()330 unsigned clauselist_size() const { return Data->getNumClauses(); } clauselist_empty()331 bool clauselist_empty() const { return Data->getClauses().empty(); } 332 clauselists()333 clauselist_range clauselists() { 334 return clauselist_range(clauselist_begin(), clauselist_end()); 335 } clauselists()336 clauselist_const_range clauselists() const { 337 return clauselist_const_range(clauselist_begin(), clauselist_end()); 338 } clauselist_begin()339 clauselist_iterator clauselist_begin() { return Data->getClauses().begin(); } clauselist_end()340 clauselist_iterator clauselist_end() { return Data->getClauses().end(); } clauselist_begin()341 clauselist_const_iterator clauselist_begin() const { 342 return Data->getClauses().begin(); 343 } clauselist_end()344 clauselist_const_iterator clauselist_end() const { 345 return Data->getClauses().end(); 346 } 347 348 /// Get the variable declared in the mapper getMapperVarRef()349 Expr *getMapperVarRef() { return cast_or_null<Expr>(Data->getChildren()[0]); } getMapperVarRef()350 const Expr *getMapperVarRef() const { 351 return cast_or_null<Expr>(Data->getChildren()[0]); 352 } 353 /// Set the variable declared in the mapper setMapperVarRef(Expr * MapperVarRefE)354 void setMapperVarRef(Expr *MapperVarRefE) { 355 Data->getChildren()[0] = MapperVarRefE; 356 } 357 358 /// Get the name of the variable declared in the mapper getVarName()359 DeclarationName getVarName() { return VarName; } 360 361 /// Get reference to previous declare mapper construct in the same 362 /// scope with the same name. 363 OMPDeclareMapperDecl *getPrevDeclInScope(); 364 const OMPDeclareMapperDecl *getPrevDeclInScope() const; 365 classof(const Decl * D)366 static bool classof(const Decl *D) { return classofKind(D->getKind()); } classofKind(Kind K)367 static bool classofKind(Kind K) { return K == OMPDeclareMapper; } castToDeclContext(const OMPDeclareMapperDecl * D)368 static DeclContext *castToDeclContext(const OMPDeclareMapperDecl *D) { 369 return static_cast<DeclContext *>(const_cast<OMPDeclareMapperDecl *>(D)); 370 } castFromDeclContext(const DeclContext * DC)371 static OMPDeclareMapperDecl *castFromDeclContext(const DeclContext *DC) { 372 return static_cast<OMPDeclareMapperDecl *>(const_cast<DeclContext *>(DC)); 373 } 374 }; 375 376 /// Pseudo declaration for capturing expressions. Also is used for capturing of 377 /// non-static data members in non-static member functions. 378 /// 379 /// Clang supports capturing of variables only, but OpenMP 4.5 allows to 380 /// privatize non-static members of current class in non-static member 381 /// functions. This pseudo-declaration allows properly handle this kind of 382 /// capture by wrapping captured expression into a variable-like declaration. 383 class OMPCapturedExprDecl final : public VarDecl { 384 friend class ASTDeclReader; 385 void anchor() override; 386 OMPCapturedExprDecl(ASTContext & C,DeclContext * DC,IdentifierInfo * Id,QualType Type,TypeSourceInfo * TInfo,SourceLocation StartLoc)387 OMPCapturedExprDecl(ASTContext &C, DeclContext *DC, IdentifierInfo *Id, 388 QualType Type, TypeSourceInfo *TInfo, 389 SourceLocation StartLoc) 390 : VarDecl(OMPCapturedExpr, C, DC, StartLoc, StartLoc, Id, Type, TInfo, 391 SC_None) { 392 setImplicit(); 393 } 394 395 public: 396 static OMPCapturedExprDecl *Create(ASTContext &C, DeclContext *DC, 397 IdentifierInfo *Id, QualType T, 398 SourceLocation StartLoc); 399 400 static OMPCapturedExprDecl *CreateDeserialized(ASTContext &C, unsigned ID); 401 402 SourceRange getSourceRange() const override LLVM_READONLY; 403 404 // Implement isa/cast/dyncast/etc. classof(const Decl * D)405 static bool classof(const Decl *D) { return classofKind(D->getKind()); } classofKind(Kind K)406 static bool classofKind(Kind K) { return K == OMPCapturedExpr; } 407 }; 408 409 /// This represents '#pragma omp requires...' directive. 410 /// For example 411 /// 412 /// \code 413 /// #pragma omp requires unified_address 414 /// \endcode 415 /// 416 class OMPRequiresDecl final : public OMPDeclarativeDirective<Decl> { 417 friend class OMPDeclarativeDirective<Decl>; 418 friend class ASTDeclReader; 419 420 virtual void anchor(); 421 OMPRequiresDecl(DeclContext * DC,SourceLocation L)422 OMPRequiresDecl(DeclContext *DC, SourceLocation L) 423 : OMPDeclarativeDirective<Decl>(OMPRequires, DC, L) {} 424 425 public: 426 /// Create requires node. 427 static OMPRequiresDecl *Create(ASTContext &C, DeclContext *DC, 428 SourceLocation L, ArrayRef<OMPClause *> CL); 429 /// Create deserialized requires node. 430 static OMPRequiresDecl *CreateDeserialized(ASTContext &C, unsigned ID, 431 unsigned N); 432 433 using clauselist_iterator = MutableArrayRef<OMPClause *>::iterator; 434 using clauselist_const_iterator = ArrayRef<const OMPClause *>::iterator; 435 using clauselist_range = llvm::iterator_range<clauselist_iterator>; 436 using clauselist_const_range = llvm::iterator_range<clauselist_const_iterator>; 437 clauselist_size()438 unsigned clauselist_size() const { return Data->getNumClauses(); } clauselist_empty()439 bool clauselist_empty() const { return Data->getClauses().empty(); } 440 clauselists()441 clauselist_range clauselists() { 442 return clauselist_range(clauselist_begin(), clauselist_end()); 443 } clauselists()444 clauselist_const_range clauselists() const { 445 return clauselist_const_range(clauselist_begin(), clauselist_end()); 446 } clauselist_begin()447 clauselist_iterator clauselist_begin() { return Data->getClauses().begin(); } clauselist_end()448 clauselist_iterator clauselist_end() { return Data->getClauses().end(); } clauselist_begin()449 clauselist_const_iterator clauselist_begin() const { 450 return Data->getClauses().begin(); 451 } clauselist_end()452 clauselist_const_iterator clauselist_end() const { 453 return Data->getClauses().end(); 454 } 455 classof(const Decl * D)456 static bool classof(const Decl *D) { return classofKind(D->getKind()); } classofKind(Kind K)457 static bool classofKind(Kind K) { return K == OMPRequires; } 458 }; 459 460 /// This represents '#pragma omp allocate ...' directive. 461 /// For example, in the following, the default allocator is used for both 'a' 462 /// and 'A::b': 463 /// 464 /// \code 465 /// int a; 466 /// #pragma omp allocate(a) 467 /// struct A { 468 /// static int b; 469 /// #pragma omp allocate(b) 470 /// }; 471 /// \endcode 472 /// 473 class OMPAllocateDecl final : public OMPDeclarativeDirective<Decl> { 474 friend class OMPDeclarativeDirective<Decl>; 475 friend class ASTDeclReader; 476 477 virtual void anchor(); 478 OMPAllocateDecl(DeclContext * DC,SourceLocation L)479 OMPAllocateDecl(DeclContext *DC, SourceLocation L) 480 : OMPDeclarativeDirective<Decl>(OMPAllocate, DC, L) {} 481 getVars()482 ArrayRef<const Expr *> getVars() const { 483 auto **Storage = reinterpret_cast<Expr **>(Data->getChildren().data()); 484 return llvm::makeArrayRef(Storage, Data->getNumChildren()); 485 } 486 getVars()487 MutableArrayRef<Expr *> getVars() { 488 auto **Storage = reinterpret_cast<Expr **>(Data->getChildren().data()); 489 return llvm::makeMutableArrayRef(Storage, Data->getNumChildren()); 490 } 491 492 void setVars(ArrayRef<Expr *> VL); 493 494 public: 495 static OMPAllocateDecl *Create(ASTContext &C, DeclContext *DC, 496 SourceLocation L, ArrayRef<Expr *> VL, 497 ArrayRef<OMPClause *> CL); 498 static OMPAllocateDecl *CreateDeserialized(ASTContext &C, unsigned ID, 499 unsigned NVars, unsigned NClauses); 500 501 typedef MutableArrayRef<Expr *>::iterator varlist_iterator; 502 typedef ArrayRef<const Expr *>::iterator varlist_const_iterator; 503 typedef llvm::iterator_range<varlist_iterator> varlist_range; 504 typedef llvm::iterator_range<varlist_const_iterator> varlist_const_range; 505 using clauselist_iterator = MutableArrayRef<OMPClause *>::iterator; 506 using clauselist_const_iterator = ArrayRef<const OMPClause *>::iterator; 507 using clauselist_range = llvm::iterator_range<clauselist_iterator>; 508 using clauselist_const_range = llvm::iterator_range<clauselist_const_iterator>; 509 varlist_size()510 unsigned varlist_size() const { return Data->getNumChildren(); } varlist_empty()511 bool varlist_empty() const { return Data->getChildren().empty(); } clauselist_size()512 unsigned clauselist_size() const { return Data->getNumClauses(); } clauselist_empty()513 bool clauselist_empty() const { return Data->getClauses().empty(); } 514 varlists()515 varlist_range varlists() { 516 return varlist_range(varlist_begin(), varlist_end()); 517 } varlists()518 varlist_const_range varlists() const { 519 return varlist_const_range(varlist_begin(), varlist_end()); 520 } varlist_begin()521 varlist_iterator varlist_begin() { return getVars().begin(); } varlist_end()522 varlist_iterator varlist_end() { return getVars().end(); } varlist_begin()523 varlist_const_iterator varlist_begin() const { return getVars().begin(); } varlist_end()524 varlist_const_iterator varlist_end() const { return getVars().end(); } 525 clauselists()526 clauselist_range clauselists() { 527 return clauselist_range(clauselist_begin(), clauselist_end()); 528 } clauselists()529 clauselist_const_range clauselists() const { 530 return clauselist_const_range(clauselist_begin(), clauselist_end()); 531 } clauselist_begin()532 clauselist_iterator clauselist_begin() { return Data->getClauses().begin(); } clauselist_end()533 clauselist_iterator clauselist_end() { return Data->getClauses().end(); } clauselist_begin()534 clauselist_const_iterator clauselist_begin() const { 535 return Data->getClauses().begin(); 536 } clauselist_end()537 clauselist_const_iterator clauselist_end() const { 538 return Data->getClauses().end(); 539 } 540 classof(const Decl * D)541 static bool classof(const Decl *D) { return classofKind(D->getKind()); } classofKind(Kind K)542 static bool classofKind(Kind K) { return K == OMPAllocate; } 543 }; 544 545 } // end namespace clang 546 547 #endif 548