//===- GlobalDecl.h - Global declaration holder -----------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // // A GlobalDecl can hold either a regular variable/function or a C++ ctor/dtor // together with its type. // //===----------------------------------------------------------------------===// #ifndef LLVM_CLANG_AST_GLOBALDECL_H #define LLVM_CLANG_AST_GLOBALDECL_H #include "clang/AST/Attr.h" #include "clang/AST/DeclCXX.h" #include "clang/AST/DeclObjC.h" #include "clang/AST/DeclOpenMP.h" #include "clang/AST/DeclTemplate.h" #include "clang/Basic/ABI.h" #include "clang/Basic/LLVM.h" #include "llvm/ADT/DenseMapInfo.h" #include "llvm/ADT/PointerIntPair.h" #include "llvm/Support/Casting.h" #include "llvm/Support/type_traits.h" #include namespace clang { enum class DynamicInitKind : unsigned { NoStub = 0, Initializer, AtExit, GlobalArrayDestructor }; enum class KernelReferenceKind : unsigned { Kernel = 0, Stub = 1, }; /// GlobalDecl - represents a global declaration. This can either be a /// CXXConstructorDecl and the constructor type (Base, Complete). /// a CXXDestructorDecl and the destructor type (Base, Complete), /// a FunctionDecl and the kernel reference type (Kernel, Stub), or /// a VarDecl, a FunctionDecl or a BlockDecl. /// /// When a new type of GlobalDecl is added, the following places should /// be updated to convert a Decl* to a GlobalDecl: /// PredefinedExpr::ComputeName() in lib/AST/Expr.cpp. /// getParentOfLocalEntity() in lib/AST/ItaniumMangle.cpp /// ASTNameGenerator::Implementation::writeFuncOrVarName in lib/AST/Mangle.cpp /// class GlobalDecl { llvm::PointerIntPair Value; unsigned MultiVersionIndex = 0; void Init(const Decl *D) { assert(!isa(D) && "Use other ctor with ctor decls!"); assert(!isa(D) && "Use other ctor with dtor decls!"); assert(!D->hasAttr() && "Use other ctor with GPU kernels!"); Value.setPointer(D); } public: GlobalDecl() = default; GlobalDecl(const VarDecl *D) { Init(D);} GlobalDecl(const FunctionDecl *D, unsigned MVIndex = 0) : MultiVersionIndex(MVIndex) { if (!D->hasAttr()) { Init(D); return; } Value.setPointerAndInt(D, unsigned(getDefaultKernelReference(D))); } GlobalDecl(const FunctionDecl *D, KernelReferenceKind Kind) : Value(D, unsigned(Kind)) { assert(D->hasAttr() && "Decl is not a GPU kernel!"); } GlobalDecl(const NamedDecl *D) { Init(D); } GlobalDecl(const BlockDecl *D) { Init(D); } GlobalDecl(const CapturedDecl *D) { Init(D); } GlobalDecl(const ObjCMethodDecl *D) { Init(D); } GlobalDecl(const OMPDeclareReductionDecl *D) { Init(D); } GlobalDecl(const OMPDeclareMapperDecl *D) { Init(D); } GlobalDecl(const CXXConstructorDecl *D, CXXCtorType Type) : Value(D, Type) {} GlobalDecl(const CXXDestructorDecl *D, CXXDtorType Type) : Value(D, Type) {} GlobalDecl(const VarDecl *D, DynamicInitKind StubKind) : Value(D, unsigned(StubKind)) {} GlobalDecl getCanonicalDecl() const { GlobalDecl CanonGD; CanonGD.Value.setPointer(Value.getPointer()->getCanonicalDecl()); CanonGD.Value.setInt(Value.getInt()); CanonGD.MultiVersionIndex = MultiVersionIndex; return CanonGD; } const Decl *getDecl() const { return Value.getPointer(); } CXXCtorType getCtorType() const { assert(isa(getDecl()) && "Decl is not a ctor!"); return static_cast(Value.getInt()); } CXXDtorType getDtorType() const { assert(isa(getDecl()) && "Decl is not a dtor!"); return static_cast(Value.getInt()); } DynamicInitKind getDynamicInitKind() const { assert(isa(getDecl()) && cast(getDecl())->hasGlobalStorage() && "Decl is not a global variable!"); return static_cast(Value.getInt()); } unsigned getMultiVersionIndex() const { assert(isa( getDecl()) && !cast(getDecl())->hasAttr() && !isa(getDecl()) && !isa(getDecl()) && "Decl is not a plain FunctionDecl!"); return MultiVersionIndex; } KernelReferenceKind getKernelReferenceKind() const { assert(((isa(getDecl()) && cast(getDecl())->hasAttr()) || (isa(getDecl()) && cast(getDecl()) ->getTemplatedDecl() ->hasAttr())) && "Decl is not a GPU kernel!"); return static_cast(Value.getInt()); } friend bool operator==(const GlobalDecl &LHS, const GlobalDecl &RHS) { return LHS.Value == RHS.Value && LHS.MultiVersionIndex == RHS.MultiVersionIndex; } bool operator!=(const GlobalDecl &Other) const { return !(*this == Other); } void *getAsOpaquePtr() const { return Value.getOpaqueValue(); } explicit operator bool() const { return getAsOpaquePtr(); } static GlobalDecl getFromOpaquePtr(void *P) { GlobalDecl GD; GD.Value.setFromOpaqueValue(P); return GD; } static KernelReferenceKind getDefaultKernelReference(const FunctionDecl *D) { return D->getLangOpts().CUDAIsDevice ? KernelReferenceKind::Kernel : KernelReferenceKind::Stub; } GlobalDecl getWithDecl(const Decl *D) { GlobalDecl Result(*this); Result.Value.setPointer(D); return Result; } GlobalDecl getWithCtorType(CXXCtorType Type) { assert(isa(getDecl())); GlobalDecl Result(*this); Result.Value.setInt(Type); return Result; } GlobalDecl getWithDtorType(CXXDtorType Type) { assert(isa(getDecl())); GlobalDecl Result(*this); Result.Value.setInt(Type); return Result; } GlobalDecl getWithMultiVersionIndex(unsigned Index) { assert(isa(getDecl()) && !cast(getDecl())->hasAttr() && !isa(getDecl()) && !isa(getDecl()) && "Decl is not a plain FunctionDecl!"); GlobalDecl Result(*this); Result.MultiVersionIndex = Index; return Result; } GlobalDecl getWithKernelReferenceKind(KernelReferenceKind Kind) { assert(isa(getDecl()) && cast(getDecl())->hasAttr() && "Decl is not a GPU kernel!"); GlobalDecl Result(*this); Result.Value.setInt(unsigned(Kind)); return Result; } }; } // namespace clang namespace llvm { template<> struct DenseMapInfo { static inline clang::GlobalDecl getEmptyKey() { return clang::GlobalDecl(); } static inline clang::GlobalDecl getTombstoneKey() { return clang::GlobalDecl:: getFromOpaquePtr(reinterpret_cast(-1)); } static unsigned getHashValue(clang::GlobalDecl GD) { return DenseMapInfo::getHashValue(GD.getAsOpaquePtr()); } static bool isEqual(clang::GlobalDecl LHS, clang::GlobalDecl RHS) { return LHS == RHS; } }; } // namespace llvm #endif // LLVM_CLANG_AST_GLOBALDECL_H