77 lines
2.8 KiB
C++
77 lines
2.8 KiB
C++
|
//===- DeclFriend.cpp - C++ Friend Declaration AST Node Implementation ----===//
|
||
|
//
|
||
|
// 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
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
//
|
||
|
// 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;
|
||
|
}
|