| //===- DeclFriend.cpp - C++ Friend Declaration AST Node Implementation ----===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file implements the AST classes related to C++ friend |
| // declarations. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "clang/AST/DeclFriend.h" |
| #include "clang/AST/Decl.h" |
| #include "clang/AST/DeclBase.h" |
| #include "clang/AST/DeclCXX.h" |
| #include "clang/AST/ASTContext.h" |
| #include "clang/AST/DeclTemplate.h" |
| #include "clang/Basic/LLVM.h" |
| #include "llvm/Support/Casting.h" |
| #include <cassert> |
| #include <cstddef> |
| |
| using namespace clang; |
| |
| void FriendDecl::anchor() {} |
| |
| FriendDecl *FriendDecl::getNextFriendSlowCase() { |
| return cast_or_null<FriendDecl>( |
| NextFriend.get(getASTContext().getExternalSource())); |
| } |
| |
| FriendDecl *FriendDecl::Create(ASTContext &C, DeclContext *DC, |
| SourceLocation L, |
| FriendUnion Friend, |
| SourceLocation FriendL, |
| ArrayRef<TemplateParameterList *> FriendTypeTPLists) { |
| #ifndef NDEBUG |
| if (Friend.is<NamedDecl *>()) { |
| const auto *D = Friend.get<NamedDecl*>(); |
| assert(isa<FunctionDecl>(D) || |
| isa<CXXRecordDecl>(D) || |
| isa<FunctionTemplateDecl>(D) || |
| isa<ClassTemplateDecl>(D)); |
| |
| // As a temporary hack, we permit template instantiation to point |
| // to the original declaration when instantiating members. |
| assert(D->getFriendObjectKind() || |
| (cast<CXXRecordDecl>(DC)->getTemplateSpecializationKind())); |
| // These template parameters are for friend types only. |
| assert(FriendTypeTPLists.empty()); |
| } |
| #endif |
| |
| std::size_t Extra = |
| FriendDecl::additionalSizeToAlloc<TemplateParameterList *>( |
| FriendTypeTPLists.size()); |
| auto *FD = new (C, DC, Extra) FriendDecl(DC, L, Friend, FriendL, |
| FriendTypeTPLists); |
| cast<CXXRecordDecl>(DC)->pushFriendDecl(FD); |
| return FD; |
| } |
| |
| FriendDecl *FriendDecl::CreateDeserialized(ASTContext &C, unsigned ID, |
| unsigned FriendTypeNumTPLists) { |
| std::size_t Extra = |
| additionalSizeToAlloc<TemplateParameterList *>(FriendTypeNumTPLists); |
| return new (C, ID, Extra) FriendDecl(EmptyShell(), FriendTypeNumTPLists); |
| } |
| |
| FriendDecl *CXXRecordDecl::getFirstFriend() const { |
| ExternalASTSource *Source = getParentASTContext().getExternalSource(); |
| Decl *First = data().FirstFriend.get(Source); |
| return First ? cast<FriendDecl>(First) : nullptr; |
| } |